How do I make 25 requests at a time with HTTP::Async in Perl?

前端 未结 2 1233
不知归路
不知归路 2021-02-09 15:38

I\'m doing a lot of HTTP requests and I chose HTTP::Async to do the job. I\'ve over 1000 requests to make, and if I simply do the following (see code below), a lot of requests t

相关标签:
2条回答
  • 2021-02-09 16:15

    You're close, you just need to combine the two approaches! :-)

    Untested, so think of it as pseudo code. In particular I am not sure if total_count is the right method to use, the documentation doesn't say. You could also just have an $active_requests counter that you ++ when adding a request and -- when you get a response.

    while (1) {
    
       # if there aren't already 25 requests "active", then add more
       while (@urls and $async->total_count < 25) {
           my $url = shift @urls;
           $async->add( ... );
       }
    
       # deal with any finished requests right away, we wait for a
       # second just so we don't spin in the main loop too fast.
       while (my $response = $async->wait_for_next_response(1)) {
          # use $response
       }
    
       # finish the main loop when there's no more work
       last unless ($async->total_count or @urls);
    
    }
    
    0 讨论(0)
  • 2021-02-09 16:29

    If you can't call wait_for_next_response fast enough because you're in the middle of executing other code, the simplest solution is to make the code interruptable by moving it to a separate thread of execution. But if you're going to start using threads, why use HTTP::Async?

    use threads;
    use Thread::Queue::Any 1.03;
    
    use constant NUM_WORKERS => 25;
    
    my $req_q = Thread::Queue::Any->new();
    my $res_q = Thread::Queue::Any->new();
    
    my @workers;
    for (1..NUM_WORKERS) {
       push @workers, async {
          my $ua = LWP::UserAgent->new();
          while (my $req = $req_q->dequeue()) {
             $res_q->enqueue( $ua->request($req) );
          }
       };    
    }
    
    for my $url (@urls) {
       $req_q->enqueue( HTTP::Request->new( GET => $url ) );
    }
    
    $req_q->enqueue(undef) for @workers;
    
    for (1..@urls) {
       my $res = $res_q->dequeue();
       ...
    }
    
    $_->join() for @workers;
    
    0 讨论(0)
提交回复
热议问题