1.什么是异步编程?
异步编程是一种编程模式,允许程序在等待某些操作(例如文件I/O或网络请求)完成时,不必停下来等待,而是继续执行其他任务。当异步操作完成时,回调函数或任务调度器会处理结果,从而提高程序的并发性和响应速度。
2.同步 vs 异步
- 同步:在调用某个方法时,调用方会一直等待该方法执行完毕,才会继续执行后续的操作。期间,调用方被阻塞。
- 异步:调用某个方法时,调用方可以立即返回,并在方法执行完毕后,通过回调或事件通知的方式处理结果,调用方不会被阻塞。
3.Java中的异步编程工具
Java从java.util.concurrent包开始支持丰富的并发工具,特别是CompletableFuture类,它是异步编程的核心工具。让我们先看看CompletableFuture的基本使用方法。
CompletableFuture:强大的异步任务管理
CompletableFuture是Java 8引入的类,它允许我们轻松地编写异步代码。它不仅可以创建异步任务,还能链式地处理任务结果。
3.1创建异步任务
我们可以通过runAsync()或supplyAsync()方法来创建异步任务。runAsync()适用于不返回结果的任务,而supplyAsync()适用于返回结果的任务。
import java.util.concurrent.CompletableFuture;public class AsyncExample {public static void main(String[] args) {// 创建一个异步任务,模拟计算CompletableFuture<Void> future = CompletableFuture.runAsync(() -> {System.out.println("正在执行异步任务...");try {Thread.sleep(2000); // 模拟耗时操作} catch (InterruptedException e) {e.printStackTrace();}System.out.println("异步任务完成");});// 主线程继续执行System.out.println("主线程继续执行其他操作...");// 等待异步任务完成future.join();}
}
3.2获取异步任务结果
使用supplyAsync()时,我们可以通过thenApply()方法来处理异步任务的结果。thenApply()接收一个函数,表示任务完成后的处理逻辑。
import java.util.concurrent.CompletableFuture;public class AsyncResultExample {public static void main(String[] args) {// 创建一个返回结果的异步任务CompletableFuture<Integer> future = CompletableFuture.supplyAsync(() -> {System.out.println("正在计算...");return 10 + 20;});// 处理结果future.thenApply(result -> {System.out.println("计算结果: " + result);return result * 2;}).thenAccept(finalResult -> {System.out.println("最终结果: " + finalResult);});// 主线程继续执行System.out.println("主线程继续运行...");}
}
3.3链式处理任务
CompletableFuture的一个强大功能是它允许链式地组合多个异步任务。每个任务可以依赖于上一个任务的结果,从而形成复杂的异步流程。
import java.util.concurrent.CompletableFuture;public class ChainedTasksExample {public static void main(String[] args) {CompletableFuture.supplyAsync(() -> {// 第一个任务System.out.println("任务1: 获取用户数据...");return "用户数据";}).thenApply(userData -> {// 第二个任务,依赖第一个任务的结果System.out.println("任务2: 处理 " + userData);return "处理后的" + userData;}).thenAccept(finalResult -> {// 第三个任务,最终处理结果System.out.println("任务3: 显示结果 " + finalResult);});System.out.println("主线程继续执行...");}
}
3.4处理异常
异步任务中可能会发生异常。CompletableFuture提供了exceptionally()方法来捕获并处理异常,确保程序不会因为某个异步任务的失败而崩溃。
import java.util.concurrent.CompletableFuture;public class ExceptionHandlingExample {public static void main(String[] args) {CompletableFuture.supplyAsync(() -> {// 模拟异常if (true) {throw new RuntimeException("任务失败了");}return 42;}).exceptionally(ex -> {System.out.println("发生异常: " + ex.getMessage());return -1; // 返回默认值}).thenAccept(result -> {System.out.println("最终结果: " + result);});System.out.println("主线程继续执行...");}
}
4.异步事件驱动模型
除了CompletableFuture,在事件驱动开发中,异步编程也是核心。例如,在处理GUI编程、网络请求、I/O操作时,通常通过异步事件驱动模型来提高系统的响应速度。
在Java中,GUI框架如Swing、JavaFX等,都会使用异步事件模型处理用户输入,避免界面阻塞。
4.1例:JavaFX中的异步事件处理
在JavaFX中,我们可以通过异步任务来避免UI卡顿。例如,当我们需要从网络获取数据时,可以使用异步任务来执行,并在任务完成后更新UI。
import javafx.application.Application;
import javafx.concurrent.Task;
import javafx.scene.Scene;
import javafx.scene.control.Label;
import javafx.scene.layout.StackPane;
import javafx.stage.Stage;public class AsyncJavaFXExample extends Application {@Overridepublic void start(Stage primaryStage) {Label label = new Label("等待中...");StackPane root = new StackPane(label);Scene scene = new Scene(root, 300, 200);primaryStage.setTitle("JavaFX异步任务示例");primaryStage.setScene(scene);primaryStage.show();// 创建异步任务Task<String> task = new Task<String>() {@Overrideprotected String call() throws Exception {// 模拟耗时操作Thread.sleep(2000);return "任务完成";}};// 当任务完成时,更新UItask.setOnSucceeded(event -> label.setText(task.getValue()));// 启动任务new Thread(task).start();}public static void main(String[] args) {launch(args);}
}