CompletableFuture的使用,多个CompletableFuture的处理

对单个CompletableFuture的处理方式可以查看https://mp.csdn.net/mdeditor/91348503#

本文主要介绍如何组合使用多个CompletableFuture。

thenCombine
把两个 CompletableFuture的任务都执行完成后,把两个任务的结果一块交给 thenCombine 来处理。thenCombine是可以有返回的。

	@Test
	public void thenCombine() throws Exception {
		CompletableFuture<String> result = CompletableFuture.supplyAsync(() -> {
			delaySec(1);
			printCurrTime("第一个CF");
			return "hello1";
		}).thenCombine(CompletableFuture.supplyAsync(() -> {
			printCurrTime("第二个CF");
			delaySec(3); 
			return "hello2";
		}), (t, u) -> {
			printCurrTime("组合");
			return t + " " + u;
		});
		System.out.println(result.get());
	}

执行结果:
在这里插入图片描述
可以看到两个线程的执行是并行的,thenCombine在两个线程都返回结果时执行。
注意:如果某个线程出现了异常,则thenCombine不会执行。

thenAcceptBoth
这个方法和上面的方法作用类似,都是接受两个线程的返回结果,并作出处理。只不过这个方法是消费式的,没有返回。注意这里的返回是指自定义接口方法中的返回。基本上所有的方法都会返回一个CompletionStage 至于泛型则由接口方法返回值决定。

	@Test
	public void thenAcceptBoth() throws Exception {
		CompletableFuture<Integer> f1 = CompletableFuture.supplyAsync(() -> {
			int t = new Random().nextInt(3);
			try {
				TimeUnit.SECONDS.sleep(t);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			System.out.println("f1=" + t);
			return t;
		});

		CompletableFuture<Integer> f2 = CompletableFuture.supplyAsync(() -> {
			int t = new Random().nextInt(3);
			try {
				TimeUnit.SECONDS.sleep(t);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			System.out.println("f2=" + t);
			return t;
		});
		TimeUnit.SECONDS.sleep(10);
		f1.thenAcceptBoth(f2, (t, u) -> {
			System.out.println("f1=" + t + ";f2=" + u + ";");
		});
	}

applyToEither
哪个线程先返回就使用谁的返回结果进入该方法。另一个线程如果执行时间过长则不再执行。该方法参数为函数式接口是有返回类型的。

	@Test
	public void applyToEither() throws Exception {
		CompletableFuture<Integer> f1 = CompletableFuture.supplyAsync(() -> {
			int t = new Random().nextInt(3);
			try {
				TimeUnit.SECONDS.sleep(t);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			System.out.println("f1=" + 1);
			printCurrTime("f1返回时间");
			return 1;
		});
		CompletableFuture<Integer> f2 = CompletableFuture.supplyAsync(() -> {
			int t = new Random().nextInt(3);
			try {
				TimeUnit.SECONDS.sleep(t);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			System.out.println("f2=" + 3);
			printCurrTime("f2返回时间");
			return 3;
		});

		CompletableFuture<Integer> result = f1.applyToEither(f2, t -> {
			System.out.println(t);
			return t * 2;
		});

		System.out.println(result.get());
	}

运行结果:
CompletableFuture的使用,多个CompletableFuture的处理_第1张图片

acceptEither
和上面的方法作用相似,但是该方法是消费式,不会线程不会返回执行结果。

@Test
	public void acceptEither() throws Exception {
		CompletableFuture<Integer> f1 = CompletableFuture.supplyAsync(() -> {
			int t = new Random().nextInt(3);
			try {
				TimeUnit.SECONDS.sleep(t);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			System.out.println("f1=" + t);
			return t;
		});

		CompletableFuture<Integer> f2 = CompletableFuture.supplyAsync(() -> {
			int t = new Random().nextInt(3);
			try {
				TimeUnit.SECONDS.sleep(t);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			System.out.println("f2=" + t);
			return t;
		});
		CompletableFuture<Void> acceptEither = f1.acceptEither(f2, t -> {
			System.out.println(t);
		});
	}

runAfterEither
该方法非阻塞。只有在该方法执行之前有线程返回该方法才会执行。

	@Test
	public void runAfterEither() throws Exception {
		CompletableFuture<Integer> f1 = CompletableFuture.supplyAsync(() -> {
			try {
				TimeUnit.SECONDS.sleep(3);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			System.out.println("f1=" + 1);
			return 1;
		});

		CompletableFuture<Integer> f2 = CompletableFuture.supplyAsync(() -> {
			try {
				TimeUnit.SECONDS.sleep(5);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			System.out.println("f2=" + 2);
			return 2;
		});

		TimeUnit.SECONDS.sleep(4);
		f1.runAfterEither(f2, () -> System.out.println("上面有一个已经完成了。"));
	}

运行结果:
CompletableFuture的使用,多个CompletableFuture的处理_第2张图片

runAfterBoth
同样为非阻塞方法,且只有当两个线程都在该方法执行之前结束。才会执行该方法。

	@Test
	public void runAfterBoth() throws Exception {
		CompletableFuture<Integer> f1 = CompletableFuture.supplyAsync(() -> {
			try {
				TimeUnit.SECONDS.sleep(3);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			System.out.println("f1=" + 1);
			return 1;
		});

		CompletableFuture<Integer> f2 = CompletableFuture.supplyAsync(() -> {
			try {
				TimeUnit.SECONDS.sleep(5);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			System.out.println("f2=" + 2);
			return 2;
		});
		TimeUnit.SECONDS.sleep(5);
		f1.runAfterBoth(f2, () -> System.out.println("上面两个任务都执行完成了。"));
	}

运行结果:
CompletableFuture的使用,多个CompletableFuture的处理_第3张图片

thenCompose
将两个线程串行连接起来,只有第一个线程返回结果时,才会将返回值作为参数传给第二个线程执行。

	@Test
	public void thenCompose() throws Exception {
		CompletableFuture<Integer> f = CompletableFuture.supplyAsync(() -> {
			int t = new Random().nextInt(3);
			System.out.println("t1=" + t);
			printCurrTime("F1返回时间");
			return t;
		}).thenCompose(	param -> {
				return CompletableFuture.supplyAsync(() -> {
					printCurrTime("F2进入时间");
					int t = param * 2;
					System.out.println("t2=" + t);
					return t;
				});

		});
		System.out.println("thenCompose result : " + f.get());
	}

执行结果:
CompletableFuture的使用,多个CompletableFuture的处理_第4张图片

end

你可能感兴趣的:(多线程)