李锋镝的博客

  • 首页
  • 时间轴
  • 评论区显眼包🔥
  • 左邻右舍
  • 博友圈
  • 关于我
    • 关于我
    • 另一个网站
    • 我的导航站
    • 网站地图
    • 赞助
  • 留言
  • 🚇开往
Destiny
自是人生长恨水长东
  1. 首页
  2. 代码人生
  3. 正文

学艺不精啊,踩了一个Lambda的一个小坑,记录下

2025年7月25日 143点热度 1人点赞 1条评论

先上代码:

package com.example.demo;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.atomic.AtomicReference;

/**
 * @ClassName: AsyncTest
 * @Author: Dylan Li
 * @Date: 2025/7/25 17:43
 */
public class AsyncTest {
    public static void main(String[] args) {
        ExecutorService executor = Executors.newSingleThreadExecutor();

        AtomicReference<String> str = new AtomicReference<>();
        List<CompletableFuture<Void>> futures = new ArrayList<>();
        CompletableFuture<Void> future = CompletableFuture.runAsync(() -> new ContextAwareRunnable(
                "context()",
                () -> {
                    str.set("Hello World.");
                    System.out.println("set success");
                }
        ), executor);
        futures.add(future);

        CompletableFuture.allOf(futures.toArray(new CompletableFuture[0])).join();

        System.out.println("str: " + str.get());
        // 关闭线程池
        executor.shutdown();
    }

    static class ContextAwareRunnable implements Runnable {
        private final Object context;
        private final Runnable task;

        public ContextAwareRunnable(Object context, Runnable task) {
            this.context = context;
            this.task = task;
        }

        @Override
        public void run() {
            // 设置上下文(关键步骤)
            setContext(context);
            try {
                task.run(); // 执行传入的业务逻辑
            } finally {
                removeContext();
            }
        }

        private void setContext(Object context) {
            System.out.println("设置上下文:" + context);
        }

        private void removeContext() {
            System.out.println("移除上下文");
        }
    }
}

猜一下执行结果是什么?

是不是觉得很简单,打印结果肯定是:

设置上下文:context()
set success
移除上下文
str: Hello World.

如果这样想,那么恭喜你……

实际上的执行结果是:

str: null

原因很简单:

Lambda表达式的主体是new ContextAwareRunnable(...) —— 这仅仅是创建了一个Runnable实例,而没有触发它的执行逻辑。因此需要显式调用run()方法,否则ContextAwareRunnable中封装的业务逻辑永远不会执行。

具体来说:
CompletableFuture.runAsync(Runnable task) 要求传入的 task 是一个可执行的 Runnable。在上面的代码中,Lambda 表达式 () -> new ContextAwareRunnable(...) 本身是一个 Runnable,但它的逻辑只是“创建对象”,而非“执行对象的逻辑”。如果不调用 run(),这个 ContextAwareRunnable 实例就只是被创建出来,其内部的业务代码不会被触发。

解决的办法有两个:

  1. 移除Lambda表达式,代码如下:

    CompletableFuture<Void> future = CompletableFuture.runAsync(new ContextAwareRunnable(
        "context()",
        () -> {
            str.set("Hello World.");
            System.out.println("set success");
        }
    ), executor);
    futures.add(future);
  2. 调用run()方法:

    CompletableFuture<Void> future = CompletableFuture.runAsync(() -> new ContextAwareRunnable(
        "context()",
        () -> {
            str.set("Hello World.");
            System.out.println("set success");
        }
    ).run(), executor);
    futures.add(future);

运行代码查看执行结果:

设置上下文:context()
set success
移除上下文
str: Hello World.
除非注明,否则均为李锋镝的博客原创文章,转载必须以链接形式标明本文链接

本文链接:https://www.lifengdi.com/dai-ma-ren-sheng/4494

相关文章

  • 项目中慎用 CompletableFuture:这些坑你必须知道
  • CompletableFuture使用详解
  • AI时代,个人技术博客的出路在哪里?
  • WordPress网站换了个字体,差点儿把样式换崩了
  • TIOBE 12月榜单:C#有望摘得年度语言,R语言重返Top 10
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可
标签: CompletableFuture Lambda Runnable 线程池
最后更新:2025年11月20日

李锋镝

既然选择了远方,便只顾风雨兼程。

打赏 点赞
< 上一篇
下一篇 >

文章评论

  • 信息发布黑铁

    再接再厉

    Windows
    Chrome 109.0.0.0 中国-江苏-苏州
    2025年7月30日
    回复
  • 1 2 3 4 5 6 7 8 9 11 12 13 14 15 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 46 47 48 49 50 51 52 53 54 55 57 58 60 61 62 63 64 65 66 67 69 72 74 76 77 78 79 80 81 82 85 86 87 90 92 93 94 95 96 97 98 99
    取消回复

    寻寻觅觅,冷冷清清,凄凄惨惨戚戚。乍暖还寒时候,最难将息。三杯两盏淡酒,怎敌他、晚来风急!雁过也,正伤心,却是旧时相识。
    满地黄花堆积,憔悴损,如今有谁堪摘?守着窗儿,独自怎生得黑!梧桐更兼细雨,到黄昏、点点滴滴。这次第,怎一个愁字了得!

    那年今日(01月25日)

    • 1979年:中国左翼文学运动开创者之一郑伯奇逝世
    • 1949年:日本帝国时期的政治家牧野伸显逝世
    • 1924年:第一届奥林匹克冬季运动会在夏蒙尼开幕
    • 1911年:中国第一部专门刑法典颁布
    • 1504年:意大利艺术家米开朗基罗完成大卫雕像
    • 更多历史事件
    最新 热点 随机
    最新 热点 随机
    AI时代,个人技术博客的出路在哪里? 什么是Meta Server? 千万级大表新增字段实战指南:告别锁表与业务中断 在 SQL 中做范围查询时,使用 BETWEEN AND 和直接用 >/=/ 深度解析 Disruptor:无锁队列的高性能实现与实践 精通Linux根目录:核心文件夹深度解析与实战指南
    玩博客的人是不是越来越少了?准备入手个亚太的ECS,友友们有什么建议吗?AI时代,个人技术博客的出路在哪里?使用WireGuard在Ubuntu 24.04系统搭建VPNWordPress实现用户评论等级排行榜插件WordPress网站换了个字体,差点儿把样式换崩了
    基于Java8的Either类 居家办公了~ 祝大家六一儿童节快乐~~~ IntelliJ IDEA 2020.3.x永久白嫖(Windows/Mac) 看病难~取药难~~ 睡觉睡不踏实
    标签聚合
    WordPress K8s 分布式 AI编程 多线程 设计模式 JAVA ElasticSearch Redis SpringBoot JVM SQL AI docker IDEA 架构 MySQL 日常 数据库 Spring
    友情链接
    • Blogs·CN
    • Honesty
    • Mr.Sun的博客
    • 临窗旋墨
    • 哥斯拉
    • 彬红茶日记
    • 志文工作室
    • 懋和道人
    • 搬砖日记
    • 旧时繁华
    • 林羽凡
    • 瓦匠个人小站
    • 皮皮社
    • 知向前端
    • 蜗牛工作室
    • 韩小韩博客
    • 风渡言

    COPYRIGHT © 2026 lifengdi.com. ALL RIGHTS RESERVED.

    域名年龄

    Theme Kratos Made By Dylan

    津ICP备2024022503号-3

    京公网安备11011502039375号