PHP语言本身是不支持多线程的. 总结了一下网上关于PHP模拟多线程的方法, 总的来说, 都是利用了PHP的好伙伴们本身所具有的多线程能力. PHP的好伙伴指的就是LINUX和APACHE啦, LAMP嘛.
另外, 既然是模拟的, 就不是真正的多线程. 其实只是多进程. 进程和线程是两个不同的概念. 好了, 以下方法都是从网上找来的.
1. 利用LINUX *** 作系统
<?php
for ($i=0$i<10$i++) {
echo $i
sleep(5)
}
?>
上面存成test.php, 然后写一段SHELL代码
#!/bin/bash
for i in 1 2 3 4 5 6 7 8 9 10
do
php -q test.php &
done
2. 利用fork子进程(其实同样是利用LINUX *** 作系统)
<?php
declare(ticks=1)
$bWaitFlag = FALSE/// 是否等待进程结束
$intNum = 10 /// 进程总数
$pids = array() /// 进程PID数组
echo ("Startn")
for($i = 0$i <$intNum$i++) {
$pids[$i] = pcntl_fork()/// 产生子进程,而且从当前行之下开试运行代码,而且不继承父进程的数据信息
if(!$pids[$i]) {
// 子进程进程代码段_Start
$str=""
sleep(5+$i)
for ($j=0$j<$i$j++) {$str.="*"}
echo "$i ->" . time() . " $str n"
exit()
// 子进程进程代码段_End
}
}
if ($bWaitFlag)
{
for($i = 0$i <$intNum$i++) {
pcntl_waitpid($pids[$i], $status, WUNTRACED)
echo "wait $i ->" . time() . "n"
}
}
echo ("Endn")
?>
3. 利用WEB SERVER, PHP不支持多线程, APACHE可是支持的, 呵呵.
假设我们现在运行的是a.php这个文档. 但是我在程式中又请求WEB服务器运行另一个b.php
那么这两个文档将是同时执行的.
<?php
function runThread()
{
$fp = fsockopen('localhost', 80, $errno, $errmsg)
fputs($fp, "GET /a.php?act=brnrn")
fclose($fp)
}
function a()
{
$fp = fopen('result_a.log', 'w')
fputs($fp, 'Set in ' . Date('h:i:s', time()) . (double)microtime() . "rn")
fclose($fp)
}
function b()
{
$fp = fopen('result_b.log', 'w')
fputs($fp, 'Set in ' . Date('h:i:s', time()) . (double)microtime() . "rn")
fclose($fp)
}
if(!isset($_GET['act'])) $_GET['act'] = 'a'
if($_GET['act'] == 'a')
{
runThread()
a()
}
else if($_GET['act'] == 'b') b()
?>
当然啦,也可以把需要多线程处理的部分交给JAVA去处理, 然后在PHP里调用, 哈哈.
<?php
system('java multiThread.java')
?>
利用WEB服务器本身的多线程来处理,从WEB服务器多次调用我们需要实现多线程的程序。
PHP中也能多线程了,那么问题也来了,那就是同步的问题。北京电脑培训知道PHP本身是不支持多线程的,所以更不会有什么像Java中synchronize的方法了。那我们该如何做呢?
1.尽量不访问同一个资源。以避免冲突。但是可以同时像数据库 *** 作。因为数据库是支持并发 *** 作的。所以在多线程的PHP中不要向同一个文件中写入数据。如果必须要写的话,用别的方法进行同步。如调用flock对文件进行加锁等。或建立临时文件,并在另外的线程中等待这个文件的消失while(file_exits('xxx'))这样就等于这个临时文件存在时,表示其实线程正在 *** 作。如果没有了这个文件,说明其它线程已经释放了这个。
2.尽量不要从runThread在执行fputs后取这个socket中读取数据。因为要实现多线程,需要的用非阻塞模式。即在像fgets这样的函数时立即返回。。所以读写数据就会出问题。如果使用阻塞模式的话,程序就不算是多线程了。他要等上面的返回才执行下面的程序。所以如果需要交换数据最后利用外面文件或数据中完成。实在想要的话就用socket_set_nonblock($fp)来实现。
说了这么多,倒底这个有没有实际的意义呢?在什么时候需要这种用这种方法呢?
答案是肯定的。大家知道。在一个不断读取网络资源的应用中,网络的速度是瓶颈。如果采多这种形式就可以同时以多个线程对不同的页面进行读取。
1、首先需要准备的应用程序包。nginx:nginx/Windows-1.0.4php:php-5.2.16-nts-Win32-VC6-x86.zip (nginx下php是以FastCGI的方式运行,所以我们下载非线程安全也就是nts的php包)
(还会用到)RunHiddenConsole:RunHiddenConsole.zip
2、安装与配置。
1)php的安装与配置。
直接解压下载好的php包,到D盘wnmp目录(D:\wnmp),这里把解压出来的文件夹重命名成php5。进入文件夹修改php.ini-recommended文件为php.ini,并用Editplus或者Notepad++打开来。找到
extension_dir = "./ext"
更改为
extension_dir = "D:/wnmp/php5/ext"
往下看,再找到
extension=php_mysql.dll
extension=php_mysqli.dll
前面指定了php的ext路径后,只要把需要的扩展包前面所对应的“”去掉,就可以了。这里打开php_mysql.dll和php_mysqli.dll,让php支持mysql。当然不要忘掉很重要的一步就是,把php5目录下的libmysql.dll文件复制到C:\Windows目录下,也可以在系统变量里面指定路径,当然这里我选择了更为方便的方法^_^。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)