« 上一篇 下一篇 »

3 个PHP 知识总结:Memcache、缓存和正则​

很多时候一个完整的系统可能运行在多个服务器上,如果这多个服务器之间需要共享session的话,那么PHP默认的files保存session的方式就无能为力了。这时我们可以考虑使用Memcache来接管session的保存与读取工作。

方法I: 在 php.ini 中全局设置
php.ini代码

session.save_handler = memcache
session.save_path = "tcp://127.0.0.1:11211"

方法II: 某个目录下的 .htaccess
php.ini代码

php_value session.save_handler "memcache"
php_value session.save_path  "tcp://127.0.0.1:11211"

方法III: 再或者在某个一个应用中
PHP代码

ini_set("session.save_handler", "memcache");
ini_set("session.save_path", "tcp://127.0.0.1:11211");

使用多个 memcached server 时用逗号”,”隔开,并且和 Memcache::addServer() 文档中说明的一样,可以带额外的参数”persistent”、”weight”、”timeout”、”retry_interval” 等等,类似这样的:”tcp://host1:port1?persistent=1&weight=2,tcp://host2:port2″ 。

如果安装的PECL是memcached(依赖libmemcached库的那个扩展),则配置应为
PHP代码

ini_set("session.save_handler", "memcached"); // 是memcached不是memcache
ini_set("session.save_path", "127.0.0.1:11211"); // 不要tcp

 

PHP缓冲区总结

分类:PHP 时间:2014年12月29日

在有些循环执行的语句,浏览器没有预期的输出,到底是什么问题呢?本篇文章主要讲解下PHP缓冲区。对于PHP来说,每一次像 echo输出等操作,先是写入到了 php buffer 里,在脚本执行完毕或者执行了强制输出缓存操作,数据才会在浏览器上显示。

我们先来看一段代码。

for ($i=10; $i>0; $i--){
    echo $i;
    flush();
    sleep(1);
}

按照php手册里的说法,该函数将当前为止程序的所有输出发送到用户的浏览器。上面的这段代码,应该隔一秒钟输出一次$i。但是实际中却不一定是这样。有可能是等了10秒钟后,所有的输出同时呈现出来。

好,我们来改一下这段代码,改成

ob_end_clean();//修改部分
for ($i=10; $i>0; $i--){
    echo $i;
    flush();
    sleep(1);
}

嘿,加了这一句ob_end_clean();,居然就OK了。实际上,我们把ob_end_clean()换成ob_end_flush()也一样OK。

我再来改一改。

for ($i=10; $i>0; $i--){
    echo $i;
    ob_flush();//修改部分
    flush();
    sleep(1);
}

打开php.ini,搜索output_buffering,我们会看到类似这样的设置 output_buffering = 4096。正如它的名字output_buffering一样,这个设置的作用就是把输出缓冲一下,缓冲大小为4096bytes.运行一下,是不是发现$i也隔一秒输出一次了?这是为什么呢?
别急,我们来看看php.ini。

在我们的第一段代码里,之所以没有按预期的输出,正是因为这个output_buffering把那些输出都缓冲了。没达到4096bytes或者脚本结束,输出是不会被发送出去的。

而第二段代码中的ob_end_clean()和ob_end_flush()的作用,就是终止缓冲。这样就不用等到有4096bytes的缓冲之后才被发送出去了。

第三段代码中,用了一句ob_flush(),它的作用就是把缓冲的数据发送出去,但是并不会终止缓冲,所以它必须在每次flush()前使用。

如果不想使用ob_end_clean(),ob_end_flush()和ob_flush(),我们就必须把php.ini里的output_buffering设得足够小,例如设为0。需要注意的是,如果你打算在脚本中使用ini_set(“output_buffering”,”0″)来设置,那么请停下来吧,这种方法是不行的。因为在脚本一开始的时候,缓冲设置就已经被载入,然后缓冲就开始了。

可能你会问了,既然ob_flush()是把缓冲的数据发送出去,那么为什么还需要用flush()???直接用下面这段代码不行吗??

for ($i=10; $i>0; $i--){
    echo $i;
    ob_flush();
    sleep(1);
}

那是不是flush()在这里就是不可缺少的呢?不是的,我们还有另外一种方法,使得当有数据输出的时候,马上被发送到浏览器。下面这两段代码就是不需要使用flush()了。(当你把output_buffering设为0的时候,连ob_flush()和ob_end_clean()都不需要了)请注意ob_flush()和flush()的区别。前者是把数据从PHP的缓冲中释放出来,后者是把不在缓冲中的或者说是被释放出来的数据发送到浏览器。所以当缓冲存在的时候,我们必须ob_flush()和flush()同时使用。

ob_implicit_flush(true);
for ($i=10; $i>0; $i--){
    echo $i;
    ob_flush(); #如果ob函数打开的情况下
    sleep(1);
}


ob_end_clean();
ob_implicit_flush(true);
for ($i=10; $i>0; $i--){
    echo $i;
    sleep(1);
}

以上所诉可能在某些浏览器中不成立。因为浏览器也有自己的规则。我是用Firefox1.5,IE6,opera8.5来测试的。其中opera就不能正常输出,因为它有一个规则,就是不遇到一个HTML标签,就绝对不输出,除非到脚本结束。而FireFox和IE还算比较正常的。请注意看上面的ob_implicit_flush(true),这个函数强制每当有输出的时候,即刻把输出发送到浏览器。这样就不需要每次输出(echo)后,都用flush()来发送到浏览器了。

最后附上一段非常有趣的代码,作者为PuTTY shell。在一个脚本周期里,每次输出,都会把前一次的输出覆盖掉。
以下代码只在firefox下可用,其他浏览器并不支持multipart/x-mixed-replace的Content-Type.

header('Content-type: multipart/x-mixed-replace;boundary=endofsection');
print "\n--endofsection\n";

$pmt = array("-", "\\", "|", "/" );
for( $i = 0; $i <10; $i ++ ){
    sleep(1);
    print "Content-type: text/plain\n\n";
    print "Part $i\t".$pmt[$i % 4];
    print "--endofsection\n";
    ob_flush();
    flush();
}
print "Content-type: text/plain\n\n";
print "The end\n";
print "--endofsection--\n";

PHP正则表达式的运用

分类:PHP 时间:2014年11月28日

正则表达式是PHP中一个非常重要的知识点,通常用来查找和替换字符串,验证用户输入的信息格式是否符合规范,如邮件格式、电话格式等等。还有采集器之类的软件中,正则也是必用不可!主要介绍PHP中几个常见的正则函数:preg_match()、preg_match_all()、preg_grep()、preg_split()。

1.preg_match()
该函数通常用于表单验证。可以按指定的正则表达式模式,对字符串进行搜索和匹配一次.该函数有两个必选参数,第一个参数需要提供用户按正则表达式语法编写的模式,第二个参数需要一个字符串。如果提供了第三个可选的数组参数matches,则可以拥有保存第一个参数中的子模式的各个部分的匹配结果。

2.preg_match_all()
该函数与preg_match()函数类似,不同的是函数preg_match()在第一次匹配之后就会停止搜索。而函数preg_match_all()则会一直搜索到指定字符串的结尾,可以获取到所有匹配到的结果。
该函数把所有可能的匹配结果放入第三个参数的数组中,并返回整个模式匹配的次数,如果出错则返回False。如果使用了第四个参数,会根据它指定的顺序将每次出现的匹配结果保存到第三个参数的数组中。第四个参数有两个预定义的值
a.PREG_PATTERN_ORDER: 它是preg_match_all() 函数的默认值,对结果排序使$matches[0]为全部模式匹配的数组,$matches[1] 为第一个括号中的子模式所匹配的字符串组成的数组,以此类推。
b.PREG_SET_ORDER: 对结果排序是$matches[0]为第一组匹配项的数组,$matches[1]为第二组匹配项的数组,以此类推。

3.preg_grep()
该函数与前两个函数不同的是匹配数组中的元素,返回与正则表达式匹配的数组单元。
该函数返回一个数组,其中包括了第二个参数数组中与给定的第一个参数模式相匹配的单元,对于输出数组的每个元素,进行依稀匹配。

4.preg_replace()
该函数可执行正则表达式的搜索和替换,是一个最强大的字符串替换处理函数。
该函数的三个参数,第一个参数匹配项,二个参数被替换的参数。第三个被搜多的参数。 如果指定第四个可选参数limit ,则仅替换limit个匹配,如果生了limit 或者其值为-1,则所有的匹配项都会被替换。

5.str_replace()
该函数是PHP提高的字符串处理函数,也可以实现字符串的替换工资。虽然没有正则表达式的替换函数功能强大,但一些简单字符串的替换要比preg_replace()函数的执行效率高。
该函数有单个必选参数,还有一个可选参数,第一个参数是目标对象,第二个参数是替换对象,第三个是被处理的字符串,该函数在第三个参数的字符串中,以区分大小写的方式搜索第一个参数提高的目标对象,并用第二个参数所提供的替换对象替换找到的所有实例。如果没有在第三个参数中搜索到目标对象,则被处理的字符串保持不变。第四个是一个可选参数,变量的引用,必须传入一个变量名称,用来保存替换的次数。
使用str_replace()的前两个参数不仅可以使用字符串,也可以使数组。

6.preg_split()
该函数使用了Perl兼容的正则表达式语法,可以按正则表达式的方法分割字符串。
该函数返回一个字符串数组,数组中元素包含通过第二个参数中的字符串,经第一个参数的正则表达式,作为匹配的边界分割的子串。如果指定了第三个字符串limit,则最多返回linmit个子串。而其中最后一个元素包含了被分割后剩余的所有部分。如果limit是-1,则意味着没有限制。 第四个参数是可选参数,
>> PREG_SPLIT_NO_EMPTY:如果设定了本标记,则preg_split()只返回非空的成分
>> PREG_SPLIT_DELIM_CAPTURE: 如果设定了本标记,定界符模式中的括号表达式也会被捕获并返回
>> PREG_SPLIT_OFFSET_CAPTURE: 如果设定了本标记,对每个出现的匹配结果也同时返回其附属的字符串偏移量。注意这改变了返回的数组的值,使其中的每个单元也是一个数组,其中第一项为匹配字符串,第二项为其在原字符串中的偏移量。

函数explode()
如果仅用某个特定的字符串进行分割,建议使用explode()函数,它不用去调用正则表达式引擎,因此速度是最快的。
该函数有三个参数,第一个参数提高一个分割字符或是字符串,第二个参数是被分割的字符串,如果提供第三个可选参数limit,则指定最多将字符串分割为多少个子串。该函数返回一个由被分割的子字符串组成的数组。

函数 implode()
与分割字符串相对应的是implode()函数,把数组中所有元素组合为一个字符串。函数join()为该函数的别名。
该函数有两个参数,第一个参数提高一个链接字符或字符串,第二个参数指定一个被链接的数组。