admin管理员组

文章数量:1348081

I can parallelize execution using Hyper and it works:

$ raku -e 'race for (^8).race(batch => 1, degree => 4) {sleep rand; .say}'
0
3
5
2
1
7
4
6

But how can I add gather/take behavior to such loop?

In this approach take won't detect that it is wrapped in gather context:

$ raku -e '
    my @x = gather race for (^8).race(batch => 1, degree => 4) {
        sleep rand;
        take $_;
    };
    @x.say
'

Died at:
    take without gather

In this approach gather will consume HyperSeq without actually "hypering" over it:

$ raku -e '
    my @x = race gather for (^8).race(batch => 1, degree => 4) {
        sleep rand;
        take $_;
    };
    @x.say
'

[0 1 2 3 4 5 6 7]

I can parallelize execution using Hyper and it works:

$ raku -e 'race for (^8).race(batch => 1, degree => 4) {sleep rand; .say}'
0
3
5
2
1
7
4
6

But how can I add gather/take behavior to such loop?

In this approach take won't detect that it is wrapped in gather context:

$ raku -e '
    my @x = gather race for (^8).race(batch => 1, degree => 4) {
        sleep rand;
        take $_;
    };
    @x.say
'

Died at:
    take without gather

In this approach gather will consume HyperSeq without actually "hypering" over it:

$ raku -e '
    my @x = race gather for (^8).race(batch => 1, degree => 4) {
        sleep rand;
        take $_;
    };
    @x.say
'

[0 1 2 3 4 5 6 7]
Share Improve this question asked 2 days ago Pawel Pabian bbkrPawel Pabian bbkr 1,2616 silver badges14 bronze badges
Add a comment  | 

3 Answers 3

Reset to default 3

What you are asking for, under the hood, is a parallelized data structure (in this case the container for the results of gather/take) that protects write operations from multiple threads. While raku syntax does not prevent an implementation like this, it does not aim to make standard data structures atomic since that adds a substantial overhead and potential for deadlocks. In this kind of requirement, raku has great concurrency syntax that will help you to map/reduce your calculations across multiple processes.

See the note at the bottom of the gather take docs which mentions a similar issue with react/whenever, namely that there is no handler for the control exception on the new threads. For example:

$ raku -e 'say gather { await start { take 42 } };'
An operation first awaited:
  in block <unit> at -e line 1

Died with the exception:
    take without gather
      in block  at -e line 1

I can parallelize execution using race and it works

But only on Tuesdays. I'm guessing you already know this first bit of my answer, but the compiler is allowed to decide whether to parallelize. To slightly but fairly misquote the relevant design doc with my added emphasis:

hyper / race request (but do not require) parallel evaluation.

----

But how can I add gather/take behavior to such loop?

Quoting the design doc again:

You can think of race as a gather with a 'try take start {...}' on parallel computation.

That is to say, using race means you already automatically get gather/take behavior for free, without actually having to use gather/take.

----

gather race ... dies with error message take without gather

race gather ... "works" but doesn't parallelize

I'm not sure what to make of those two scenarios.

----

In his answer to What concurrency mechanisms are provided by Raku ... ?, Jonathan Worthington mentions this about gather/take:

  • The gather/take mechanism for lazily producing values. This is seen in various languages as generator functions; in Raku, however, there is not a single stack frame limit, so they are coroutine powerful. (This isn't of in any context involving asynchronous programming, but it's still meets the definition of concurrency).

I'm unsure about the parenthetical. I can't parse "This isn't of in any context involving", and the latter part of the sentence seems ambiguous.

----

One thing that jumped out at me is that gather/take defaults to being lazy. Use of race should override that, and make it eager, but I thought to myself that Rakudo could plausibly have a bug that left it lazy, and wondered if lazy behavior was leading to the behavior seen in your race gather ... code. However, I tried inserting an explicit eager where I thought it might be needed and nothing I tried made a difference.

本文标签: rakuUse gathertake with raceStack Overflow