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

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

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

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


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

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

java 复制代码
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 传递数据。

java 复制代码
class ChildThread implements Runnable {
    private final String message;
    public ChildThread(String message) {
        this.message = message;
    }
    @Override
    public 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 等并发容器共享数据。

java 复制代码
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

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

java 复制代码
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 获取子线程的执行结果。

java 复制代码
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); // 输出 42
        executor.shutdown();
    }
}

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


6. 使用回调(Callback)机制

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

java 复制代码
interface Callback {
    void onComplete(String result);
}

class ChildThread implements Runnable {
    private final Callback callback;
    public ChildThread(Callback callback) {
        this.callback = callback;
    }
    @Override
    public 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())。
  • 避免死锁:合理设计锁的获取顺序,避免嵌套锁竞争。
相关推荐
lee_curry2 小时前
第四章 jvm中的垃圾回收器
java·jvm·垃圾收集器
九转成圣3 小时前
Java 性能优化实战:如何将海量扁平数据高效转化为类目字典树?
java·开发语言·json
wxy不爱写代码3 小时前
C++多线程
面试·职场和发展
SmartRadio3 小时前
ESP32-S3 双模式切换实现:兼顾手机_路由器连接与WiFi长距离通信
开发语言·网络·智能手机·esp32·长距离wifi
laowangpython3 小时前
Rust 入门:GitHub 热门内存安全编程语言
开发语言·其他·rust·github
我叫汪枫3 小时前
在后台管理系统中,如何递归和选择保留的思路来过滤菜单
开发语言·javascript·node.js·ecmascript
_.Switch3 小时前
东方财富股票数据JS逆向:secids字段和AES加密实战
开发语言·前端·javascript·网络·爬虫·python·ecmascript
软件技术NINI3 小时前
webkit简介及工作流程
开发语言·前端·javascript·udp·ecmascript·webkit·yarn
Brendan_0013 小时前
JavaScript的Stomp.over
开发语言·javascript·ecmascript
念2343 小时前
f5 shape分析
开发语言·javascript·ecmascript