PHP源码阅读笔记:nl2br, ltrim, rtrim, trim函数
string nl2br ( string string )
Returns string with ‘
‘ inserted before all newlines.
在代码中有注释如下:
/* it is really faster to scan twice and allocate mem once insted scanning once
and constantly reallocing */
程序先计算需要替换的个数,然后一次性计算需要分配的内存大小。从而减少了每次替换都重新分配内存的开销。
由此可见PHP源码的作者的程序优化上下了不少功夫。
源码摘抄如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
|
str = Z_STRVAL_PP(zstr); // 字符串开始位置
end = str + Z_STRLEN_PP(zstr); // 字符串结束地址
/* it is really faster to scan twice and allocate mem once insted scanning once
and constantly reallocing */
while (str < end) { // 计算需要替换的位置个数
if (*str == 'r') {
if (*(str+1) == 'n') {
str++;
}
repl_cnt++;
} else if (*str == 'n') {
if (*(str+1) == 'r') {
str++;
}
repl_cnt++;
}
str++;
}
if (repl_cnt == 0) { // 如果没有可替换的字符串,直接返回
RETURN_STRINGL(Z_STRVAL_PP(zstr), Z_STRLEN_PP(zstr), 1);
}
// 给新生成的字符串分配内存
new_length = Z_STRLEN_PP(zstr) + repl_cnt * (sizeof("<br />") - 1);
tmp = target = emalloc(new_length + 1);
str = Z_STRVAL_PP(zstr);
while (str < end) {
switch (*str) {
case 'r': // 没有break,直接转下个case
case 'n':
*target++ = '<';
*target++ = 'b';
*target++ = 'r';
*target++ = ' ';
*target++ = '/';
*target++ = '>';
if ((*str == 'r' && *(str+1) == 'n') || (*str == 'n' && *(str+1) == 'r')) {
*target++ = *str++;
}
/* lack of a break; is intentional */
default:
*target++ = *str;
}
str++;
}
*target = ' |