Skip to content

Commit e508aec

Browse files
committedApr 28, 2021
Improve javadoc
1 parent bd07eb6 commit e508aec

File tree

2 files changed

+54
-35
lines changed

2 files changed

+54
-35
lines changed
 

‎src/java.base/share/classes/java/util/concurrent/Executors.java

+47-27
Original file line numberDiff line numberDiff line change
@@ -240,13 +240,12 @@ public static ExecutorService newCachedThreadPool(ThreadFactory threadFactory) {
240240
}
241241

242242
/**
243-
* Creates an Executor that starts a new Thread for each task. The Executor
244-
* is <i>owned</i> by the Thread that creates it.
243+
* Creates an Executor that starts a new Thread for each task.
245244
*
246245
* <p> An Executor created by this method is intended to be used in a
247246
* <em>structured manner</em>. It is <em>owned</em> by the Thread that creates
248-
* it and must be {@linkplain ExecutorService#close() closed} by the thread
249-
* when it is finished with the executor. Failure to invoke the {@code
247+
* it and must be {@linkplain ExecutorService#close() closed} by the same
248+
* thread when it is finished with the executor. Failure to invoke the {@code
250249
* close} method may result in a memory leak. The {@code close}, {@link
251250
* ExecutorService#shutdown() shutdown}, and {@link ExecutorService#shutdownNow()
252251
* shutdownNow} methods throw {@code IllegalCallerException} if invoked by
@@ -260,7 +259,8 @@ public static ExecutorService newCachedThreadPool(ThreadFactory threadFactory) {
260259
*
261260
* <p> Invoking {@link Future#cancel(boolean) cancel(true)} on a {@link
262261
* Future Future} representing the pending result of a task submitted to
263-
* the Executor will {@link Thread#interrupt() interrupt} the thread.
262+
* the Executor will {@link Thread#interrupt() interrupt} the thread
263+
* executing the task.
264264
*
265265
* @apiNote
266266
* The {@link #newUnownedThreadExecutor(ThreadFactory)} method should be
@@ -272,21 +272,37 @@ public static ExecutorService newCachedThreadPool(ThreadFactory threadFactory) {
272272
* @since 99
273273
*/
274274
public static ExecutorService newThreadExecutor(ThreadFactory threadFactory) {
275-
return new ThreadExecutor(threadFactory, null, false);
275+
return new ThreadExecutor(threadFactory, null, true);
276276
}
277277

278278
/**
279-
* Creates an Executor that starts a new Thread for each task. The Executor
280-
* is <i>owned</i> by the Thread that creates it.
281-
*
282-
* The Executor is created with a deadline. If the deadline is reached before
283-
* the Executor has terminated then it is {@link ExecutorService#shutdown()
284-
* shutdown} and its threads are {@linkplain Thread#interrupt() interrupted}
285-
* to cancel the remaining tasks.
286-
* If this method is invoked with a deadline that has already expired then
287-
* it returns an Executor that is already shutdown (any attempt to submit
279+
* Creates an Executor that starts a new Thread for each task and with a
280+
* deadline. If the deadline is reached before the Executor has terminated
281+
* then it is {@link ExecutorService#shutdown() shutdown} and its threads
282+
* are {@linkplain Thread#interrupt() interrupted} to cancel the remaining
283+
* tasks. If this method is invoked with a deadline that has already expired
284+
* then it returns an Executor that is already shutdown (any attempt to submit
288285
* a task will fail with {@code RejectedExecutionException}).
289286
*
287+
* <p> An Executor created by this method is intended to be used in a
288+
* <em>structured manner</em>. It is <em>owned</em> by the Thread that creates
289+
* it and must be {@linkplain ExecutorService#close() closed} by the same
290+
* thread when it is finished with the executor. Failure to invoke the {@code
291+
* close} method may result in a memory leak. The {@code close}, {@code
292+
* shutdown} and {@link ExecutorService#shutdownNow() shutdownNow} methods
293+
* throw {@code IllegalCallerException} if invoked by other threads.
294+
* Executors created by this method enforce strict nesting and must be
295+
* closed in the reverse order that they are created in. The {@code close}
296+
* method throws {@code IllegalStateException} if invoked to close
297+
* executors created by this method in a different order. Once closed by
298+
* its owner, further attempts to close the executor by its owner has no
299+
* effect.
300+
*
301+
* <p> Invoking {@link Future#cancel(boolean) cancel(true)} on a {@link
302+
* Future Future} representing the pending result of a task submitted to
303+
* the Executor will {@link Thread#interrupt() interrupt} the thread
304+
* executing the task.
305+
*
290306
* @param threadFactory the factory to use when creating new threads
291307
* @param deadline the deadline
292308
* @return a newly created executor
@@ -295,14 +311,12 @@ public static ExecutorService newThreadExecutor(ThreadFactory threadFactory) {
295311
*/
296312
public static ExecutorService newThreadExecutor(ThreadFactory threadFactory,
297313
Instant deadline) {
298-
Objects.requireNonNull(threadFactory);
299314
Objects.requireNonNull(deadline);
300-
return new ThreadExecutor(threadFactory, deadline, false);
315+
return new ThreadExecutor(threadFactory, deadline, true);
301316
}
302317

303318
/**
304-
* Creates an Executor that starts a new virtual Thread for each task. The
305-
* Executor is <i>owned</i> by the Thread that creates it.
319+
* Creates an Executor that starts a new virtual Thread for each task.
306320
*
307321
* <p> This method is equivalent to creating an Executor with the {@link
308322
* #newThreadExecutor(ThreadFactory)} method and specifying a ThreadFactory
@@ -313,12 +327,12 @@ public static ExecutorService newThreadExecutor(ThreadFactory threadFactory,
313327
*/
314328
public static ExecutorService newVirtualThreadExecutor() {
315329
ThreadFactory factory = Thread.ofVirtual().factory();
316-
return new ThreadExecutor(factory, null, false);
330+
return new ThreadExecutor(factory, null, true);
317331
}
318332

319333
/**
320-
* Creates an Executor that starts a new virtual Thread for each task. The
321-
* Executor is <i>owned</i> by the Thread that creates it.
334+
* Creates an Executor that starts a new virtual Thread for each task and
335+
* with a deadline.
322336
*
323337
* <p> This method is equivalent to creating an Executor with the {@link
324338
* #newThreadExecutor(ThreadFactory, Instant)} method and specifying a
@@ -332,15 +346,21 @@ public static ExecutorService newVirtualThreadExecutor() {
332346
public static ExecutorService newVirtualThreadExecutor(Instant deadline) {
333347
Objects.requireNonNull(deadline);
334348
ThreadFactory factory = Thread.ofVirtual().factory();
335-
return new ThreadExecutor(factory, deadline, false);
349+
return new ThreadExecutor(factory, deadline, true);
336350
}
337351

338352
/**
339353
* Creates an Executor that starts a new thread for each task.
340354
*
341-
* The resulting Executor is not owned to any thread, meaning any thread can
342-
* {@link ExecutorService#shutdown() shutdown} the Executor or invoke {@link
343-
* ExecutorService#close() close} if they have the appropriate permission.
355+
* <p> Unlike {@link #newThreadExecutor(ThreadFactory)}, the Executor is not
356+
* owned by any thread so any thread can {@link ExecutorService#shutdown()
357+
* shutdown} the Executor or invoke {@link ExecutorService#close() close}
358+
* if they have the appropriate permission.
359+
*
360+
* <p> Invoking {@link Future#cancel(boolean) cancel(true)} on a {@link
361+
* Future Future} representing the pending result of a task submitted to
362+
* the Executor will {@link Thread#interrupt() interrupt} the thread
363+
* executing the task.
344364
*
345365
* @apiNote
346366
* This method is intended for unstructured usage such as cases where an
@@ -353,7 +373,7 @@ public static ExecutorService newVirtualThreadExecutor(Instant deadline) {
353373
* @since 99
354374
*/
355375
public static ExecutorService newUnownedThreadExecutor(ThreadFactory threadFactory) {
356-
return new ThreadExecutor(threadFactory, null, true);
376+
return new ThreadExecutor(threadFactory, null, false);
357377
}
358378

359379
/**

‎src/java.base/share/classes/java/util/concurrent/ThreadExecutor.java

+7-8
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2019, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2019, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -44,7 +44,6 @@
4444
import jdk.internal.misc.InnocuousThread;
4545
import static java.util.concurrent.TimeUnit.NANOSECONDS;
4646

47-
4847
/**
4948
* An ExecutorService that executes each task in its own thread.
5049
*/
@@ -85,9 +84,9 @@ class ThreadExecutor implements ExecutorService {
8584
*
8685
* @param factory the thread factory
8786
* @param deadline the deadline, null for no deadline
88-
* @param shared true if not owned by the caller thread
87+
* @param owned true if owned by the caller thread
8988
*/
90-
ThreadExecutor(ThreadFactory factory, Instant deadline, boolean shared) {
89+
ThreadExecutor(ThreadFactory factory, Instant deadline, boolean owned) {
9190
Objects.requireNonNull(factory);
9291
Future<?> timer = null;
9392
if (deadline != null) {
@@ -103,14 +102,14 @@ class ThreadExecutor implements ExecutorService {
103102

104103
this.factory = factory;
105104
this.timerTask = timer;
106-
if (shared) {
107-
this.owner = null;
108-
this.previous = null;
109-
} else {
105+
if (owned) {
110106
Thread owner = Thread.currentThread();
111107
this.owner = owner;
112108
this.previous = ThreadFields.latestThreadExecutor(owner);
113109
ThreadFields.setLatestThreadExecutor(this);
110+
} else {
111+
this.owner = null;
112+
this.previous = null;
114113
}
115114
}
116115

0 commit comments

Comments
 (0)
Please sign in to comment.