当前位置: 首页 > news >正文

Java高频面试之并发编程-11

hello啊,各位观众姥爷们!!!本baby今天又来报道了!哈哈哈哈哈嗝🐶

面试官:父子线程如何共享数据?

在Java中,父子线程共享数据可以通过以下几种方式实现,具体选择取决于应用场景和需求:


1. 通过共享对象成员变量

父线程和子线程共享同一个对象的成员变量,需使用同步机制确保线程安全。

class SharedData {private int value;public synchronized void setValue(int value) { this.value = value; }public synchronized int getValue() { return value; }
}public class Main {public static void main(String[] args) {SharedData data = new SharedData();data.setValue(100);Thread childThread = new Thread(() -> {System.out.println("子线程读取数据: " + data.getValue()); // 输出 100});childThread.start();}
}

注意事项

  • 使用 synchronizedLock 确保原子性。
  • 使用 volatile 保证可见性(适用于简单变量的读写)。

2. 通过构造器参数传递初始数据

父线程在创建子线程时,通过构造器或 Runnable 传递数据。

class ChildThread implements Runnable {private final String message;public ChildThread(String message) {this.message = message;}@Overridepublic void run() {System.out.println("子线程接收消息: " + message);}
}public class Main {public static void main(String[] args) {String message = "Hello from parent";Thread childThread = new Thread(new ChildThread(message));childThread.start();}
}

适用场景:初始化时传递数据,后续无动态更新。


3. 使用线程安全的数据结构

通过 ConcurrentHashMapBlockingQueue 等并发容器共享数据。

import java.util.concurrent.ConcurrentHashMap;public class Main {private static ConcurrentHashMap<String, String> sharedMap = new ConcurrentHashMap<>();public static void main(String[] args) {sharedMap.put("key", "初始值");Thread childThread = new Thread(() -> {sharedMap.put("key", "子线程修改后的值");});childThread.start();try {childThread.join(); // 等待子线程结束} catch (InterruptedException e) {e.printStackTrace();}System.out.println("父线程读取数据: " + sharedMap.get("key")); // 输出 "子线程修改后的值"}
}

优点:无需显式同步,高并发性能好。


4. 使用 InheritableThreadLocal

子线程继承父线程的线程局部变量(适用于传递初始化值)。

public class Main {private static InheritableThreadLocal<String> inheritableThreadLocal = new InheritableThreadLocal<>();public static void main(String[] args) {inheritableThreadLocal.set("父线程设置的值");Thread childThread = new Thread(() -> {System.out.println("子线程读取数据: " + inheritableThreadLocal.get()); // 输出 "父线程设置的值"});childThread.start();}
}

限制:子线程创建后,父线程对 InheritableThreadLocal 的修改不会影响子线程。


5. 使用 FutureCallable

父线程通过 Future 获取子线程的执行结果。

import java.util.concurrent.*;public class Main {public static void main(String[] args) throws ExecutionException, InterruptedException {ExecutorService executor = Executors.newSingleThreadExecutor();Future<Integer> future = executor.submit(() -> {return 42; // 子线程计算结果});int result = future.get(); // 父线程阻塞等待结果System.out.println("父线程获取结果: " + result); // 输出 42executor.shutdown();}
}

适用场景:需要异步获取子线程执行结果。


6. 使用回调(Callback)机制

子线程完成任务后通过回调接口通知父线程。

interface Callback {void onComplete(String result);
}class ChildThread implements Runnable {private final Callback callback;public ChildThread(Callback callback) {this.callback = callback;}@Overridepublic void run() {String result = "处理完成";callback.onComplete(result);}
}public class Main {public static void main(String[] args) {new Thread(new ChildThread(result -> {System.out.println("父线程接收回调结果: " + result); // 输出 "处理完成"})).start();}
}

总结与选型建议

方式适用场景线程安全要求灵活性
共享对象成员变量简单数据共享,需频繁更新高(需同步)
构造器参数传递初始化时传递数据低(仅初始化)
线程安全数据结构高并发环境下的数据共享低(容器内部已处理)
InheritableThreadLocal传递线程局部初始化值低(仅初始化)
Future/Callable异步获取子线程结果无(结果单向传递)
回调机制异步通知父线程低(回调方法需线程安全)

注意事项

  • 可见性与原子性:共享变量需使用 volatile 或同步机制确保可见性,复合操作需保证原子性。
  • 资源释放:使用线程池时,确保及时关闭并清理资源(如 ExecutorService.shutdown())。
  • 避免死锁:合理设计锁的获取顺序,避免嵌套锁竞争。

在这里插入图片描述

http://www.xdnf.cn/news/216649.html

相关文章:

  • 第三部分:赋予网页灵魂 —— JavaScript(下)
  • Spring Boot - 配置管理与自动化配置进阶
  • 【Bash】可以请您解释性地说明一下“2>1”这个语法吗?
  • Windows 系统下使用 Docker 搭建Redis 集群(6 节点,带密码)
  • C++日更八股--first
  • SpringBoot应用:Docker与Kubernetes全栈实战秘籍
  • git fetch和git pull的区别
  • 域对齐是什么
  • 判断用户选择的Excel单元格区域是否跨页?
  • 力扣hot100——239.滑动窗口最大值
  • 在大数据环境下,使用spingboot为Android APP推送数据方案
  • 【Machine Learning Q and AI 读书笔记】- 02 自监督学习
  • 主流微前端框架比较
  • java面试题目
  • Nacos源码—2.Nacos服务注册发现分析四
  • 三种机器学习类型
  • Glide 如何加载远程 Base64 图片
  • MobileNetV2: 反向残差和线性瓶颈
  • 应急演练考试排查-DC01
  • 【动态导通电阻】GaN功率器件中动态导通电阻退化的机制、表征及建模方法
  • AI 的未来是开源?DeepSeek 正在书写新篇章!
  • 算法基础学习|02归并排序——分治
  • 封装js方法 构建树结构和扁平化树结构
  • 20_大模型微调和训练之-基于LLamaFactory+LoRA微调LLama3后格式合并
  • 水力压裂多裂缝扩展诱发光纤应变演化试验研究
  • 基于Mamba2的文本生成实战
  • 什么是 MCP?AI 应用的“USB-C”标准接口详解
  • AI赋能的问答系统:2025年API接口实战技巧
  • Vulkan与OpenGL的对比
  • 服务器主动发送响应?聊天模块如何实现?