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());
    }
}
相关文章
|
7月前
|
存储 缓存 Java
【高薪程序员必看】万字长文拆解Java并发编程!(5):深入理解JMM:Java内存模型的三大特性与volatile底层原理
JMM,Java Memory Model,Java内存模型,定义了主内存,工作内存,确保Java在不同平台上的正确运行主内存Main Memory:所有线程共享的内存区域,所有的变量都存储在主存中工作内存Working Memory:每个线程拥有自己的工作内存,用于保存变量的副本.线程执行过程中先将主内存中的变量读到工作内存中,对变量进行操作之后再将变量写入主内存,jvm概念说明主内存所有线程共享的内存区域,存储原始变量(堆内存中的对象实例和静态变量)工作内存。
240 0
|
11月前
|
监控 Kubernetes Java
阿里面试:5000qps访问一个500ms的接口,如何设计线程池的核心线程数、最大线程数? 需要多少台机器?
本文由40岁老架构师尼恩撰写,针对一线互联网企业的高频面试题“如何确定系统的最佳线程数”进行系统化梳理。文章详细介绍了线程池设计的三个核心步骤:理论预估、压测验证和监控调整,并结合实际案例(5000qps、500ms响应时间、4核8G机器)给出具体参数设置建议。此外,还提供了《尼恩Java面试宝典PDF》等资源,帮助读者提升技术能力,顺利通过大厂面试。关注【技术自由圈】公众号,回复“领电子书”获取更多学习资料。
|
12月前
|
安全 Java API
java如何请求接口然后终止某个线程
通过本文的介绍,您应该能够理解如何在Java中请求接口并根据返回结果终止某个线程。合理使用标志位或 `interrupt`方法可以确保线程的安全终止,而处理好网络请求中的各种异常情况,可以提高程序的稳定性和可靠性。
195 6
|
安全 Java
在 Java 中使用实现 Runnable 接口的方式创建线程
【10月更文挑战第22天】通过以上内容的介绍,相信你已经对在 Java 中如何使用实现 Runnable 接口的方式创建线程有了更深入的了解。在实际应用中,需要根据具体的需求和场景,合理选择线程创建方式,并注意线程安全、同步、通信等相关问题,以确保程序的正确性和稳定性。
502 11
|
Java 开发者
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
170 4
|
Java
java线程接口
Thread的构造方法创建对象的时候传入了Runnable接口的对象 ,Runnable接口对象重写run方法相当于指定线程任务,创建线程的时候绑定了该线程对象要干的任务。 Runnable的对象称之为:线程任务对象 不是线程对象 必须要交给Thread线程对象。 通过Thread的构造方法, 就可以把任务对象Runnable,绑定到Thread对象中, 将来执行start方法,就会自动执行Runable实现类对象中的run里面的内容。
118 1
|
Java
为什么一般采用实现Runnable接口创建线程?
因为使用实现Runnable接口的同时我们也能够继承其他类,并且可以拥有多个实现类,那么我们在拥有了Runable方法的同时也可以使用父类的方法;而在Java中,一个类只能继承一个父类,那么在继承了Thread类后我们就不能再继承其他类了。
142 0
|
存储 Java
高并发编程之多线程锁和Callable&Future 接口
高并发编程之多线程锁和Callable&Future 接口
206 1
|
并行计算 Java 大数据
Callable和Future
Callable和Future
|
11月前
|
Java 程序员
Java社招面试中的高频考点:Callable、Future与FutureTask详解
大家好,我是小米。本文主要讲解Java多线程编程中的三个重要概念:Callable、Future和FutureTask。它们在实际开发中帮助我们更灵活、高效地处理多线程任务,尤其适合社招面试场景。通过 Callable 可以定义有返回值且可能抛出异常的任务;Future 用于获取任务结果并提供取消和检查状态的功能;FutureTask 则结合了两者的优势,既可执行任务又可获取结果。掌握这些知识不仅能提升你的编程能力,还能让你在面试中脱颖而出。文中结合实例详细介绍了这三个概念的使用方法及其区别与联系。希望对大家有所帮助!
544 60

热门文章

最新文章