Java多线程基础-16:简述Java并发编程JUC中的Callable接口

简介: Callable接口是Java中用于描述带有返回值任务的接口,与Runnable相对,后者无返回值。Callable的call()方法用于执行具体任务并返回结果。

一、什么是Callable接口?


Callable 和 Runnable 相对,都是描述一个 “任务”。Callable 描述的是带有返回值的任务,而Runnable 描述的是不带返回值的任务。


可以把Runnable想象成一个没有参数和返回值的异步方法,而Callable与Runnable类似,但是是有返回值的,方便程序员借助多线程的方式计算结果。


Callable 接口是一个函数式接口,只有一个方法 call():



类型参数V就是call方法返回值的类型。例如,Callable<Integer>就表示一个最终返回Integer对象的异步计算:


        Callable<Integer> callable = new Callable<Integer>() {
            @Override
            public Integer call() throws Exception {
                ...
            }
        };


我们来看一下Callable的简单使用。


二、Callable接口的简单使用


代码示例:创建一个线程,计算 1 + 2 + 3 + ... + 1000,使用 Callable 实现。


首先,创建一个匿名内部类,并实现 Callable 接口。Callable 是带有泛型参数的,泛型参数就表示返回值的类型。这里的泛型参数用Integer。然后重写 Callable 的 call 方法,完成累加的过程,直接通过返回值返回计算结果。


        Callable<Integer> callable = new Callable<Integer>() {
            @Override
            public Integer call() throws Exception {
                int sum = 0;
                for (int i = 1; i <= 1000; i++) {
                    sum += i;
                }
                return sum;
            }
        };


当然,由于Callable是一个函数式接口,也可以用lambda表达式的方式来定义:


1.       Callable<Integer> callable = () -> {
            int sum = 0;
            for (int i = 1; i <= 1000; i++) {
                sum += i;
            }
            return sum;
        };


创建好callable任务后,需要一个线程来启动。


注意,这里并不是在构造Thread时直接将callable传入,而是要先通过FutureTask包装一下,再将FutureTask传入Thread的构造方法。


Future


Future接口代表一个异步计算的结果,可以在后台线程中进行计算,而不会阻塞当前线程。其中的 get() 方法可以获取这个结果,而且它的调用会阻塞,直到计算完成(类似于 join() )。如果运行该计算的线程被中断,get() 方法将抛出InterruptedException。如果计算已经完成,那么get 方法立即返回。


FutureTask


执行Callable的一种方法是使用FutureTask,它实现了Future和Runnable接口,所以可以构造一个线程来运行这个任务。


创建线程 t,在线程 t 的构造方法中传入 FutureTask。此时 t 就会执行 FutureTask 内部的 Callable 的 call 方法,完成计算。最终计算结果会存到 futureTask 中。


         FutureTask<Integer> futureTask = new FutureTask<>(callable);
        Thread t = new Thread(futureTask);


如何理解FutureTask?


可以把它理解成吃麻辣烫用到的“小票”。想象去吃麻辣烫,当餐点好后后厨就开始做了,同时前台会给你一张 “小票” 。这个小票就是FutureTask。 它意味着后面我们可以随时凭这张小票去查看自己的这份麻辣烫做出来了没。



最后,在主线程中调用 futureTask.get() ,获取到 FutureTask 中的结果。如何保证主线程中调用 get() 的时候,t 线程已经执行完了呢?由于FutureTask实现了Future接口,因此它的get()方法重写于Future中的get()方法,可以阻塞等待 t 线程的任务完成后,再获取结果。


完整代码


import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;
 
public class Test {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        Callable<Integer> callable = () -> {
            int sum = 0;
            for (int i = 1; i <= 1000; i++) {
                sum += i;
            }
            return sum;
        };
 
        FutureTask<Integer> futureTask = new FutureTask<>(callable);
        Thread t = new Thread(futureTask);
        t.start();
 
        System.out.println(futureTask.get());
    }
}


三、总结:Callable


1、理解Callable


Callable 和 Runnable 相对,都是描述一个 “任务”。 Callable 描述的是带有返回值的任务,而Runnable 描述的是不带返回值的任务。Callable 通常需要搭配 FutureTask 来使用,FutureTask 用来保存 Callable 的返回结果。因为 Callable 往往是在另一个线程中执行的。执行完成的时间并不确定,FutureTask 就可以负责这个等待结果出来的工作。


2、创建线程的方式汇总


在🔗Thread类及其用法 一文中,曾介绍过几种常见的线程创建的方式。这里将Callable也补充进去:


(1)继承Thread类


通过继承Thread类并重写run()方法来创建一个新线程。


class MyThread extends Thread {
    public void run() {
        // 线程执行的代码逻辑
        System.out.println("Thread running!");
    }
}
 
public class ThreadExample {
    public static void main(String[] args) {
        MyThread thread = new MyThread();
        thread.start(); // 启动线程
    }
}


(2)实现Runnable接口(匿名内部类同理)


实现Runnable接口并重写run()方法,然后通过将实现了Runnable接口的对象传递给Thread类的构造方法来创建线程。


class MyRunnable implements Runnable {
    public void run() {
        // 线程执行的代码逻辑
        System.out.println("Thread running!");
    }
}
 
public class RunnableExample {
    public static void main(String[] args) {
        MyRunnable myRunnable = new MyRunnable();
        Thread thread = new Thread(myRunnable);
        thread.start(); // 启动线程
    }
}


(2)实现Runnable接口(匿名内部类同理)


实现Runnable接口并重写run()方法,然后通过将实现了Runnable接口的对象传递给Thread类的构造方法来创建线程。

class MyRunnable implements Runnable {
    public void run() {
        // 线程执行的代码逻辑
        System.out.println("Thread running!");
    }
}
 
public class RunnableExample {
    public static void main(String[] args) {
        MyRunnable myRunnable = new MyRunnable();
        Thread thread = new Thread(myRunnable);
        thread.start(); // 启动线程
    }
}


匿名内部类:


public class Test {
    public static void main(String[] args) {
        // 匿名内部类 Runnable
        // 注意:Runnable实例作为Thread构造器的参数传入
        Thread t = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("i am t!");
            }
        });
 
        t.start();
 
        // main 线程中的方法
        System.out.println("i am main!");
    }
}


(3)使用lambda表达式(最常用)


public class Test {
    public static void main(String[] args) {
        // lambda表达式
        Thread t  = new Thread(() -> {
            System.out.println("i am t!");
        });
 
        t.start();
 
        // main 线程中的方法
        System.out.println("i am main!");
    }
}


(4)实现Callable


public class Test {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        Callable<Integer> callable = () -> {
            int sum = 0;
            for (int i = 1; i <= 1000; i++) {
                sum += i;
            }
            return sum;
        };
 
        FutureTask<Integer> futureTask = new FutureTask<>(callable);
        Thread t = new Thread(futureTask);
        t.start();
 
        System.out.println(futureTask.get());
    }
}
相关文章
|
5月前
|
数据采集 JSON Java
Java爬虫获取1688店铺所有商品接口数据实战指南
本文介绍如何使用Java爬虫技术高效获取1688店铺商品信息,涵盖环境搭建、API调用、签名生成及数据抓取全流程,并附完整代码示例,助力市场分析与选品决策。
|
4月前
|
算法 安全 Java
除了类,Java中的接口和方法也可以使用泛型吗?
除了类,Java中的接口和方法也可以使用泛型吗?
186 11
|
3月前
|
Java Go 开发工具
【Java】(9)抽象类、接口、内部的运用与作用分析,枚举类型的使用
抽象类必须使用abstract修饰符来修饰,抽象方法也必须使用abstract修饰符来修饰,抽象方法不能有方法体。抽象类不能被实例化,无法使用new关键字来调用抽象类的构造器创建抽象类的实例。抽象类可以包含成员变量、方法(普通方法和抽象方法都可以)、构造器、初始化块、内部类(接 口、枚举)5种成分。抽象类的构造器不能用于创建实例,主要是用于被其子类调用。抽象类中不一定包含抽象方法,但是有抽象方法的类必定是抽象类abstract static不能同时修饰一个方法。
250 0
|
5月前
|
存储 缓存 安全
Java集合框架(二):Set接口与哈希表原理
本文深入解析Java中Set集合的工作原理及其实现机制,涵盖HashSet、LinkedHashSet和TreeSet三大实现类。从Set接口的特性出发,对比List理解去重机制,并详解哈希表原理、hashCode与equals方法的作用。进一步剖析HashSet的底层HashMap实现、LinkedHashSet的双向链表维护顺序特性,以及TreeSet基于红黑树的排序功能。文章还包含性能对比、自定义对象去重、集合运算实战和线程安全方案,帮助读者全面掌握Set的应用与选择策略。
375 23
|
5月前
|
存储 安全 Java
Java集合框架(一):List接口及其实现类剖析
本文深入解析Java中List集合的实现原理,涵盖ArrayList的动态数组机制、LinkedList的链表结构、Vector与Stack的线程安全性及其不推荐使用的原因,对比了不同实现的性能与适用场景,帮助开发者根据实际需求选择合适的List实现。
|
3月前
|
Java
如何在Java中进行多线程编程
Java多线程编程常用方式包括:继承Thread类、实现Runnable接口、Callable接口(可返回结果)及使用线程池。推荐线程池以提升性能,避免频繁创建线程。结合同步与通信机制,可有效管理并发任务。
208 6
|
6月前
|
Java API 微服务
为什么虚拟线程将改变Java并发编程?
为什么虚拟线程将改变Java并发编程?
361 83
|
3月前
|
Java 调度 数据库
Python threading模块:多线程编程的实战指南
本文深入讲解Python多线程编程,涵盖threading模块的核心用法:线程创建、生命周期、同步机制(锁、信号量、条件变量)、线程通信(队列)、守护线程与线程池应用。结合实战案例,如多线程下载器,帮助开发者提升程序并发性能,适用于I/O密集型任务处理。
411 0
|
8月前
|
机器学习/深度学习 消息中间件 存储
【高薪程序员必看】万字长文拆解Java并发编程!(9-2):并发工具-线程池
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的强力并发工具-线程池,废话不多说让我们直接开始。
335 0
|
4月前
|
算法 Java
Java多线程编程:实现线程间数据共享机制
以上就是Java中几种主要处理多线程序列化资源以及协调各自独立运行但需相互配合以完成任务threads 的技术手段与策略。正确应用上述技术将大大增强你程序稳定性与效率同时也降低bug出现率因此深刻理解每项技术背后理论至关重要.
386 16