Java 使用轮询获取线程返回数据

简介: 本文目录1. 多线程的特点2. 使用轮询强制等待3. 轮询的弊端4. 小结

1. 多线程的特点

一般程序中,只有一个主线程,自上而下顺序执行。

但是多线程程序中,多个线程的执行是并发的,程序员无法在设计时预测各个线程的执行结束时间。

看下面的线程类,用来读取文件大小,然后把结果放到线程局部变量中。

/**
* 读取文件大小
*/
public class ReadFileRunnable implements Runnable {
   /**
    * 文件名
    */
   private String fileName;
   public ReadFileRunnable(String fileName) {
    this.fileName = fileName;
   }
   /**
    * 文件大小,默认为-2
    */
   private long length = -2;
   public long getLength() {
    return length;
   }
   @Override
   public void run() {
    File f = new File(fileName);
    if (f.exists() && f.isFile()) {
      this.length = f.length();
    } else {
      this.length = -1;// 文件不存在
    }
   }
}
如果按照一般的思维,我们先启动线程获取文件大小,然后输出文件大小。
  public static void main(String[] args) {
    // 启动线程
    ReadFileRunnable writeFileRunnable = new ReadFileRunnable("D:\\temp\\1.txt");
    Thread thread = new Thread(writeFileRunnable);
    thread.start();
    // 输出结果
    System.out.println("length:" + writeFileRunnable.getLength());// 输出-2
  }
运行多次,输出均为-2,这是因为我们启动的线程尚未执行完成,下面输出结果的语句就已经执行了。这是因为我们启动的线程需要读取文件,属于IO操作,速度肯定是比较慢的。
2. 使用轮询强制等待
最简单的解决办法,就是使用轮询,一直检查线程执行的结果。
  public static void main(String[] args) {
    // 启动线程
    ReadFileRunnable writeFileRunnable = new ReadFileRunnable("D:\\temp\\1.txt");
    Thread thread = new Thread(writeFileRunnable);
    thread.start();
    // 一直等待
    while (true) {
      if (writeFileRunnable.getLength() == -2) {// 无结果
        continue;
      } else {
        System.out.println("length:" + writeFileRunnable.getLength());// 输出-2
        break;// 直到获取结果结束,此时可以输出文件大小
      }
    }
  }
3. 轮询的弊端
使用轮询,看似解决了问题,但是实际上非常浪费性能。我们测试下:
  public static void main(String[] args) {
    // 启动线程
    ReadFileRunnable writeFileRunnable = new ReadFileRunnable("D:\\temp\\1.txt");
    Thread thread = new Thread(writeFileRunnable);
    thread.start();
    // 一直等待
    int i = 0;
    while (true) {
      i++;
      if (writeFileRunnable.getLength() == -2) {// 无结果
        continue;
      } else {
        System.out.println("循环执行次数:" + i);// 循环执行次数:14365
        System.out.println("length:" + writeFileRunnable.getLength());// 输出-2
        break;// 直到获取结果结束
      }
    }
  }
也就是说,我们检查了一万多次,只有一次是有效的,这不得不说是极大的浪费啊。
4. 小结
多线程程序中,可以使用轮询获取线程返回的数据,但是非常浪费性能,这种方法一般不推荐。
相关文章
|
6天前
|
存储 监控 Java
【Java并发】【线程池】带你从0-1入门线程池
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是编写高端CRUD应用。2025年我正在沉淀中,博客更新速度加快,期待与你一起成长。 线程池是一种复用线程资源的机制,通过预先创建一定数量的线程并管理其生命周期,避免频繁创建/销毁线程带来的性能开销。它解决了线程创建成本高、资源耗尽风险、响应速度慢和任务执行缺乏管理等问题。
115 60
【Java并发】【线程池】带你从0-1入门线程池
|
2天前
|
Java 调度
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
当我们创建一个`ThreadPoolExecutor`的时候,你是否会好奇🤔,它到底发生了什么?比如:我传的拒绝策略、线程工厂是啥时候被使用的? 核心线程数是个啥?最大线程数和它又有什么关系?线程池,它是怎么调度,我们传入的线程?...不要着急,小手手点上关注、点赞、收藏。主播马上从源码的角度带你们探索神秘线程池的世界...
30 0
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
|
17天前
|
数据采集 JSON Java
Java爬虫获取微店快递费用item_fee API接口数据实现
本文介绍如何使用Java开发爬虫程序,通过微店API接口获取商品快递费用(item_fee)数据。主要内容包括:微店API接口的使用方法、Java爬虫技术背景、需求分析和技术选型。具体实现步骤为:发送HTTP请求获取数据、解析JSON格式的响应并提取快递费用信息,最后将结果存储到本地文件中。文中还提供了完整的代码示例,并提醒开发者注意授权令牌、接口频率限制及数据合法性等问题。
|
7天前
|
SQL 数据建模 BI
【YashanDB 知识库】用 yasldr 配置 Bulkload 模式作单线程迁移 300G 的业务数据到分布式数据库,迁移任务频繁出错
问题描述 详细版本:YashanDB Server Enterprise Edition Release 23.2.4.100 x86_64 6db1237 影响范围: 离线数据迁移场景,影响业务数据入库。 外场将部分 NewCIS 的报表业务放到分布式数据库,验证 SQL 性能水平。 操作系统环境配置: 125G 内存 32C CPU 2T 的 HDD 磁盘 问题出现的步骤/操作: 1、部署崖山分布式数据库 1mm 1cn 3dn 单线启动 yasldr 数据迁移任务,设置 32 线程的 bulk load 模式 2、观察 yasldr.log 是否出现如下错
|
18天前
|
Java 程序员 开发者
Java社招面试题:一个线程运行时发生异常会怎样?
大家好,我是小米。今天分享一个经典的 Java 面试题:线程运行时发生异常,程序会怎样处理?此问题考察 Java 线程和异常处理机制的理解。线程发生异常,默认会导致线程终止,但可以通过 try-catch 捕获并处理,避免影响其他线程。未捕获的异常可通过 Thread.UncaughtExceptionHandler 处理。线程池中的异常会被自动处理,不影响任务执行。希望这篇文章能帮助你深入理解 Java 线程异常处理机制,为面试做好准备。如果你觉得有帮助,欢迎收藏、转发!
79 14
|
21天前
|
安全 Java 程序员
Java 面试必问!线程构造方法和静态块的执行线程到底是谁?
大家好,我是小米。今天聊聊Java多线程面试题:线程类的构造方法和静态块是由哪个线程调用的?构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节有助于掌握Java多线程机制。下期再见! 简介: 本文通过一个常见的Java多线程面试题,详细讲解了线程类的构造方法和静态块是由哪个线程调用的。构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节对掌握Java多线程编程至关重要。
51 13
|
2月前
|
存储 NoSQL Java
使用Java和Spring Data构建数据访问层
本文介绍了如何使用 Java 和 Spring Data 构建数据访问层的完整过程。通过创建实体类、存储库接口、服务类和控制器类,实现了对数据库的基本操作。这种方法不仅简化了数据访问层的开发,还提高了代码的可维护性和可读性。通过合理使用 Spring Data 提供的功能,可以大幅提升开发效率。
68 21
|
22天前
|
安全 Java 开发者
【JAVA】封装多线程原理
Java 中的多线程封装旨在简化使用、提高安全性和增强可维护性。通过抽象和隐藏底层细节,提供简洁接口。常见封装方式包括基于 Runnable 和 Callable 接口的任务封装,以及线程池的封装。Runnable 适用于无返回值任务,Callable 支持有返回值任务。线程池(如 ExecutorService)则用于管理和复用线程,减少性能开销。示例代码展示了如何实现这些封装,使多线程编程更加高效和安全。
|
25天前
|
缓存 安全 Java
面试中的难题:线程异步执行后如何共享数据?
本文通过一个面试故事,详细讲解了Java中线程内部开启异步操作后如何安全地共享数据。介绍了异步操作的基本概念及常见实现方式(如CompletableFuture、ExecutorService),并重点探讨了volatile关键字、CountDownLatch和CompletableFuture等工具在线程间数据共享中的应用,帮助读者理解线程安全和内存可见性问题。通过这些方法,可以有效解决多线程环境下的数据共享挑战,提升编程效率和代码健壮性。
58 6
|
12天前
|
Java API 数据处理
深潜数据海洋:Java文件读写全面解析与实战指南
通过本文的详细解析与实战示例,您可以系统地掌握Java中各种文件读写操作,从基本的读写到高效的NIO操作,再到文件复制、移动和删除。希望这些内容能够帮助您在实际项目中处理文件数据,提高开发效率和代码质量。
19 0