【异步】Java 的 8 种异步实现方式

异步执行对于开发者来说并不陌生,在实际的开发过程中,很多场景多会使用到异步,相比同步执行,异步可以大大缩短请求链路耗时时间,比如:发送短信、邮件。

异步的八种实现方式:

  1. 线程异步 Thread/Runnable
  2. Future + Callable
  3. 异步框架 CompletableFuture
  4. Spring 注解 @Async
  5. Spring ApplicationEvent 事件
  6. 第三方异步框架,比如 Hutool 的 ThreadUtil
  7. Guava 异步
  8. 消息队列

1、线程异步

public class ThreadTest implements Runnable{

    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName());
    }

    public static void main(String[] args) {
        ThreadTest threadTest = new ThreadTest();
        new Thread(threadTest).start();
    }

}

当然,如果每次都创建一个 Thread 线程,频繁的创建、销毁,浪费系统资源,我们可以采用线程池:【Thread】线程池的 7 种创建方式及自定义线程池

2、Future 异步

public class FutureTest {

    public static void main(String[] args) throws ExecutionException, InterruptedException {
        ExecutorService executor = Executors.newFixedThreadPool(1);
        Future<String> future = executor.submit(() -> {
            Thread.sleep(2000);
            return "this is future execute final result!!!";
        });
        //这里需要返回值时会阻塞主线程
        String result = future.get();
        System.out.println(result);
        executor.shutdown();
    }

}

Future的不足之处的包括以下几点:

  1. 无法被动接收异步任务的计算结果:虽然我们可以主动将异步任务提交给线程池中的线程来执行,但是待异步任务执行结束之后,主线程无法得到任务完成与否的通知,它需要通过get方法主动获取任务执行的结果。
  2. Future件彼此孤立:有时某一个耗时很长的异步任务执行结束之后,你想利用它返回的结果再做进一步的运算,该运算也会是一个异步任务,两者之间的关系需要程序开发人员手动进行绑定赋予,Future并不能将其形成一个任务流(pipeline),每一个Future都是彼此之间都是孤立的,所以才有了后面的CompletableFuture,CompletableFuture就可以将多个Future串联起来形成任务流。
  3. Futrue没有很好的错误处理机制:截止目前,如果某个异步任务在执行发的过程中发生了异常,调用者无法被动感知,必须通过捕获get方法的异常才知晓异步任务执行是否出现了错误,从而在做进一步的判断处理

3、CompletableFuture
关于 CompletableFuture 更多详情请看:【异步】Futurn、FutureTask、CompletionService、CompletableFuture

public static void thenRunAsync() throws ExecutionException, InterruptedException {
    CompletableFuture<Integer> cf1 = CompletableFuture.supplyAsync(() -> {
        System.out.println(Thread.currentThread() + " cf1 do something....");
        return 1;
    });
    CompletableFuture<Void> cf2 = cf1.thenRunAsync(() -> {
        System.out.println(Thread.currentThread() + " cf2 do something...");
    });
    //等待任务1执行完成
    System.out.println("cf1结果->" + cf1.get());
    //等待任务2执行完成
    System.out.println("cf2结果->" + cf2.get());
}

4、Spring 注解 @Async

@Configuration
public class ThreadPoolConfig {

    @Bean("taskExecutor")
    public Executor taskExecutor() {
        //返回可用处理器的Java虚拟机的数量 12
        int i = Runtime.getRuntime().availableProcessors();
        System.out.println("系统最大线程数  : " + i);
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        //核心线程池大小
        executor.setCorePoolSize(16);
        //最大线程数
        executor.setMaxPoolSize(20);
        //配置队列容量,默认值为Integer.MAX_VALUE
        executor.setQueueCapacity(99999);
        //活跃时间
        executor.setKeepAliveSeconds(60);
        //线程名字前缀
        executor.setThreadNamePrefix("asyncServiceExecutor -");
        //设置此执行程序应该在关闭时阻止的最大秒数,以便在容器的其余部分继续关闭之前等待剩余的任务完成他们的执行
        executor.setAwaitTerminationSeconds(60);
        //等待所有的任务结束后再关闭线程池
        executor.setWaitForTasksToCompleteOnShutdown(true);
        return executor;
    }

}
@Service
@EnableAsync
public class AsyncServiceImpl implements AsyncService {

    @Override
    @Async("taskExecutor")
    public String sendSms() {
        System.out.println(Thread.currentThread().getName());
        return null;
    }

    @Override
    @Async("taskExecutor")
    public String sendEmail() {
        System.out.println(Thread.currentThread().getName());
        return null;
    }

}

在实际项目中, 使用 @Async 调用线程池,推荐等方式是是使用自定义线程池的模式,不推荐直接使用 @Async 直接实现异步

5、Spring ApplicationEvent 事件

Spring 中使用事件只需要以下的几个步骤:

  1. 定义事件,继承 ApplicationEvent
  2. 定义监听,要么实现 ApplicationListener 接口,要么在方法上添加 @EventListener 注解
  3. 定义发布事件接口,调用 ApplicationContext.publishEvent() 或者 ApplicationEventPublisher.publishEvent();
  4. 业务调用发布事件
@Getter
@Setter
public class BaseEvent<T> extends ApplicationEvent {

    private T data;

    public BaseEvent(Object source) {
        super(source);
    }

    public BaseEvent(Object source, T data) {
        super(source);
        this.data = data;
    }

}
@Component
public class BaseEventListener implements ApplicationListener<BaseEvent<UserVo>> {

    @Override
    @Async("taskExecutor")
    public void onApplicationEvent(BaseEvent<UserVo> baseEvent) {
        UserVo eventData = baseEvent.getData();
        // TODO 业务处理
    }

}
@Autowired
private ApplicationContext applicationContext;

@GetMapping("/pubEvent")
public void pubEvent() {
    BaseEvent<UserVo> baseEvent = new BaseEvent<>("event", new UserVo());
    applicationContext.publishEvent(baseEvent);
}

6、Hutool 的 ThreadUtil

public static void main(String[] args) {
    for (int i = 0; i < 3; i++) {
        ThreadUtil.execAsync(() -> {
            ThreadLocalRandom threadLocalRandom = ThreadLocalRandom.current();
            int number = threadLocalRandom.nextInt(20) + 1;
            System.out.println(number);
        });
        log.info("当前第:" + i + "个线程");
    }

    log.info("task finish!");
}

7、 Guava 异步

public static void test() {
    ListeningExecutorService executorService = MoreExecutors.listeningDecorator(Executors.newCachedThreadPool());
    final ListenableFuture<Integer> listenableFuture = executorService.submit(() -> {
        log.info("callable execute...");
        TimeUnit.SECONDS.sleep(1);
        return 1;
    });

    Futures.addCallback(listenableFuture, new FutureCallback<Integer>() {
        @Override
        public void onSuccess(@Nullable Integer integer) {
            System.out.println("Get listenable future's result with callback " + integer);
        }
        @Override
        public void onFailure(Throwable throwable) {
            throwable.printStackTrace();
        }
    }, Executors.newCachedThreadPool());
}

8、 消息队列

常用的消息队列:RabbitMqRocketMq

  • 3
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java 中使用 gRPC 传输文件的异步方式,需要借助 gRPC 的流式传输能力。 以下是一个基本的 gRPC 文件传输的异步实现示例: 首先,在.proto文件中定义文件传输的方法: ``` service FileTransferService { rpc uploadFile(stream FileChunk) returns (UploadStatus); } message FileChunk { bytes data = 1; } message UploadStatus { bool success = 1; } ``` 接下来,在Java实现异步上传文件的客户端: ``` // 定义异步上传文件的回调接口 public interface FileUploadCallback { void onSuccess(); void onError(Throwable t); } // 客户端异步上传文件 public class FileUploadClient { private final FileTransferServiceStub stub; public FileUploadClient(FileTransferServiceStub stub) { this.stub = stub; } public void uploadFile(File file, FileUploadCallback callback) { try { FileInputStream inputStream = new FileInputStream(file); StreamObserver<FileChunk> requestObserver = stub.uploadFile(new StreamObserver<UploadStatus>() { @Override public void onNext(UploadStatus status) { if (status.getSuccess()) { callback.onSuccess(); } else { callback.onError(new RuntimeException("Upload failed")); } } @Override public void onError(Throwable t) { callback.onError(t); } @Override public void onCompleted() { // Do nothing } }); byte[] buffer = new byte[4096]; int bytesRead = -1; while ((bytesRead = inputStream.read(buffer)) != -1) { FileChunk chunk = FileChunk.newBuilder().setData(ByteString.copyFrom(buffer, 0, bytesRead)).build(); requestObserver.onNext(chunk); } requestObserver.onCompleted(); } catch (IOException e) { callback.onError(e); } } } ``` 最后,在Java实现异步接收文件的服务端: ``` // 服务端异步接收文件 public class FileTransferServiceImpl extends FileTransferServiceGrpc.FileTransferServiceImplBase { @Override public StreamObserver<FileChunk> uploadFile(StreamObserver<UploadStatus> responseObserver) { return new StreamObserver<FileChunk>() { private FileOutputStream outputStream; @Override public void onNext(FileChunk chunk) { try { if (outputStream == null) { File tempFile = File.createTempFile("temp", "upload"); outputStream = new FileOutputStream(tempFile); } outputStream.write(chunk.getData().toByteArray()); } catch (IOException e) { onError(e); } } @Override public void onError(Throwable t) { if (outputStream != null) { try { outputStream.close(); } catch (IOException e) { // Ignore } } responseObserver.onError(t); } @Override public void onCompleted() { try { outputStream.close(); responseObserver.onNext(UploadStatus.newBuilder().setSuccess(true).build()); responseObserver.onCompleted(); } catch (IOException e) { onError(e); } } }; } } ``` 以上就是一个基本的 gRPC 文件传输的异步实现示例。需要注意的是,在实际应用中,可能需要对上传和下载的文件进行分片,以便更好地管理和控制传输过程。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值