PHP+shell实现
多线程的方法 先写个简单的php代码,这里为了让脚本执行时间更长,方便看效果,sleep一下,呵呵!先看下test.php的代码:ls PHP代码: ?php for ($i=0$i10$i++) { echo $i sleep(10) } ? 在看下shell脚本的代码,非常简单 #!/bin/bash for i in 1 2 3 4 5 6 7 8 9 10 do /usr/bin/php -q /var/www/html/test.php done 注意到在请求php代码的那行有一个符号吗,这个是关键,不加的话是不能进行多线程的,表示讲服务推送到后台执行,因此,在 shell的每次的循环中不必等php的代码全部执行完在请求下一个
文件,而是同时进行的磨裂,这样就实现了多线程,下面运行下shell看下效果,这里你将 看到10个test.php
进程再跑,再利用linux的定时器,定时请求这个shell,在处理一些需要多线程的任务,例如,批量下载时,非常好用! php中用WEB服务器实现多线程 假设我们现在运行的是a.php这个文件. 但是我在程序中又请求WEB服务器运行另一个b.php,那么这两个文件将是同时执行的.(PS: 一个链接请求凳州发送之后, WEB服务器就会执行它, 而不管客户端是否已经退出) 有些时候, 我们想运行的不是另一个文件, 而是本文件中的一部分代码.该怎么办呢? 其实可是通过参数来控制a.php来运行哪一段程序. //a.php,b.php PHP代码:-------------------------------------------------------------------------------- ?php function runThread() { $fp = fsockopen('localhost', 80, $errno, $errmsg) fputs($fp, GET /b.php?act=brnrn)//这里的第二个参数是HTTP协议中规定的请求头 //不明白的请看RFC中的定义 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() ? -------------------------------------------------------------------------------- 打开result_a.log 和 result_b.log 比较一下两个文件的中访问的时间. 大家会发现, 这两个的确是在不同线程中运行的.有些时间完全一样. 上面只是一个简单的例子, 大家可以改进成其它形式. 既然PHP中也能多线程了, 那么问题也来了, 那就是同步的问题. 我们知道瞎粗闭 PHP本身是不支持多线程的. 所以更不会有什么像Java 中synchronize的方法了. 那我们该如何做呢. 1. 尽量不访问同一个资源. 以避免冲突. 但是可以同时像数据库 *** 作. 因为数据库是支持并发 *** 作的. 所以在多线程的PHP中不要向同一个文件中写入数据. 如果必须要写的话, 用别的方法进行同步.. 如调用 flock对文件进行加锁等. 或建立临时文件并在另外的线程中等待这个文件的消失 while(file_exits('xxx'))这样就等于这个临时文件存在时, 表示其实线程正在 *** 作 如果没有了这个文件, 说明其它线程已经释放了这个. 2. 尽量不要从runThread在执行fputs后取这个socket中读取数据. 因为要实现多线程, 需要的用非阻塞模式. 即在像fgets这样的函数时立即返回.. 所以读写数据就会出问题. 如果使用阻塞模式的话, 程序就不算是多线程了. 他要等上面的返回才执行下面的程序. 所以如果需要交换数据最后利用外面文件或数据中完成. 实在想要的话就用socket_set_nonblock($fp) 来实现. 说了这么多, 倒底这个有没有实际的意义呢? 在什么时候需要这种用这种方法呢 ? 答案是肯定的. 大家知道. 在一个不断读取网络资源的应用中, 网络的速度是瓶颈. 如果采多这种形式就可以同时以多个线程对不同的页面进行读取. 本人做的一个能从8848、soaso这些商城网站搜索信息的程序。还有一个从阿里巴巴网站上读取商业信息和公司目录的程序也用到了此技术。 因为这两个程序都是要不断的链接它们的服务器读取信息并保存到数据库。 利用此技术正好消除了在等待响应时的瓶颈。 php模拟实现多线程的三种方法 PHP语言本身是不支持多线程的. 总结了一下网上关于PHP模拟多线程的方法, 总的来说, 都是利用了PHP的好伙伴们本身所具有的多线程能力. PHP的好伙伴指的就是LINUX和APACHE啦, LAMP嘛. 另外, 既然是模拟的, 就不是真正的多线程. 其实只是多进程. 进程和线程是两个不同的概念. 好了, 以下方法都是从网上找来的. 1. 利用LINUX *** 作系统 ?php for ($i=0$i10$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()flock的选项-n等同于--nonblock,表示谨知非阻塞模式。man flock可以看到详细说明:
Fail (with an exit code of 1) rather than wait if the lock cannot be immediately acquired.
翻译过来就是,若不能立即获得文件锁,就直接出错返回1而不是继续等待(默认flock是会继续等待直到获得文件锁)。
3就是fd(即文件描述符)祥宏消。
文件锁只是用于防止多个程序同时访问一个文件的冲突情况。
Shell自动分配fd的话,原本就不会使用已被占用的fd,而是自动递增。
这里显式指定绝毁fd为3(3<>mylockfile),是为了flock有 *** 作的对象。只有unlock后文件才能被其他程序访问,这样就可以保证文件访问不冲突。
并发访问文桥困件,根本不需要担心啊,直接弄就可以敏扒念了
如果是并发此基写文件,那就得用锁机制flock了,http://blog.uouo123.com/post/714.html
评论列表(0条)