使用PHP并行下载页面

前端之家收集整理的这篇文章主要介绍了使用PHP并行下载页面前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我必须废弃一个网站,我需要获取多个URL,然后逐个处理它们.目前的过程有点像这样.

我从此页面获取基本URL并获取所有辅助URL,然后为每个辅助URL我获取该URL,处理找到的页面,下载一些照片(这需要相当长的时间)并将此数据存储到数据库,然后获取下一个URL并重复这个过程.

在这个过程中,我认为我在每次迭代开始时浪费一些时间来获取辅助URL.所以我试图在处理第一次迭代时并行获取下一个URL.

我想到的解决方案是,从主进程调用PHP脚本,比如下载器,它将下载所有URL(使用curl_multi或wget)并将它们存储在某个数据库中.

我的问题是

>如何异步调用这样的downloder,我不希望我的主脚本等到downloder完成.
>存储下载数据的任何位置,例如共享内存.当然,除了数据库.
>存储和检索时数据是否有可能被破坏,如何避免这种情况?
>另外,如果有人有更好的计划,请指导我.

当我听到有人使用curl_multi_exec时,通常会发现他们只是用100个网址加载它,然后在完成后等待,然后全部处理它们,然后用接下来的100个网址重新开始…怪我,我是这样做,但后来我发现有可能在某些事情仍在进行时删除/添加curl_multi的句柄,并且它确实节省了大量时间,特别是如果你重用已经打开的连接.我写了一个小型库来处理带回调的请求队列;我当然不是在这里发布完整版本(“小”仍然是相当多的代码),但这里有一个简化版本,主要给你一般的想法:
  1. public function launch() {
  2. $channels = $freeChannels = array_fill(0,$this->maxConnections,NULL);
  3. $activeJobs = array();
  4. $running = 0;
  5. do {
  6. // pick jobs for free channels:
  7. while ( !(empty($freeChannels) || empty($this->jobQueue)) ) {
  8. // take free channel,(re)init curl handle and let
  9. // queued object set options
  10. $chId = key($freeChannels);
  11. if (empty($channels[$chId])) {
  12. $channels[$chId] = curl_init();
  13. }
  14. $job = array_pop($this->jobQueue);
  15. $job->init($channels[$chId]);
  16. curl_multi_add_handle($this->master,$channels[$chId]);
  17. $activeJobs[$chId] = $job;
  18. unset($freeChannels[$chId]);
  19. }
  20. $pending = count($activeJobs);
  21.  
  22. // launch them:
  23. if ($pending > 0) {
  24. while(($mrc = curl_multi_exec($this->master,$running)) == CURLM_CALL_MULTI_PERFORM);
  25. // poke it while it wants
  26. curl_multi_select($this->master);
  27. // wait for some activity,don't eat cpu
  28. while ($running < $pending && ($info = curl_multi_info_read($this->master))) {
  29. // some connection(s) finished,locate that job and run response handler:
  30. $pending--;
  31. $chId = array_search($info['handle'],$channels);
  32. $content = curl_multi_getcontent($channels[$chId]);
  33. curl_multi_remove_handle($this->master,$channels[$chId]);
  34. $freeChannels[$chId] = NULL;
  35. // free up this channel
  36. if ( !array_key_exists($chId,$activeJobs) ) {
  37. // impossible,but...
  38. continue;
  39. }
  40. $activeJobs[$chId]->onComplete($content);
  41. unset($activeJobs[$chId]);
  42. }
  43. }
  44. } while ( ($running > 0 && $mrc == CURLM_OK) || !empty($this->jobQueue) );
  45. }

在我的版本中,$jobs实际上是单独的类,而不是控制器或模型的实例.他们只处理设置cURL选项,解析响应并调用给定的回调onComplete.
使用此结构,只要池中的某些内容完成,新请求就会立即启动.

当然,如果不只是检索需要时间而且处理也不会真正拯救你……而且它并不是真正的并行处理.但我仍然希望它有所帮助.

猜你在找的PHP相关文章