update at 2021-09-12 21:03:19 by ehlxr
This commit is contained in:
260
budd-demo/src/main/java/io/github/ehlxr/ManagedBlockerTest.java
Normal file
260
budd-demo/src/main/java/io/github/ehlxr/ManagedBlockerTest.java
Normal file
@@ -0,0 +1,260 @@
|
||||
/*
|
||||
* The MIT License (MIT)
|
||||
*
|
||||
* Copyright © 2021 xrv <xrg@live.com>
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*/
|
||||
|
||||
package io.github.ehlxr;
|
||||
|
||||
import com.google.common.base.Stopwatch;
|
||||
|
||||
import java.time.LocalTime;
|
||||
import java.time.format.DateTimeFormatter;
|
||||
import java.util.List;
|
||||
import java.util.concurrent.*;
|
||||
import java.util.function.Consumer;
|
||||
import java.util.function.Supplier;
|
||||
import java.util.stream.Collectors;
|
||||
import java.util.stream.Stream;
|
||||
|
||||
/**
|
||||
* @author ehlxr
|
||||
* @since 2021-09-02 15:52.
|
||||
*/
|
||||
public class ManagedBlockerTest {
|
||||
static String threadDateTimeInfo() {
|
||||
return DateTimeFormatter.ISO_TIME.format(LocalTime.now()) + Thread.currentThread().getName();
|
||||
}
|
||||
|
||||
static void test1() {
|
||||
List<RecursiveTask<String>> tasks = Stream.generate(() -> new RecursiveTask<String>() {
|
||||
private static final long serialVersionUID = 1303648247573881735L;
|
||||
|
||||
@Override
|
||||
protected String compute() {
|
||||
System.out.println(threadDateTimeInfo() + ":simulate io task blocking for 2 seconds···");
|
||||
try {
|
||||
//线程休眠2秒模拟IO调用阻塞
|
||||
TimeUnit.SECONDS.sleep(2);
|
||||
} catch (InterruptedException e) {
|
||||
throw new Error(e);
|
||||
}
|
||||
return threadDateTimeInfo() + ": io blocking task returns successfully";
|
||||
}
|
||||
}).limit(28).collect(Collectors.toList());
|
||||
|
||||
tasks.forEach(ForkJoinTask::fork);
|
||||
tasks.forEach(e -> {
|
||||
try {
|
||||
System.out.println(e.get());
|
||||
} catch (Exception ex) {
|
||||
ex.printStackTrace();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
public static void main(String[] args) {
|
||||
Stopwatch started = Stopwatch.createStarted();
|
||||
test4();
|
||||
System.out.println(started);
|
||||
}
|
||||
|
||||
static void test2() {
|
||||
List<IoBlockerTask<String>> tasks = Stream.generate(() -> new IoBlockerTask<>(() -> {
|
||||
System.out.println(threadDateTimeInfo() + ":simulate io task blocking for 2 seconds···");
|
||||
try {
|
||||
TimeUnit.SECONDS.sleep(2);
|
||||
} catch (InterruptedException e) {
|
||||
throw new Error(e);
|
||||
}
|
||||
return threadDateTimeInfo() + ": io blocking task returns successfully";
|
||||
})).limit(28).collect(Collectors.toList());
|
||||
|
||||
tasks.forEach(ForkJoinTask::fork);
|
||||
tasks.forEach(e -> {
|
||||
try {
|
||||
System.out.println(e.get());
|
||||
} catch (Exception ex) {
|
||||
ex.printStackTrace();
|
||||
}
|
||||
});
|
||||
|
||||
}
|
||||
|
||||
static void test3() {
|
||||
List<IoBlockerAction> actions = Stream.generate(() -> new IoBlockerAction(() -> {
|
||||
System.out.println(threadDateTimeInfo() + ":simulate io action blocking for 2 seconds···");
|
||||
try {
|
||||
TimeUnit.SECONDS.sleep(2);
|
||||
} catch (InterruptedException e) {
|
||||
throw new Error(e);
|
||||
}
|
||||
System.out.println(threadDateTimeInfo() + ": io blocking action successfully");
|
||||
})).limit(56).collect(Collectors.toList());
|
||||
|
||||
//如果不使用 ManagedBlocker 耗时估算: 56/7(8 cpu - 1) = 8 * 2 =16s,实际 约等于 2s
|
||||
|
||||
actions.forEach(ForkJoinTask::fork);
|
||||
actions.forEach(ForkJoinTask::join);
|
||||
}
|
||||
|
||||
static void test4() {
|
||||
List<IoBlockerAction2<Integer>> actions = Stream.generate(() -> new IoBlockerAction2<>((x) -> {
|
||||
System.out.println(threadDateTimeInfo() + ":simulate io action blocking for 2 seconds···" + x);
|
||||
try {
|
||||
TimeUnit.SECONDS.sleep(2);
|
||||
} catch (InterruptedException e) {
|
||||
throw new Error(e);
|
||||
}
|
||||
System.out.println(threadDateTimeInfo() + ": io blocking action successfully " + x);
|
||||
}, 1)).limit(56).collect(Collectors.toList());
|
||||
|
||||
//如果不使用 ManagedBlocker 耗时估算: 56/7(8 cpu - 1) = 8 * 2 =16s,实际 约等于 2s
|
||||
|
||||
actions.forEach(x -> {
|
||||
x.fork();
|
||||
System.out.println("fork {}"+ x);
|
||||
});
|
||||
actions.forEach(x -> {
|
||||
x.join();
|
||||
System.out.println("join {}"+ x);
|
||||
});
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
class IoBlockerAction2<T> extends RecursiveAction {
|
||||
private static final long serialVersionUID = -4114798302790287199L;
|
||||
private final MyManagedBlockerImpl<T> blocker;
|
||||
|
||||
public IoBlockerAction2(Consumer<T> consumer, T t) {
|
||||
this.blocker = new MyManagedBlockerImpl<>(consumer, t);
|
||||
}
|
||||
|
||||
static class MyManagedBlockerImpl<T> implements ForkJoinPool.ManagedBlocker {
|
||||
private final Consumer<T> consumer;
|
||||
private final T t;
|
||||
|
||||
public MyManagedBlockerImpl(Consumer<T> consumer, T t) {
|
||||
this.consumer = consumer;
|
||||
this.t = t;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean block() {
|
||||
consumer.accept(t);
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isReleasable() {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void compute() {
|
||||
try {
|
||||
ForkJoinPool.managedBlock(blocker);
|
||||
} catch (InterruptedException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
class IoBlockerAction extends RecursiveAction {
|
||||
private static final long serialVersionUID = -4114798302790287199L;
|
||||
private final MyManagedBlockerImpl blocker;
|
||||
|
||||
public IoBlockerAction(Runnable runnable) {
|
||||
this.blocker = new MyManagedBlockerImpl(runnable);
|
||||
}
|
||||
|
||||
static class MyManagedBlockerImpl implements ForkJoinPool.ManagedBlocker {
|
||||
private final Runnable runnable;
|
||||
|
||||
public MyManagedBlockerImpl(Runnable runnable) {
|
||||
this.runnable = runnable;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean block() {
|
||||
runnable.run();
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isReleasable() {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void compute() {
|
||||
try {
|
||||
ForkJoinPool.managedBlock(blocker);
|
||||
// setRawResult(blocker.result);
|
||||
// return getRawResult();
|
||||
} catch (InterruptedException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
class IoBlockerTask<T> extends RecursiveTask<T> {
|
||||
private static final long serialVersionUID = -4114798302790287199L;
|
||||
private final MyManagedBlockerImpl<T> blocker;
|
||||
|
||||
public IoBlockerTask(Supplier<T> supplier) {
|
||||
this.blocker = new MyManagedBlockerImpl<>(supplier);
|
||||
}
|
||||
|
||||
static class MyManagedBlockerImpl<T> implements ForkJoinPool.ManagedBlocker {
|
||||
private final Supplier<T> supplier;
|
||||
private T result;
|
||||
|
||||
public MyManagedBlockerImpl(Supplier<T> supplier) {
|
||||
this.supplier = supplier;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean block() {
|
||||
result = supplier.get();
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isReleasable() {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected T compute() {
|
||||
try {
|
||||
ForkJoinPool.managedBlock(blocker);
|
||||
setRawResult(blocker.result);
|
||||
return getRawResult();
|
||||
} catch (InterruptedException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user