进一步探索多线程的实现 | 带你学《Java语言高级特性》之五

简介: Runnable接口中并没有提供返回一个值的功能,本节提到的Callable则对这一问题进行了解决。让我们一起来进一步探索多线程的运行吧。

上一篇:深入剖析Thread与Runnable关系 | 带你学《Java语言高级特性》之四
【本节目标】
通过阅读本节内容,你将了解到Callable的相关功能及其实现方法,对Callable和Runnable之间的区别和联系有一定的认识,了解到多线程在正常运行中的各种状态和状态转变的时机,对多线程有更深入的理解。

Callable接口实现多继承

从最传统的开发来讲如果要进行多线程的实现肯定依靠Runnable,但是Runnable接口有一个缺点,当线程执行完毕之后无法获取一个返回值,所以从JDK1.5后提出了一个新的线程实现接口:java.util.concurrent.Callable接口。首先来观察这个接口的定义:

@FunctionalInterface
public interface Callable<V>{
    public V call() throws Exception;
}

可以发现Callable定义的时候可以设置一个泛型,此泛型的类型就是返回数据的类型,这样的好处在于可以避免向下转型所带来的的安全隐患。

image.png
图一 Callable

范例:使用Callable实现多线程处理

import java.util.concurrent.Callable;
import java.util.concurrent.FutureTask;

class MyThread implements Callable<String> {    //线程的主体类
    @Override
    public String call() throws Exception {
        for (int x = 0; x < 10; x++) {
            System.out.println("*********** 线程执行、x = " + x);
        }
        return "线程执行完毕。";
    }
}
public class ThreadDemo {
    public static void main(String[] args) throws Exception {
        FutureTask<String> task = new FutureTask(new MyThread());
        new Thread(task).start();
        System.out.println("【线程返回数据】" + task.get());
    }
}

image.png
图二 使用Callable实现多线程处理

面试题:请解释Runnable和Callable的区别?

  • Runnable是在JDK1.0时提出的多线程的实现接口,而Callable是JDK1.5后提出的;
  • java.lang.Runnable接口之中只提供一个run()方法,并且没有返回值;
  • java.util.concurrent.Callable接口提供有call()方法,可以有返回值;

多线程运行状态

对于多线程的开发而言,编写程序的过程之中总是按照:定义线程主体类,然后通过Thread类进行线程的启动,但并不意味着调用了start()方法,线程就已经开始运行了,因为整体的线程处理有自己的一套运行的状态。

image.png
图三 线程运行状态

  1. 任何一个线程的对象都应该使用Thread类进行封装,所以线程的启动使用的是start(),但是启动的时候实际上若干个线程都将进入到一种就绪状态,现在并没有执行;
  2. 进入到就绪状态后就需要等待进行资源调度,当某一个线程调度成功之后则进入到运行状态(run()方法),但是所有的线程不可能一直持续执行下去,中间需要产生一些暂停的状态,例如:某个线程执行一段时间之后就将需要让出资源,而后这个线程就将进入到阻塞状态,随后重新回归到就绪状态。
  3. 当run()方法执行完毕之后,实际上该线程的主要任务也就结束了,那么此时就可以直接进入到停止状态。

想学习更多的Java的课程吗?从小白到大神,从入门到精通,更多精彩不容错过!免费为您提供更多的学习资源。
本内容视频来源于阿里云大学

下一篇:线程起名,分工有序 | 带你学《Java语言高级特性》之六
更多Java面向对象编程文章查看此处

相关文章
|
1月前
|
存储 Java 索引
用Java语言实现一个自定义的ArrayList类
自定义MyArrayList类模拟Java ArrayList核心功能,支持泛型、动态扩容(1.5倍)、增删改查及越界检查,底层用Object数组实现,适合学习动态数组原理。
85 4
|
1月前
|
Java
Java语言实现字母大小写转换的方法
Java提供了多种灵活的方法来处理字符串中的字母大小写转换。根据具体需求,可以选择适合的方法来实现。在大多数情况下,使用 String类或 Character类的方法已经足够。但是,在需要更复杂的逻辑或处理非常规字符集时,可以通过字符流或手动遍历字符串来实现更精细的控制。
219 18
|
1月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
135 1
|
1月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
157 1
|
2月前
|
数据采集 存储 弹性计算
高并发Java爬虫的瓶颈分析与动态线程优化方案
高并发Java爬虫的瓶颈分析与动态线程优化方案
Java 数据库 Spring
134 0
|
2月前
|
存储 Java Apache
Java语言操作INI配置文件策略
以上步骤展示了基本策略,在实际项目中可能需要根据具体需求进行调整优化。例如,在多线程环境中操作同一份配置时需要考虑线程安全问题;大型项目可能还需考虑性能问题等等。
164 15
|
2月前
|
算法 Java
Java多线程编程:实现线程间数据共享机制
以上就是Java中几种主要处理多线程序列化资源以及协调各自独立运行但需相互配合以完成任务threads 的技术手段与策略。正确应用上述技术将大大增强你程序稳定性与效率同时也降低bug出现率因此深刻理解每项技术背后理论至关重要.
219 16
|
3月前
|
缓存 并行计算 安全
关于Java多线程详解
本文深入讲解Java多线程编程,涵盖基础概念、线程创建与管理、同步机制、并发工具类、线程池、线程安全集合、实战案例及常见问题解决方案,助你掌握高性能并发编程技巧,应对多线程开发中的挑战。
|
3月前
|
算法 Java
Java语言实现链表反转的方法
这种反转方法不需要使用额外的存储空间,因此空间复杂度为,它只需要遍历一次链表,所以时间复杂度为,其中为链表的长度。这使得这种反转链表的方法既高效又实用。
365 0