php中如何实现多进程
php中如何实现多进程
一、总结
一句话总结:
php多进程需要pcntl,posix扩展支持
可以通过 php - m 查看,没安装的话需要重新编译php,加上参数--enable-pcntl,posix一般默认会有
1、php多进程使用场景?
日常任务中,有时需要通过php脚本执行一些日志分析,队列处理等任务,当数据量比较大时,可以使用多进程来处理
2、php的cli模式是什么?
命令行模式:cli : Command Line Interface(命令行接口)
3、php多进程使用限制?
多进程实现只能在cli模式下,在web服务器环境下,会出现无法预期的结果,我测试报错:Call to undefined function: pcntl_fork()
4、php多进程核心函数?
pcntl_fork(创建子进程)、pcntl_wait(阻塞当前进程)
pcntl_fork: 一次调用两次返回,在父进程中返回子进程pid,在子进程中返回0,出错返回-1。pcntl_wait ( int &$status [, int $options ] ): 阻塞当前进程,直到任意一个子进程退出或收到一个结束当前进程的信号,注意是结束当前进程的信号,子进程结束发送的SIGCHLD不算。使用$status返回子进程的状态码,并可以指定第二个参数来说明是否以阻塞状态调用 阻塞方式调用的,函数返回值为子进程的pid,如果没有子进程返回值为-1; 非阻塞方式调用,函数还可以在有子进程在运行但没有结束的子进程时返回0。pcntl_waitpid ( int $pid , int &$status [, int $options ] ) 功能同pcntl_wait,区别为waitpid为等待指定pid的子进程。当pid为-1时pcntl_waitpid与pcntl_wait 一样。在pcntl_wait和pcntl_waitpid两个函数中的$status中存了子进程的状态信息。
5、php中一个始终保持固定个数的子进程在跑的例子?
根据需求使用pcntl_fork(创建子进程)、pcntl_wait(阻塞当前进程)等核心函数
<?php//最大的子进程数量 $maxChildPro = 8;//当前的子进程数量 $curChildPro = 0;//当子进程退出时,会触发该函数,当前子进程数-1 function sig_handler($sig) {global $curChildPro;switch ($sig) {case SIGCHLD:echo 'SIGCHLD', PHP_EOL;$curChildPro--;break;} }//配合pcntl_signal使用,简单的说,是为了让系统产生时间云,让信号捕捉函数能够捕捉到信号量 declare(ticks = 1);//注册子进程退出时调用的函数。SIGCHLD:在一个进程终止或者停止时,将SIGCHLD信号发送给其父进程。 pcntl_signal(SIGCHLD, "sig_handler");while (true) {$curChildPro++;$pid = pcntl_fork();if ($pid) { //父进程运行代码,达到上限时父进程阻塞等待任一子进程退出后while循环继续if ($curChildPro >= $maxChildPro) {pcntl_wait($status);}} else { //子进程运行代码$s = rand(2, 6);sleep($s);echo "child sleep $s second quit", PHP_EOL;exit;} }
二、php多进程总结
参考:php多进程总结
https://www.cnblogs.com/leezhxing/p/5223289.html">php多进程总结
场景:日常任务中,有时需要通过php脚本执行一些日志分析,队列处理等任务,当数据量比较大时,可以使用多进程来处理。
准备:php多进程需要pcntl,posix扩展支持,可以通过 php - m 查看,没安装的话需要重新编译php,加上参数--enable-pcntl,posix一般默认会有。
注意:
多进程实现只能在cli模式下,在web服务器环境下,会出现无法预期的结果,我测试报错:
Call to undefined function: pcntl_fork()
一个错误 pcntl_fork causing “errno=32 Broken pipe” #474 ,看https://github.com/phpredis/phpredis/issues/474
注意两点:如果是在循环中创建子进程,那么子进程中最后要exit,防止子进程进入循环。
子进程中的打开连接不能拷贝,使用的还是主进程的,需要用多例模式。
pcntl_fork:
一次调用两次返回,在父进程中返回子进程pid,在子进程中返回0,出错返回-1。
pcntl_wait ( int &$status [, int $options ] ):
阻塞当前进程,直到任意一个子进程退出或收到一个结束当前进程的信号,注意是结束当前进程的信号,子进程结束发送的SIGCHLD不算。使用$status返回子进程的状态码,并可以指定第二个参数来说明是否以阻塞状态调用
阻塞方式调用的,函数返回值为子进程的pid,如果没有子进程返回值为-1;
非阻塞方式调用,函数还可以在有子进程在运行但没有结束的子进程时返回0。
/** 确保这个函数只能运行在SHELL中 */ if (substr(php_sapi_name(), 0, 3) !== 'cli') {die("cli mode only"); }
#!/bin/bashfor((i=1;i<=8;i++)) do /usr/local/bin/php multiprocessTest.php & donewait
上面的shell程序,列了一个很简单的多进程程序,用一个for循环,实现了8进程并发来跑multiprocessTest.php这个程序。最后的wait语句,也可以使主进程,再等待所有进程都执行完后再往下执行的需求。
这个程序是没有问题的,很多现有的代码也都这样实现,但是这个程序的并发数是不可控的,即我们无法根据机器的核数去调度每一个进程的开关。
若我们的机器有8核或者更多,上面的程序是没有问题的,所有核都能充分利用,并且互相之间,没有争抢资源的情况出现。
但我们的机器要没有8核的话会是什么情况,同一时间运行的进程数多于核数,那么系统就会出现进程分配调度的问题,争抢资源也跟着相应而来,一个进程不能保证独立连续的执行,所有的进程运行会听从系统的调度,这样就会有更多的不确定因素出现。
<?php//最大的子进程数量 $maxChildPro = 8;//当前的子进程数量 $curChildPro = 0;//当子进程退出时,会触发该函数,当前子进程数-1 function sig_handler($sig) {global $curChildPro;switch ($sig) {case SIGCHLD:echo 'SIGCHLD', PHP_EOL;$curChildPro--;break;} }//配合pcntl_signal使用,简单的说,是为了让系统产生时间云,让信号捕捉函数能够捕捉到信号量 declare(ticks = 1);//注册子进程退出时调用的函数。SIGCHLD:在一个进程终止或者停止时,将SIGCHLD信号发送给其父进程。 pcntl_signal(SIGCHLD, "sig_handler");while (true) {$curChildPro++;$pid = pcntl_fork();if ($pid) { //父进程运行代码,达到上限时父进程阻塞等待任一子进程退出后while循环继续if ($curChildPro >= $maxChildPro) {pcntl_wait($status);}} else { //子进程运行代码$s = rand(2, 6);sleep($s);echo "child sleep $s second quit", PHP_EOL;exit;} }
<?php$childs = array();// Fork10个子进程 for ($i = 0; $i < 10; $i++) {$pid = pcntl_fork();if ($pid == -1)die('Could not fork');if ($pid) {echo "parent \n";$childs[] = $pid;} else { // Sleep $i+1 (s). 子进程可以得到$i参数sleep($i + 1);// 子进程需要exit,防止子进程也进入for循环exit();} }while (count($childs) > 0) {foreach ($childs as $key => $pid) {$res = pcntl_waitpid($pid, $status, WNOHANG);//-1代表error, 大于0代表子进程已退出,返回的是子进程的pid,非阻塞时0代表没取到退出子进程if ($res == -1 || $res > 0)unset($childs[$key]);}sleep(1); }
<?php function _fetchLog() {$password = $this->_getPassword();$online_log_path = NginxConf::getArchiveDir($this->_stat_day);$task_log_path = QFrameConfig::getConfig('LOG_PATH');$children = array();$success = true;foreach($this->_server_list as $host => $value){$local_dir = $this->_prepareLocalDir($host);$task_log = "$task_log_path/fetch_log.$host";$cmd = "sshpass -p $password rsync -av -e 'ssh -o StrictHostKeyChecking=no' $host:$online_log_path/* $local_dir >> $task_log 2>&1";$pid = pcntl_fork();if(-1 === $pid){LogSvc::log('stat_pv_by_citycode_error', 'could not fork');exit('could not fork');}else if(0 === $pid){system($cmd, $return_value);if(0 !== $return_value){LogSvc::log('stat_pv_by_citycode_error', "rsync $host error");}exit($return_value);}else{$children[$pid] = 1;}}while(!empty($children)){$pid = pcntl_waitpid(-1, $status, WNOHANG);if(0 === $pid){sleep(1);}else{if(0 !== pcntl_wexitstatus($status)){$success = false;}unset($children[$pid]);}}return $success; }
posix_kill(posix_getpid(), SIGHUP);
为自己生成SIGHUP信号
declare(ticks = 1); //php < 5.3
pcntl_signal()
函数仅仅是注册信号和它的处理方法,真正接收到信号并调用其处理方法的是pcntl_signal_dispatch()
函数必须在循环里调用,为了检测是否有新的信号等待dispatching。
pcntl_signal_dispatch()
declare(ticks = 1);
表示每执行一条低级指令,就检查一次信号,如果检测到注册的信号,就调用其信号处理器。
kill [PID]
命令,未加任何其他参数的话,程序会接收到一个SIGTERM信号。
<?php// 定义一个处理器,接收到SIGINT信号后只输出一行信息 function signalHandler($signal) {if ($signal == SIGINT) {echo 'SIGINT', PHP_EOL;} }// 信号注册:当接收到SIGINT信号时,调用signalHandler()函数 pcntl_signal(SIGINT, 'signalHandler');/*** PHP < 5.3 使用* 配合pcntl_signal使用,表示每执行一条低级指令,就检查一次信号,如果检测到注册的信号,就调用其信号处理器。*/ if (!function_exists("pcntl_signal_dispatch")) {declare(ticks=1); }while (true) {$s = sleep(10);echo $s, PHP_EOL; //信号会唤醒sleep,返回剩余的秒数。// do somethingfor ($i = 0; $i < 5; $i++) {echo $i . PHP_EOL;usleep(100000);}/*** PHP >= 5.3* 调用已安装的信号处理器* 必须在循环里调用,为了检测是否有新的信号等待dispatching。*/if (!function_exists("pcntl_signal_dispatch")) {pcntl_signal_dispatch();}}
<?phpdeclare(ticks = 1);function signal_handler($signal) {print "Caught SIGALRM\n";pcntl_alarm(5); }pcntl_signal(SIGALRM, "signal_handler", true); pcntl_alarm(5);for(;;) { }
<?php/*** 父进程通过pcntl_wait等待子进程退出* 子进程通过信号kill自己,也可以在父进程中发送kil信号结束子进程*///生成子进程 $pid = pcntl_fork(); if($pid == -1){die('could not fork'); }else{if($pid){$status = 0; //阻塞父进程,直到子进程结束,不适合需要长时间运行的脚本.//可使用pcntl_wait($status, WNOHANG)实现非阻塞式pcntl_wait($status);exit;}else{ //结束当前子进程,以防止生成僵尸进程if(function_exists("posix_kill")){posix_kill(getmypid(), SIGTERM);}else{system('kill -9'. getmypid());}exit;} }
2. 使用kill函数,php中是posix_kill,向原有的父进程发送空信号(kill(pid, 0))。使用这个方法对某个进程的存在性进行检查,而不会真的发送信号。所以,如果这个函数返回-1表示父进程已经退出。
(1) 父进程通过wait和waitpid等函数等待子进程结束,这会导致父进程挂起。它不适合子进程需要长时间运行的情况(会导致超时)。
执行wait()或waitpid()系统调用,则子进程在终止后会立即把它在进程表中的数据返回给父进程,此时系统会立即删除该进入点。在这种情形下就不会产生defunct进程。
(2) 如果父进程很忙,那么可以用signal函数为SIGCHLD安装handler。在子进程结束后,父进程会收到该信号,可以在handler中调用wait回收。
(3) 如果父进程不关心子进程什么时候结束,那么可以用signal(SIGCLD, SIG_IGN)或signal(SIGCHLD, SIG_IGN)通知内核,自己对子进程的结束不感兴趣,那么子进程结束后,内核会回收,并不再给父进程发送信号
(4)fork两次,父进程fork一个子进程,然后继续工作,子进程fork一个孙进程后退出,那么孙进程被init接管,孙进程结束后,init会回收。不过子进程的回收还要自己做。
ps -A -ostat,ppid,pid,cmd | grep -e '^[Zz]'
命令注解:-A 参数列出所有进程-o 自定义输出字段 我们设定显示字段为 stat(状态), ppid(进程父id), pid(进程id),cmd(命令)这四个参数
状态为 z或者Z 的进程为僵尸进程,所以我们使用grep抓取stat状态为zZ进程
运行结果如下:
这时,可以使用 kill -HUP 5255 杀掉这个进程。如果再次查看僵尸进程还存在,可以kill -HUP 5253(父进程)。
如果有多个僵尸进程,可以通过
ps -A -ostat,ppid,pid,cmd | grep -e '^[Zz]'|awk 'print{$2}'|xargs kill -9
处理。
多进程--进程间通信(IPC)
场景:日常任务中,有时需要通过php脚本执行一些日志分析,队列处理等任务,当数据量比较大时,可以使用多进程来处理。
准备:php多进程需要pcntl,posix扩展支持,可以通过 php - m 查看,没安装的话需要重新编译php,加上参数--enable-pcntl,posix一般默认会有。
注意:
多进程实现只能在cli模式下,在web服务器环境下,会出现无法预期的结果,我测试报错:
Call to undefined function: pcntl_fork()
一个错误 pcntl_fork causing “errno=32 Broken pipe” #474 ,看https://github.com/phpredis/phpredis/issues/474
注意两点:如果是在循环中创建子进程,那么子进程中最后要exit,防止子进程进入循环。
子进程中的打开连接不能拷贝,使用的还是主进程的,需要用多例模式。
pcntl_fork:
一次调用两次返回,在父进程中返回子进程pid,在子进程中返回0,出错返回-1。
pcntl_wait ( int &$status [, int $options ] ):
阻塞当前进程,直到任意一个子进程退出或收到一个结束当前进程的信号,注意是结束当前进程的信号,子进程结束发送的SIGCHLD不算。使用$status返回子进程的状态码,并可以指定第二个参数来说明是否以阻塞状态调用
阻塞方式调用的,函数返回值为子进程的pid,如果没有子进程返回值为-1;
非阻塞方式调用,函数还可以在有子进程在运行但没有结束的子进程时返回0。
/** 确保这个函数只能运行在SHELL中 */ if (substr(php_sapi_name(), 0, 3) !== 'cli') {die("cli mode only"); }
#!/bin/bashfor((i=1;i<=8;i++)) do /usr/local/bin/php multiprocessTest.php & donewait
上面的shell程序,列了一个很简单的多进程程序,用一个for循环,实现了8进程并发来跑multiprocessTest.php这个程序。最后的wait语句,也可以使主进程,再等待所有进程都执行完后再往下执行的需求。
这个程序是没有问题的,很多现有的代码也都这样实现,但是这个程序的并发数是不可控的,即我们无法根据机器的核数去调度每一个进程的开关。
若我们的机器有8核或者更多,上面的程序是没有问题的,所有核都能充分利用,并且互相之间,没有争抢资源的情况出现。
但我们的机器要没有8核的话会是什么情况,同一时间运行的进程数多于核数,那么系统就会出现进程分配调度的问题,争抢资源也跟着相应而来,一个进程不能保证独立连续的执行,所有的进程运行会听从系统的调度,这样就会有更多的不确定因素出现。
<?php//最大的子进程数量 $maxChildPro = 8;//当前的子进程数量 $curChildPro = 0;//当子进程退出时,会触发该函数,当前子进程数-1 function sig_handler($sig) {global $curChildPro;switch ($sig) {case SIGCHLD:echo 'SIGCHLD', PHP_EOL;$curChildPro--;break;} }//配合pcntl_signal使用,简单的说,是为了让系统产生时间云,让信号捕捉函数能够捕捉到信号量 declare(ticks = 1);//注册子进程退出时调用的函数。SIGCHLD:在一个进程终止或者停止时,将SIGCHLD信号发送给其父进程。 pcntl_signal(SIGCHLD, "sig_handler");while (true) {$curChildPro++;$pid = pcntl_fork();if ($pid) { //父进程运行代码,达到上限时父进程阻塞等待任一子进程退出后while循环继续if ($curChildPro >= $maxChildPro) {pcntl_wait($status);}} else { //子进程运行代码$s = rand(2, 6);sleep($s);echo "child sleep $s second quit", PHP_EOL;exit;} }
<?php$childs = array();// Fork10个子进程 for ($i = 0; $i < 10; $i++) {$pid = pcntl_fork();if ($pid == -1)die('Could not fork');if ($pid) {echo "parent \n";$childs[] = $pid;} else { // Sleep $i+1 (s). 子进程可以得到$i参数sleep($i + 1);// 子进程需要exit,防止子进程也进入for循环exit();} }while (count($childs) > 0) {foreach ($childs as $key => $pid) {$res = pcntl_waitpid($pid, $status, WNOHANG);//-1代表error, 大于0代表子进程已退出,返回的是子进程的pid,非阻塞时0代表没取到退出子进程if ($res == -1 || $res > 0)unset($childs[$key]);}sleep(1); }
<?php function _fetchLog() {$password = $this->_getPassword();$online_log_path = NginxConf::getArchiveDir($this->_stat_day);$task_log_path = QFrameConfig::getConfig('LOG_PATH');$children = array();$success = true;foreach($this->_server_list as $host => $value){$local_dir = $this->_prepareLocalDir($host);$task_log = "$task_log_path/fetch_log.$host";$cmd = "sshpass -p $password rsync -av -e 'ssh -o StrictHostKeyChecking=no' $host:$online_log_path/* $local_dir >> $task_log 2>&1";$pid = pcntl_fork();if(-1 === $pid){LogSvc::log('stat_pv_by_citycode_error', 'could not fork');exit('could not fork');}else if(0 === $pid){system($cmd, $return_value);if(0 !== $return_value){LogSvc::log('stat_pv_by_citycode_error', "rsync $host error");}exit($return_value);}else{$children[$pid] = 1;}}while(!empty($children)){$pid = pcntl_waitpid(-1, $status, WNOHANG);if(0 === $pid){sleep(1);}else{if(0 !== pcntl_wexitstatus($status)){$success = false;}unset($children[$pid]);}}return $success; }
posix_kill(posix_getpid(), SIGHUP);
为自己生成SIGHUP信号
declare(ticks = 1); //php < 5.3
pcntl_signal()
函数仅仅是注册信号和它的处理方法,真正接收到信号并调用其处理方法的是pcntl_signal_dispatch()
函数必须在循环里调用,为了检测是否有新的信号等待dispatching。
pcntl_signal_dispatch()
declare(ticks = 1);
表示每执行一条低级指令,就检查一次信号,如果检测到注册的信号,就调用其信号处理器。
kill [PID]
命令,未加任何其他参数的话,程序会接收到一个SIGTERM信号。
<?php// 定义一个处理器,接收到SIGINT信号后只输出一行信息 function signalHandler($signal) {if ($signal == SIGINT) {echo 'SIGINT', PHP_EOL;} }// 信号注册:当接收到SIGINT信号时,调用signalHandler()函数 pcntl_signal(SIGINT, 'signalHandler');/*** PHP < 5.3 使用* 配合pcntl_signal使用,表示每执行一条低级指令,就检查一次信号,如果检测到注册的信号,就调用其信号处理器。*/ if (!function_exists("pcntl_signal_dispatch")) {declare(ticks=1); }while (true) {$s = sleep(10);echo $s, PHP_EOL; //信号会唤醒sleep,返回剩余的秒数。// do somethingfor ($i = 0; $i < 5; $i++) {echo $i . PHP_EOL;usleep(100000);}/*** PHP >= 5.3* 调用已安装的信号处理器* 必须在循环里调用,为了检测是否有新的信号等待dispatching。*/if (!function_exists("pcntl_signal_dispatch")) {pcntl_signal_dispatch();}}
<?phpdeclare(ticks = 1);function signal_handler($signal) {print "Caught SIGALRM\n";pcntl_alarm(5); }pcntl_signal(SIGALRM, "signal_handler", true); pcntl_alarm(5);for(;;) { }
<?php/*** 父进程通过pcntl_wait等待子进程退出* 子进程通过信号kill自己,也可以在父进程中发送kil信号结束子进程*///生成子进程 $pid = pcntl_fork(); if($pid == -1){die('could not fork'); }else{if($pid){$status = 0; //阻塞父进程,直到子进程结束,不适合需要长时间运行的脚本.//可使用pcntl_wait($status, WNOHANG)实现非阻塞式pcntl_wait($status);exit;}else{ //结束当前子进程,以防止生成僵尸进程if(function_exists("posix_kill")){posix_kill(getmypid(), SIGTERM);}else{system('kill -9'. getmypid());}exit;} }
2. 使用kill函数,php中是posix_kill,向原有的父进程发送空信号(kill(pid, 0))。使用这个方法对某个进程的存在性进行检查,而不会真的发送信号。所以,如果这个函数返回-1表示父进程已经退出。
(1) 父进程通过wait和waitpid等函数等待子进程结束,这会导致父进程挂起。它不适合子进程需要长时间运行的情况(会导致超时)。
执行wait()或waitpid()系统调用,则子进程在终止后会立即把它在进程表中的数据返回给父进程,此时系统会立即删除该进入点。在这种情形下就不会产生defunct进程。
(2) 如果父进程很忙,那么可以用signal函数为SIGCHLD安装handler。在子进程结束后,父进程会收到该信号,可以在handler中调用wait回收。
(3) 如果父进程不关心子进程什么时候结束,那么可以用signal(SIGCLD, SIG_IGN)或signal(SIGCHLD, SIG_IGN)通知内核,自己对子进程的结束不感兴趣,那么子进程结束后,内核会回收,并不再给父进程发送信号
(4)fork两次,父进程fork一个子进程,然后继续工作,子进程fork一个孙进程后退出,那么孙进程被init接管,孙进程结束后,init会回收。不过子进程的回收还要自己做。
ps -A -ostat,ppid,pid,cmd | grep -e '^[Zz]'
命令注解:-A 参数列出所有进程-o 自定义输出字段 我们设定显示字段为 stat(状态), ppid(进程父id), pid(进程id),cmd(命令)这四个参数
状态为 z或者Z 的进程为僵尸进程,所以我们使用grep抓取stat状态为zZ进程
运行结果如下:
这时,可以使用 kill -HUP 5255 杀掉这个进程。如果再次查看僵尸进程还存在,可以kill -HUP 5253(父进程)。
如果有多个僵尸进程,可以通过
ps -A -ostat,ppid,pid,cmd | grep -e '^[Zz]'|awk 'print{$2}'|xargs kill -9
处理。
多进程--进程间通信(IPC)
多进程--守护进程
多进程--socket实现简单TCP server
<?phpstatic public function sendSDKMsg($version) {/*{{{*/if(!self::sendRandChance(self::EA_LAST_TIME_KEY.":".$version)) return false;$fp = @fsockopen( "udp://".self::UDP_HOST , self::UDP_PORT , $errno );if( !$fp ) return false;stream_set_timeout( $fp , 0 , 100 );stream_set_blocking( $fp , 0 );$sysinfo = posix_uname();$msg = $version." - ".$sysinfo['nodename']." - ".date('Y-m-d H:i:s',time());$res = fwrite( $fp , $msg );fclose($fp); }/*}}}*/static public function sendRandChance($key) {/*{{{*/$now = microtime(true);if(function_exists("eaccelerator_get")){$lastInserTime = eaccelerator_get($key);if(!$lastInserTime) $lastInserTime = 0;if( ($now - $lastInserTime) < self::SEND_INTERVAL ) return false;eaccelerator_put($key, $now);return true;}else if(function_exists("apc_fetch")){$lastInserTime = apc_fetch($key);if(!$lastInserTime) $lastInserTime = 0;if( ($now - $lastInserTime) < self::SEND_INTERVAL ) return false;apc_store($key, $now);return true;}$rand = rand(1,60);if((time()%60 == $rand) && rand(0,20) == 3){return true;}return false; }/*}}}*/
转载于:https://www.cnblogs.com/Renyi-Fan/p/10909584.html
php中如何实现多进程相关推荐
- c 语言 多进程,VC++中进程与多进程管理的方法详解
本文实例讲述了VC++中进程与多进程管理的方法,分享给大家供大家参考.具体方法分析如下: 摘要: 本文主要介绍了多任务管理中的多进程管理技术,对进程的互斥运行.子进程的创建与结束等作了较详细的阐述. ...
- VC++中进程与多进程管理的方法[转]
VC++中进程与多进程管理的方法 关键词: VC++6.0:进程:环境变量:子进程 进程 进程是当前操作系统下一个被加载到内存的.正在运行的应用程序的实例.每一个进程都是由内核对象和地址空间 ...
- Android中设置组件多进程
MultiProcessComponent 源码地址 github源码下载地址https://github.com/onlynight/MultiProcessComponent 概述 这个demo中 ...
- python多线程和多进程的区别_python中多线程与多进程的区别
线程的概念: 线程是操作系统中进行运算调度的最小单位.它被包含在进程之中,是进程中的实际运作单位.一条线程指的是进程中一个单一顺序的控制流,一个进程可以有多个线程,每条线程可以同时执行不同的任务.一个 ...
- python 多线程 廖雪峰_python中多线程与多进程中的数据共享问题
之前在写多线程与多进程的时候,因为一般情况下都是各自完成各自的任务,各个子线程或者各个子进程之前并没有太多的联系,如果需要通信的话我会使用队列或者数据库来完成,但是最近我在写一些多线程与多进程的代码时 ...
- io密集型和cpu密集型_一次说明白Python爬虫中多线程,多进程,异步IO编程
图/文:迷神 我们在Python爬虫中,重要的是讲究速度,如果有10万或者100万Url地址,写过爬虫的都会知道,那估计是非常慢的.我们的Python爬虫一般IO密集型业务,Python爬虫程序需要发 ...
- [转]Python中多线程与多进程中的数据共享问题!
之前在写多线程与多进程的时候,因为一般情况下都是各自完成各自的任务,各个子线程或者各个子进程之前并没有太多的联系,如果需要通信的话我会使用队列或者数据库来完成,但是最近我在写一些多线程与多进程的代码时 ...
- pysot 中的异步多进程切图
孪生网络算法对数据需求量较大,训练一般使用多个数据集.检测数据集普遍图大目标多,pysot 第一步准备数据,屏蔽了不同数据集的差异,也提升了数据加载效率.网络设定的模板大小为 127×127127\t ...
- java 中如何实现多进程_在Java中可以使用哪些方法来实现Java的多进程运行模式?...
在Java中我们可以使用两种方法来实现这种要求.最简单的方法就是通过Runtime中的exec方法执行java classname.如果执行成功,这个方法返回一个Process对象,如果执行失败,将抛 ...
最新文章
- Python中使用中文正则表达式匹配指定的中文字符串
- 广东计算机一级考试可以用计算器吗,你考试用不用计算器?
- 2018.10.26 NOIP模拟 瓶子 (dp/贪心)
- 【通知】3月第三周直播预告,模型精简前沿技术,人脸分析与编辑,图像风格化...
- Android中使用画笔和画布绘制一个矩形
- 使用Node.js部署智能合約(Smart Contract)
- python时间计算_日期天数差计算(Python)
- 【转载】如何知道自己适合做什么
- [cpp] 重载运算符规律总结
- hdu5187 奇怪题
- Dojo1.6新特性概览
- 一名Java开发人员装机必备软件
- 数据预处理:中英文印刷字体图片分类数据集生成
- php com adodb,php COM 连接ADODB.Connection数据库
- VFP全面控制EXCEL
- 显卡和cuda版本、cuda驱动对应关系
- 逻辑回归实现鸢尾花分类
- iis信息服务器win8,win8配置iis服务器
- BACnet/IP之BACnet4j学习java代码例子属性读写同网段跨主机02
- WPF自定义控件与样式(13)-自定义窗体Window 自适应内容大小消息框MessageBox
热门文章
- docker harbor 域名_docker 安装Harbor
- html保存按钮代码_如何防止Joomla编辑器删除HTML代码
- layui如何获取父节点的父节点_区块链如何运用merkle tree验证交易真实性
- Mysql事务控制语言
- 已解决 selenium.common.exceptions.NoSuchWindowException: Message: no such window
- php 取字符串的首字母,php取得字符串首字母的方法,php取得字符串_PHP教程
- axios post json_助你解析Axios原理之一:如何实现多种请求方式
- mysql.data已拥有为,MYSQL LOAD DATA INFILE忽略重复行(自动增量作为主键)
- java 各组件单击总数_java 获取面板上有多少个组件
- MAYA中average normal