<div dir="ltr">Yeah, I think I missed the task cancellation and the joining part.<div><br></div><div>When the mapConcurrent() thread is interrupted, it propagates the cancellation to the child threads. </div><div><br></div><div>And if any of the children threads (running the user-provided Function) receives that cancellation, it will still need to catch it and handle interruption.</div><div><br></div><div>That means the interruption isn't swallowed. If the user code wants to abort, they can always just throw an unchecked exception and catch it from the caller of mapConcurrent().</div><div><br></div><div>That should do for most of the IO type work.</div><div><br></div><div>This makes sense.</div><div><div><br></div><div><br></div></div></div><br><div class="gmail_quote gmail_quote_container"><div dir="ltr" class="gmail_attr">On Mon, May 26, 2025 at 12:30 AM Viktor Klang <<a href="mailto:viktor.klang@oracle.com">viktor.klang@oracle.com</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex"><div class="msg-2363802228770409313">
<div dir="ltr">
<div style="font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Yes, the updated Gatherers.mapConcurrent will continue under interruption until done, and then restore the interrupt status of the calling thread.</div>
<div style="font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
It cannot (reasonably) do anything else—if it were to throw InterruptedException, it couldn't since that is a checked exception and there's no tracking of checked exceptions throughout a Stream, and that is the specified behavior to handle thread interrupts
 (i.e. clear flag and throw InterruptedException, which is not possible in this case; or, make sure that interrupt status is maintained, which is possible in this case.</div>
<div style="font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
I have given it some more thought in the meantime, but I haven't arrived in something more satisfactory than that, yet.</div>
<div style="font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div id="m_-2363802228770409313Signature" style="color:inherit">
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Cheers,<br>
√</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b><br>
</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b>Viktor Klang</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Software Architect, Java Platform Group<br>
Oracle</div>
</div>
<div id="m_-2363802228770409313appendonsend"></div>
<hr style="display:inline-block;width:98%">
<div id="m_-2363802228770409313divRplyFwdMsg" dir="ltr"><font face="Calibri, sans-serif" style="font-size:11pt" color="#000000"><b>From:</b> Jige Yu <<a href="mailto:yujige@gmail.com" target="_blank">yujige@gmail.com</a>><br>
<b>Sent:</b> Monday, 26 May 2025 03:07<br>
<b>To:</b> Viktor Klang <<a href="mailto:viktor.klang@oracle.com" target="_blank">viktor.klang@oracle.com</a>><br>
<b>Cc:</b> <a href="mailto:core-libs-dev@openjdk.org" target="_blank">core-libs-dev@openjdk.org</a> <<a href="mailto:core-libs-dev@openjdk.org" target="_blank">core-libs-dev@openjdk.org</a>><br>
<b>Subject:</b> Re: [External] : Re: mapConcurrent() with InterruptedException</font>
<div> </div>
</div>
<div>
<div dir="ltr">Thanks Viktor!
<div><br>
</div>
<div>Sorry, I was giving myself more time to read the code, but then I lost track.</div>
<div><br>
</div>
<div>If I'm reading the code right, the current behavior is that if the current thread is interrupted, the mapConcurrent() would suppress the interruption until all elements are processed? (It will then restore the interruption bit)</div>
<div><br>
</div>
<div>Doesn't this break fail fast?</div>
<div><br>
</div>
<div>For example, I may have a mapConcurrent() called from within structured concurrency, and that fans out to two operations:</div>
<div>†</div>
<div>scope.fork(() -> doA());</div>
<div>scope.fork(() -> doB());  // calls mapConcurrent() internally</div>
<div>scope.join().throwIfFailed();</div>
<div><br>
</div>
<div>If doA() fails, doB() will be cancelled (thread interrupted).</div>
<div><br>
</div>
<div>If mapConcurrent() ignores the interruption, and if doB has a long list to process, it'll continue to consume system resources even when the caller no longer needs the results, no?</div>
<div><br>
</div>
</div>
<br>
<div>
<div dir="ltr">On Fri, Feb 7, 2025 at 2:16 AM Viktor Klang <<a href="mailto:viktor.klang@oracle.com" target="_blank">viktor.klang@oracle.com</a>> wrote:<br>
</div>
<blockquote style="margin:0px 0px 0px 0.8ex;border-left:1px solid rgb(204,204,204);padding-left:1ex">
<div>
<div dir="ltr">
<div style="font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
>Sorry, did the PR stop using Semaphore?</div>
<div style="font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
No, not that PR. See: <a href="https://urldefense.com/v3/__https://github.com/openjdk/jdk/commit/450636ae28b84ded083b6861c6cba85fbf87e16e__;!!ACWV5N9M2RV99hQ!OKfWrvJEDCkiHcT5b4a6gfeomZzZeq3jv6ir2vf2I5WWIhujtb-by_VuQFaAPDIuAA0qAwpTUjllXMPF$" id="m_-2363802228770409313x_m_-6675981207680419905OWAd7f42e96-d899-2e64-4fc9-cf9a90f88fdd" target="_blank">
https://github.com/openjdk/jdk/commit/450636ae28b84ded083b6861c6cba85fbf87e16e</a></div>
<div style="font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
The problem with interruption under parallel evaluation is that there is no general support for propagation of interruption in CountedCompleters. Adding support for such in (at least) GathererOp needs further study before contemplating making any changes to
 mapConcurrent()'s interruption policy.</div>
<div style="font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905Signature" style="color:inherit">
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Cheers,<br>
√</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b><br>
</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b>Viktor Klang</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Software Architect, Java Platform Group<br>
Oracle</div>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905appendonsend" style="color:inherit"></div>
<div style="font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<hr style="display:inline-block;width:98%">
<div id="m_-2363802228770409313x_m_-6675981207680419905divRplyFwdMsg" dir="ltr" style="color:inherit"><span style="font-family:Calibri,sans-serif;font-size:11pt;color:rgb(0,0,0)"><b>From:</b> Jige Yu <<a href="mailto:yujige@gmail.com" target="_blank">yujige@gmail.com</a>><br>
<b>Sent:</b> Thursday, 6 February 2025 17:04<br>
<b>To:</b> Viktor Klang <<a href="mailto:viktor.klang@oracle.com" target="_blank">viktor.klang@oracle.com</a>><br>
<b>Cc:</b> <a href="mailto:core-libs-dev@openjdk.org" target="_blank">core-libs-dev@openjdk.org</a> <<a href="mailto:core-libs-dev@openjdk.org" target="_blank">core-libs-dev@openjdk.org</a>><br>
<b>Subject:</b> Re: [External] : Re: mapConcurrent() with InterruptedException</span>
<div> </div>
</div>
<div style="direction:ltr">Sorry, did the PR stop using Semaphore?</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">I had naively thought that mapConcurrent() will keep a buffer of Future of all currently-running concurrent tasks (it can be a ConcurrentMap<TaskKey, Future> if we don't have to ensure FIFO).</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">Upon interruption, the main thread can call .cancel(true) on all pending Futures; optionally join with the VTs (if we need to block until all VTs exit); then propagate exception.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">Upon completion, each task just removes itself from the ConcurrentMap.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">Just in case it adds anything.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr"><br>
</div>
<br>
<div style="direction:ltr">On Thu, Feb 6, 2025 at 6:47 AM Viktor Klang <<a href="mailto:viktor.klang@oracle.com" id="m_-2363802228770409313x_m_-6675981207680419905OWAf2d116cc-daec-612f-54a5-d1ec25c12ff9" target="_blank">viktor.klang@oracle.com</a>> wrote:</div>
<blockquote style="margin:0px 0px 0px 0.8ex;padding-left:1ex;border-left:1px solid rgb(204,204,204)">
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
After some more investigation it seems tractable to propagate interruption of the caller in sequential mode, but parallel mode will require much bigger considerations.</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
I made a comment to that effect on the JBS issue: <a href="https://bugs.openjdk.org/browse/JDK-8349462?focusedId=14750017&page=com.atlassian.jira.plugin.system.issuetabpanels%3Acomment-tabpanel#comment-14750017" id="m_-2363802228770409313x_m_-6675981207680419905OWA1d803da0-b194-7fdd-c0ce-7045929b22ab" target="_blank">
https://bugs.openjdk.org/browse/JDK-8349462?focusedId=14750017&page=com.atlassian.jira.plugin.system.issuetabpanels%3Acomment-tabpanel#comment-14750017</a></div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184Signature" style="color:inherit">
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Cheers,<br>
√</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b><br>
</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b>Viktor Klang</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Software Architect, Java Platform Group<br>
Oracle</div>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184appendonsend" style="color:inherit">
</div>
<hr style="direction:ltr;display:inline-block;width:98%">
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184divRplyFwdMsg" dir="ltr" style="color:inherit">
<span style="font-family:Calibri,sans-serif;font-size:11pt;color:rgb(0,0,0)"><b>From:</b> Viktor Klang <<a href="mailto:viktor.klang@oracle.com" id="m_-2363802228770409313x_m_-6675981207680419905OWA2756a7aa-480a-a058-1eaf-d85246f8662d" target="_blank">viktor.klang@oracle.com</a>><br>
<b>Sent:</b> Thursday, 6 February 2025 11:51<br>
<b>To:</b> Jige Yu <<a href="mailto:yujige@gmail.com" id="m_-2363802228770409313x_m_-6675981207680419905OWA5469c14d-c33c-22db-2425-bc796d75a515" target="_blank">yujige@gmail.com</a>><br>
<b>Cc:</b> <a href="mailto:core-libs-dev@openjdk.org" id="m_-2363802228770409313x_m_-6675981207680419905OWAd72d835b-06b8-0fb1-44b5-1a76243ee34a" target="_blank">
core-libs-dev@openjdk.org</a> <<a href="mailto:core-libs-dev@openjdk.org" id="m_-2363802228770409313x_m_-6675981207680419905OWAc89193c1-31c4-3687-00e8-4f569fcfdf72" target="_blank">core-libs-dev@openjdk.org</a>><br>
<b>Subject:</b> Re: [External] : Re: mapConcurrent() with InterruptedException</span>
<div> </div>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
I think alignment in behavior between parallel Stream and mapConcurrent in terms of how interruptions are handled is a possible path forward.</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
I decided to close the PR for now as I realized my parallel Stream example had misled me regarding its exception throwing, so I'll need to go back and refine the solution.</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
It still seems solvable though.</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_Signature" style="color:inherit">
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Cheers,<br>
√</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b><br>
</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b>Viktor Klang</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Software Architect, Java Platform Group<br>
Oracle</div>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_appendonsend" style="color:inherit">
</div>
<hr style="direction:ltr;display:inline-block;width:98%">
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_divRplyFwdMsg" dir="ltr" style="color:inherit">
<span style="font-family:Calibri,sans-serif;font-size:11pt;color:rgb(0,0,0)"><b>From:</b> Jige Yu <<a href="mailto:yujige@gmail.com" id="m_-2363802228770409313x_m_-6675981207680419905OWAbc62567b-0679-f2bc-25ae-86db30bd0221" target="_blank">yujige@gmail.com</a>><br>
<b>Sent:</b> Wednesday, 5 February 2025 19:20<br>
<b>To:</b> Viktor Klang <<a href="mailto:viktor.klang@oracle.com" id="m_-2363802228770409313x_m_-6675981207680419905OWAfb1e01fa-e2c3-b610-64da-fcc21b151d8e" target="_blank">viktor.klang@oracle.com</a>><br>
<b>Cc:</b> <a href="mailto:core-libs-dev@openjdk.org" id="m_-2363802228770409313x_m_-6675981207680419905OWA809718a5-b5ce-a134-3348-ebdd7ee0c2c6" target="_blank">
core-libs-dev@openjdk.org</a> <<a href="mailto:core-libs-dev@openjdk.org" id="m_-2363802228770409313x_m_-6675981207680419905OWAe29873e6-ff7c-ddad-21d1-720d87975cdc" target="_blank">core-libs-dev@openjdk.org</a>><br>
<b>Subject:</b> Re: [External] : Re: mapConcurrent() with InterruptedException</span>
<div> </div>
</div>
<div style="direction:ltr">Oh good call!</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">I forgot to check what parallel streams do upon interruption (didn't think they do any blocking calls, but at least the main thread must block).</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">On Wed, Feb 5, 2025 at 8:18 AM Viktor Klang <<a href="mailto:viktor.klang@oracle.com" id="m_-2363802228770409313x_m_-6675981207680419905OWA158806a8-12c0-2a9e-ca0f-3b8795b6f7d2" target="_blank">viktor.klang@oracle.com</a>> wrote:</div>
<blockquote style="margin:0px 0px 0px 0.8ex;padding-left:1ex;border-left:1px solid rgb(204,204,204)">
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Hi Jige,<br>
<br>
I opened an issue to track the concern, and I have proposed a change which seems to align well with how parallel streams behave under caller thread interruption.</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
I've opened the following PR for review:</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<a href="https://urldefense.com/v3/__https://github.com/openjdk/jdk/pull/23467__;!!ACWV5N9M2RV99hQ!M4nB0WxHU_1jcbZoNGy3DD81Oefr7BuObJ-7FHb_mw51HqMBI0BjFy97H6GnjZaX402UQZK89E_mLvMu$" id="m_-2363802228770409313x_m_-6675981207680419905OWA24a4f6e7-fa61-8e5e-b5f7-4549d0721265" target="_blank">https://github.com/openjdk/jdk/pull/23467</a><br>
<br>
If you are able to make a local OpenJDK build with that solution you could check if it addresses your use-cases (or not).</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964LPBorder_GTaHR0cHM6Ly9naXRodWIuY29tL29wZW5qZGsvamRrL3B1bGwvMjM0Njc." style="width:100%;margin-top:16px;margin-bottom:16px;max-width:800px;min-width:424px;color:inherit">
<table id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964LPContainer930895" style="border-width:1px;border-style:solid;border-color:rgb(200,200,200);border-radius:2px;width:100%">
<tbody>
<tr>
<td style="width:254px;height:296px">
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964LPImageContainer930895" style="margin-right:12px;height:120px;overflow:hidden;width:240px;color:inherit">
<a href="https://urldefense.com/v3/__https://github.com/openjdk/jdk/pull/23467__;!!ACWV5N9M2RV99hQ!M4nB0WxHU_1jcbZoNGy3DD81Oefr7BuObJ-7FHb_mw51HqMBI0BjFy97H6GnjZaX402UQZK89E_mLvMu$" id="m_-2363802228770409313x_m_-6675981207680419905OWA00a1a52d-050e-4191-5040-d09daaa4f887" target="_blank"><img id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964LPThumbnailImageId930895" width="240" height="120" style="width: 240px; height: 120px; display: block;" src="https://opengraph.githubassets.com/00e04f8a63bde12217df087df7ef8edee563adf7e925d07c75bdeae092180094/openjdk/jdk/pull/23467"></a></div>
</td>
<td style="width:100%;height:296px">
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964LPTitle930895" style="font-size:21px;font-weight:300;margin-right:8px;font-family:wf_segoe-ui_light,"Segoe UI Light","Segoe WP Light","Segoe UI","Segoe WP",Tahoma,Arial,sans-serif;margin-bottom:12px;color:inherit">
<a href="https://urldefense.com/v3/__https://github.com/openjdk/jdk/pull/23467__;!!ACWV5N9M2RV99hQ!M4nB0WxHU_1jcbZoNGy3DD81Oefr7BuObJ-7FHb_mw51HqMBI0BjFy97H6GnjZaX402UQZK89E_mLvMu$" id="m_-2363802228770409313x_m_-6675981207680419905OWA5f9372a6-a2f8-b6a1-9874-0f731f4560f2" style="text-decoration:none" target="_blank">8349462:
 Gatherers.mapConcurrent could support async interrupts by viktorklang-ora · Pull Request #23467 · openjdk/jdk</a></div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964LPDescription930895" style="font-size:14px;max-height:100px;color:rgb(102,102,102);font-family:wf_segoe-ui_normal,"Segoe UI","Segoe WP",Tahoma,Arial,sans-serif;margin-bottom:12px;margin-right:8px;overflow:hidden">
This change is likely going to need some extra verbiage in the spec for mapConcurrent, and thus a CSR. This behavior aligns mapConcurrent with how parallel streams work in conjunction with interrup...</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964LPMetadata930895" style="font-size:14px;font-weight:400;color:rgb(166,166,166);font-family:wf_segoe-ui_normal,"Segoe UI","Segoe WP",Tahoma,Arial,sans-serif">
<a href="https://urldefense.com/v3/__http://github.com__;!!ACWV5N9M2RV99hQ!M4nB0WxHU_1jcbZoNGy3DD81Oefr7BuObJ-7FHb_mw51HqMBI0BjFy97H6GnjZaX402UQZK89DBkMefT$" id="m_-2363802228770409313x_m_-6675981207680419905OWA0f077dba-17cf-8c14-9fc6-318aed714e5d" target="_blank">github.com</a></div>
</td>
</tr>
</tbody>
</table>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964Signature" style="color:inherit">
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Cheers,<br>
√</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b><br>
</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b>Viktor Klang</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Software Architect, Java Platform Group<br>
Oracle</div>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964appendonsend" style="color:inherit">
</div>
<hr style="direction:ltr;display:inline-block;width:98%">
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964divRplyFwdMsg" dir="ltr" style="color:inherit">
<span style="font-family:Calibri,sans-serif;font-size:11pt;color:rgb(0,0,0)"><b>From:</b> Jige Yu <<a href="mailto:yujige@gmail.com" id="m_-2363802228770409313x_m_-6675981207680419905OWA14ef00ec-b376-943c-4d4b-db5036c65d05" target="_blank">yujige@gmail.com</a>><br>
<b>Sent:</b> Wednesday, 5 February 2025 16:24<br>
<b>To:</b> Viktor Klang <<a href="mailto:viktor.klang@oracle.com" id="m_-2363802228770409313x_m_-6675981207680419905OWA13449932-59af-f9fc-7523-747d6070875f" target="_blank">viktor.klang@oracle.com</a>><br>
<b>Cc:</b> <a href="mailto:core-libs-dev@openjdk.org" id="m_-2363802228770409313x_m_-6675981207680419905OWA3695acc4-fdfb-0d57-a31f-54d63e369b9a" target="_blank">
core-libs-dev@openjdk.org</a> <<a href="mailto:core-libs-dev@openjdk.org" id="m_-2363802228770409313x_m_-6675981207680419905OWA64ed8d7d-c5e2-7299-5168-b4d86f88e5fc" target="_blank">core-libs-dev@openjdk.org</a>><br>
<b>Subject:</b> Re: [External] : Re: mapConcurrent() with InterruptedException</span>
<div> </div>
</div>
<div style="direction:ltr">Thanks Viktor!</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">I understand the problem.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">The main reason I asked is because I want to understand how the core Java team thinks of throwing an unchecked exception.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">As explained above, I consider losing cancellability a big deal, a deal breaker even. And I thought throwing unchecked is more acceptable. Because the most common reason the mapConcurrent() VT can be interrupted is due to cancellation
 from a parent mapConcurrent(), or a parent Structured Concurrency scope. The cancellation could be either from an organic exception, or from the downstream not needing more elements, like maybe due to findFirst() already getting an element.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">In both cases, since the concurrent operation is already cancelled (result ignored), what exception pops up to the top level isn't that big of a deal (perhaps only a log record will be seen?)</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">But if the core Java team considers it a bad idea, I would love to learn and adjust.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">On Tue, Feb 4, 2025 at 4:41 AM Viktor Klang <<a href="mailto:viktor.klang@oracle.com" id="m_-2363802228770409313x_m_-6675981207680419905OWAa62d22ea-afce-a312-7c9c-5cec17063b8a" target="_blank">viktor.klang@oracle.com</a>> wrote:</div>
<blockquote style="margin:0px 0px 0px 0.8ex;padding-left:1ex;border-left:1px solid rgb(204,204,204)">
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Hi,</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;line-height:normal;margin:0px;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
The problem is that mapConcurrent cannot throw InterruptedException because that is a checked exception, so we cannot clear the interrupted flag and throw that exception.</div>
<div style="direction:ltr;line-height:normal;margin:0px;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;line-height:normal;margin:0px;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
So the updated semantics is to not cut the stream short but instead run to completion, restoring the interruption flag.</div>
<div style="direction:ltr;line-height:normal;margin:0px;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;line-height:normal;margin:0px;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
There exists a couple of alternatives to this approach which I am contemplating, but they need to be further explored before I consider moving forward with any of them.</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964x_m_5064905880974080681Signature" style="color:inherit">
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Cheers,<br>
√</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b><br>
</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b>Viktor Klang</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Software Architect, Java Platform Group<br>
Oracle</div>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964x_m_5064905880974080681appendonsend" style="color:inherit">
</div>
<hr style="direction:ltr;display:inline-block;width:98%">
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964x_m_5064905880974080681divRplyFwdMsg" dir="ltr" style="color:inherit">
<span style="font-family:Calibri,sans-serif;font-size:11pt;color:rgb(0,0,0)"><b>From:</b> Jige Yu <<a href="mailto:yujige@gmail.com" id="m_-2363802228770409313x_m_-6675981207680419905OWA90a4d359-8686-f02d-f685-a2b0feae6aee" target="_blank">yujige@gmail.com</a>><br>
<b>Sent:</b> Monday, 27 January 2025 17:00<br>
<b>To:</b> Viktor Klang <<a href="mailto:viktor.klang@oracle.com" id="m_-2363802228770409313x_m_-6675981207680419905OWA8e659b32-c53d-4505-0d35-26f2b697405a" target="_blank">viktor.klang@oracle.com</a>><br>
<b>Cc:</b> <a href="mailto:core-libs-dev@openjdk.org" id="m_-2363802228770409313x_m_-6675981207680419905OWA9b063da0-b72c-ec31-14ad-50a9b2662b70" target="_blank">core-libs-dev@openjdk.org</a> <<a href="mailto:core-libs-dev@openjdk.org" id="m_-2363802228770409313x_m_-6675981207680419905OWA87845bc6-b92e-eeb3-bdf6-949f36db96a2" target="_blank">core-libs-dev@openjdk.org</a>><br>
<b>Subject:</b> Re: [External] : Re: mapConcurrent() with InterruptedException</span>
<div> </div>
</div>
<div style="direction:ltr">Thanks Viktor!</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">It looks like the current fix ignores interruption.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">I want to make sure my concern of it defeating cancellation is heard and understood.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">The scenarios I worry about is for a mapConcurrent() that fans out to another method call, which internally calls mapConcurrent() as implementation detail.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">An example:</div>
<div style="direction:ltr"><br>
</div>
<pre style="margin-top:0px;margin-bottom:0px;border-radius:6px;min-height:52px"><div style="direction:ltr;line-height:1.45;font-size:13.6px;color:rgb(31,35,40)">List<RefundResponse> refundHelper(transaction) {
  transaction.creditCardAccounts.stream()
    .gather(mapConcurrent(acct -> service.refund(acct))
    .toList();
}
transactions.stream()
    .gather(mapConcurrent(transaction -> refundHelper(transaction));</div></pre>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">It seems undesirable that in such a case all the service.refund() calls become non cancellable, because the only way the outer mapConcurrent() cancels the refundHelper() calls is through Thread.interrupt() the virtual threads that
 call refundHelper(), which would then be disabled by the inner mapConcurrent().</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">Does this example make sense to you? I can further explain if anything isn't clear.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">But I want to make sure the decision to disable interruption is deliberate judgement call that such nested mapConcurrent() is unlikely,or not important.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">Cheers,</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">On Mon, Jan 27, 2025 at 6:11 AM Viktor Klang <<a href="mailto:viktor.klang@oracle.com" id="m_-2363802228770409313x_m_-6675981207680419905OWA9241d290-4eea-4c15-6d78-1ce94bcb0c40" target="_blank">viktor.klang@oracle.com</a>> wrote:</div>
<blockquote style="margin:0px 0px 0px 0.8ex;padding-left:1ex;border-left:1px solid rgb(204,204,204)">
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Hi!</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Please see: <a href="https://urldefense.com/v3/__https://github.com/openjdk/jdk/pull/23100__;!!ACWV5N9M2RV99hQ!IzQs0G26M7ZGPwJ3YJpCcS0gxi6BjqoBux2T5u0cHud_zb_mHLfiIrASSZiP0ynNgnaAuwuOh__WinK8$" id="m_-2363802228770409313x_m_-6675981207680419905OWA620d4b91-8524-eb52-3075-ce784c9092e1" target="_blank">
https://github.com/openjdk/jdk/pull/23100</a></div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964x_m_5064905880974080681x_m_7996180640990855094Signature" style="color:inherit">
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Cheers,<br>
√</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b><br>
</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b>Viktor Klang</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Software Architect, Java Platform Group<br>
Oracle</div>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964x_m_5064905880974080681x_m_7996180640990855094appendonsend" style="color:inherit">
</div>
<hr style="direction:ltr;display:inline-block;width:98%">
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964x_m_5064905880974080681x_m_7996180640990855094divRplyFwdMsg" dir="ltr" style="color:inherit">
<span style="font-family:Calibri,sans-serif;font-size:11pt;color:rgb(0,0,0)"><b>From:</b> Jige Yu <<a href="mailto:yujige@gmail.com" id="m_-2363802228770409313x_m_-6675981207680419905OWAf89fae2e-6237-7ab8-4472-6781cbbb366e" target="_blank">yujige@gmail.com</a>><br>
<b>Sent:</b> Sunday, 26 January 2025 23:03<br>
<b>To:</b> Viktor Klang <<a href="mailto:viktor.klang@oracle.com" id="m_-2363802228770409313x_m_-6675981207680419905OWA0dc8becd-a859-acc4-facf-217e069d4182" target="_blank">viktor.klang@oracle.com</a>><br>
<b>Cc:</b> <a href="mailto:core-libs-dev@openjdk.org" id="m_-2363802228770409313x_m_-6675981207680419905OWA771362a6-b5e1-f766-3006-7e0cebbfc33b" target="_blank">
core-libs-dev@openjdk.org</a> <<a href="mailto:core-libs-dev@openjdk.org" id="m_-2363802228770409313x_m_-6675981207680419905OWAa89125c8-72fd-4934-3d7b-965900cebb3e" target="_blank">core-libs-dev@openjdk.org</a>><br>
<b>Subject:</b> [External] : Re: mapConcurrent() with InterruptedException</span>
<div> </div>
</div>
<div style="direction:ltr">Checking in on what you've found out, Viktor.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">From where we left off, I understand that you were looking at alternatives instead of silent truncation?</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">Have you reached any conclusion?</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">We touched on disallowing interruption during mapConcurrent(). I still have concerns with disabling cancellation, because it basically undoes this API note from the
<a href="https://cr.openjdk.org/~alanb/sc-20240503/java.base/java/util/stream/Gatherers.html#mapConcurrent(int,java.util.function.Function)" id="m_-2363802228770409313x_m_-6675981207680419905OWAe301736d-dc65-4edb-bf2a-b86bdbb6ab26" target="_blank">
javadoc</a>:</div>
<div style="direction:ltr"><br>
</div>
<dl style="color:inherit"><dt style="font-family:"DejaVu Sans",Arial,Helvetica,sans-serif;font-size:0.856em;font-weight:bold;margin:10px 0px 0px;color:rgb(53,56,51)">
API Note:</dt></dl>
<dl style="color:inherit"><dd style="margin:5px 0px 10px 10px;font-size:14px;color:rgb(53,56,51)">
In progress tasks will be attempted to be cancelled, on a best-effort basis, in situations where the downstream no longer wants to receive any more elements.</dd></dl>
<div style="direction:ltr">In reality, people will use mapConcurrent() to fan out rpcs. Sometimes these rpcs are just a single blocking call; yet sometimes they may themselves be a Structured Concurrency scope, with 2 or 3 rpcs that constitute a single logical
 operation. Under two conditions, cancellation is imho important semantic:</div>
<ol start="1" style="direction:ltr">
<li style="direction:ltr">The downstream code uses filter().findFirst(), and when it sees an element, it will return and no longer needs the other pending rpcs to complete. If cancellation is disabled, these unnecessary rpcs will waste system resources.</li><li style="direction:ltr">One of the rpc throws and the Stream pipeline needs to propagate the exception. Again, if the other rpcs cannot be cancelled, we'll have many zombie rpcs.</li></ol>
<div style="direction:ltr">Zombie rpcs may or may not be a deal breaker, depending on the specific use case. But for a JDK library, losing cancellation would have a negative impact on usability.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">My 2c,</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">On Fri, Jan 3, 2025 at 9:18 AM Viktor Klang <<a href="mailto:viktor.klang@oracle.com" id="m_-2363802228770409313x_m_-6675981207680419905OWA88dcf683-1bb4-acff-9546-5f46bdb7dc55" target="_blank">viktor.klang@oracle.com</a>> wrote:</div>
<blockquote style="margin:0px 0px 0px 0.8ex;padding-left:1ex;border-left:1px solid rgb(204,204,204)">
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Hi Ben,<br>
<br>
Thanks for raising these questions—getting feedback is crucial in the Preview stage of features.<br>
<br>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
I wrote a reply to the Reddit thread so I'll just summarize here:</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
It is important to note that <i>mapConcurrent()</i> is not a part of the Structured Concurrency JEPs, so it is not designed to join SC scopes.</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
I'm currently experimenting with ignoring-but-restoring interrupts on the "calling thread" for
<i>mapConcurrent()</i>, as well as capping work-in-progress to <i>maxConcurrency</i> (not only capping the concurrency but also the amount of completed-but-yet-to-be-pushed work). Both of these adjustments should increase predictability of behavior in the face
 of blocking operations with variable delays.</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Another adjustment I'm looking at right now is to harden/improve the cleanup to wait for concurrent tasks to acknowledge cancellation, so that once the finisher is done executing the VTs are known to have terminated.</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
As for not preserving the encounter order, that would be a completely different thing, and I'd encourage you to experiment with that if that functionality would be interesting for your use-case(s).</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Again, thanks for your feedback!</div>
<div style="direction:ltr;font-family:Aptos,Aptos_EmbeddedFont,Aptos_MSFontService,Calibri,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964x_m_5064905880974080681x_m_7996180640990855094x_m_-2136516963982356393Signature" style="color:inherit">
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Cheers,<br>
√</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<br>
</div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b><br>
</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
<b>Viktor Klang</b></div>
<div style="font-family:Calibri,Arial,Helvetica,sans-serif;font-size:12pt;color:rgb(0,0,0)">
Software Architect, Java Platform Group<br>
Oracle</div>
</div>
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964x_m_5064905880974080681x_m_7996180640990855094x_m_-2136516963982356393appendonsend" style="color:inherit">
</div>
<hr style="direction:ltr;display:inline-block;width:98%">
<div id="m_-2363802228770409313x_m_-6675981207680419905x_m_3463815916163574184x_x_m_6472783015755741964x_m_5064905880974080681x_m_7996180640990855094x_m_-2136516963982356393divRplyFwdMsg" dir="ltr" style="color:inherit">
<span style="font-family:Calibri,sans-serif;font-size:11pt;color:rgb(0,0,0)"><b>From:</b> core-libs-dev <<a href="mailto:core-libs-dev-retn@openjdk.org" id="m_-2363802228770409313x_m_-6675981207680419905OWA7c9d96ef-5acd-14c6-9ccc-c87eb38305f0" target="_blank">core-libs-dev-retn@openjdk.org</a>>
 on behalf of Jige Yu <<a href="mailto:yujige@gmail.com" id="m_-2363802228770409313x_m_-6675981207680419905OWAb737cd29-7e16-fbcd-a5cc-3eab35876929" target="_blank">yujige@gmail.com</a>><br>
<b>Sent:</b> Friday, 3 January 2025 17:53<br>
<b>To:</b> <a href="mailto:core-libs-dev@openjdk.org" id="m_-2363802228770409313x_m_-6675981207680419905OWA7dc1be26-004c-5c6e-d259-6abe54de9768" target="_blank">
core-libs-dev@openjdk.org</a> <<a href="mailto:core-libs-dev@openjdk.org" id="m_-2363802228770409313x_m_-6675981207680419905OWAb878e9b3-3873-b1b0-ee6d-d0d0c5e633ff" target="_blank">core-libs-dev@openjdk.org</a>><br>
<b>Subject:</b> mapConcurrent() with InterruptedException</span>
<div> </div>
</div>
<div style="direction:ltr">Hi Java Experts,</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">I sent this email incorrectly to loom-dev@ and was told on Reddit that core-libs-dev is the right list.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">The question is about the behavior of mapConcurrent() when the thread is interrupted.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">Currently mapConcurrent()'s finisher phase will re-interrupt the thread, then stop at whatever element that has already been processed and return.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">This strikes me as a surprising behavior, because for example if I'm running:</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">   Stream.of(1, 2, 3)</div>
<div style="direction:ltr">        .gather(mapConcurrent(i -> i * 2))</div>
<div style="direction:ltr">        .toList()</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">and the thread is being interrupted, the result could be any of [2], [2, 4] or [2, 4, 6].</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">Since thread interruption is cooperative, there is no guarantee that the thread being interrupted will just abort. It's quite possible that it'll keep going and then will use for example [2] as the result of doubling the list of [1,
 2, 3], which is imho incorrect.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">In the <a href="https://urldefense.com/v3/__https://www.reddit.com/r/java/comments/1hr8xyu/observations_of_gatherersmapconcurrent/?utm_source=share&utm_medium=web3x&utm_name=web3xcss&utm_term=1&utm_content=share_button__;!!ACWV5N9M2RV99hQ!L1LHRE2pnYPg43nM0J0dCoV4agscV_rybIV9jY97xn9XJi9d7VoPma4jhx4J4GBeoeQmVud8M4PjPb7L$" id="m_-2363802228770409313x_m_-6675981207680419905OWA0b47c17e-1a41-0f9e-97a5-b78bcb815820" target="_blank">
Reddit</a> thread, someone argued that interruption rarely happens so it's more of a theoretical issue. But interruption can easily happen in Structured Concurrency or in mapConcurrent() itself if any subtask has failed in order to cancel/interrupt the other
 ongoing tasks.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">There had been discussion about alternative strategies:</div>
<ol start="1" style="direction:ltr">
<li style="direction:ltr">Don't respond to interruption and just keep running to completion.</li><li style="direction:ltr">Re-interrupt thread and wrap the InterruptedException in a standard unchecked exception (StructuredConcurrencyInterruptedException?).</li></ol>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">I have concerns with option 1 because it disables cancellation propagation when mapConcurrent() itself is used in a subtask of a parent mapConcurrent() or in a StructuredConcurrencyScope.</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">Both equivalent Future-composition async code, or C++'s fiber trees support cancellation propagation and imho it's a critical feature or else it's possible that a zombie thread is still sending RPCs long after the main thread has
 exited (failed, or falled back to some default action).</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">My arguments for option 2:</div>
<ol start="1" style="direction:ltr">
<li style="direction:ltr">InterruptedException is more error prone than traditional checked exceptions for
<b>users</b> to catch and handle. They can forget to re-interrupt the thread. It's so confusing that even seasoned programmers may not know they are
<b>supposed to</b> re-interrupt the thread.</li><li style="direction:ltr">With Stream API using functional interfaces like Supplier, Function, the option of just tacking on "throws IE" isn't available to many users.</li><li style="direction:ltr">With Virtual Threads, it will be more acceptable, or even become common to do blocking calls from a stream operation (including but exclusive to mapConcurrent()). So the chance users are forced to deal with IE will become substantially
 higher.</li><li style="direction:ltr">Other APIs such as the Structured Concurrency API have already started wrapping system checked exceptions like ExecutionException, TimeoutException in unchecked exceptions (
<span style="font-family:-apple-system,"system-ui","Segoe UI",Roboto,"Helvetica Neue",Arial,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol",sans-serif;font-size:14px">
<a href="https://urldefense.com/v3/__https://download.java.net/java/early_access/loom/docs/api/java.base/java/util/concurrent/StructuredTaskScope.html*join()__;Iw!!ACWV5N9M2RV99hQ!L1LHRE2pnYPg43nM0J0dCoV4agscV_rybIV9jY97xn9XJi9d7VoPma4jhx4J4GBeoeQmVud8MxGG4HzA$" id="m_-2363802228770409313x_m_-6675981207680419905OWAc421012e-a0f9-c99c-c697-79362761a3ec" style="margin-top:0px" target="_blank">join()</a></span> for
 example).</li><li style="direction:ltr">Imho, exceptions that we'd rather users not catch and handle but instead should mostly just propagate up as is, should be unchecked.</li></ol>
<div style="direction:ltr">There is also a <a href="https://urldefense.com/v3/__https://www.reddit.com/r/java/comments/1hr8xyu/comment/m4z4f8c/?utm_source=share&utm_medium=web3x&utm_name=web3xcss&utm_term=1&utm_content=share_button__;!!ACWV5N9M2RV99hQ!L1LHRE2pnYPg43nM0J0dCoV4agscV_rybIV9jY97xn9XJi9d7VoPma4jhx4J4GBeoeQmVud8MyZYl02k$" id="m_-2363802228770409313x_m_-6675981207680419905OWA94a21379-c4d4-102e-eba5-b8cfed3a031b" target="_blank">
side discussion</a> about whether mapConcurrent() is better off preserving input order or push to downstream as soon as an element is computed. I'd love to discuss that topic too but maybe it's better to start a separate thread?</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">Thank you and cheers!</div>
<div style="direction:ltr"><br>
</div>
<div style="direction:ltr">Ben Yu</div>
<div style="direction:ltr"><br>
</div>
</blockquote>
</blockquote>
</blockquote>
</blockquote>
</blockquote>
</div>
</div>
</blockquote>
</div>
</div>
</div>
</div></blockquote></div>