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天前
|
Java
并发编程的艺术:Java线程与锁机制探索
【6月更文挑战第21天】**并发编程的艺术:Java线程与锁机制探索** 在多核时代,掌握并发编程至关重要。本文探讨Java中线程创建(`Thread`或`Runnable`)、线程同步(`synchronized`关键字与`Lock`接口)及线程池(`ExecutorService`)的使用。同时,警惕并发问题,如死锁和饥饿,遵循最佳实践以确保应用的高效和健壮。
17 2
|
4天前
|
存储 缓存 Java
并发编程-Java内存模型到底是什么
并发编程-Java内存模型到底是什么
|
1天前
|
Java 程序员 调度
Java并发编程之Executor框架深度解析
【6月更文挑战第24天】在Java的并发编程领域,Executor框架是处理多线程任务的核心。本文将深入探讨Executor框架的设计哲学、核心组件以及如何高效利用这一框架来提升程序的性能和响应性。我们将通过实例演示如何正确配置和使用Executor,并讨论常见的陷阱与最佳实践。
|
3天前
|
Java
Java并发编程:深入理解synchronized与ReentrantLock
【6月更文挑战第22天】本文将深入探讨Java并发编程中两个重要的同步机制:synchronized关键字和ReentrantLock类。我们将通过实例分析它们之间的差异,以及在实际应用中如何根据场景选择恰当的同步工具。
|
2天前
|
Java
Java中,有两种主要的方式来创建和管理线程:`Thread`类和`Runnable`接口。
【6月更文挑战第24天】Java创建线程有两种方式:`Thread`类和`Runnable`接口。`Thread`直接继承受限于单继承,适合简单情况;`Runnable`实现接口可多继承,利于资源共享和任务复用。推荐使用`Runnable`以提高灵活性。启动线程需调用`start()`,`Thread`直接启动,`Runnable`需通过`Thread`实例启动。根据项目需求选择适当方式。
12 2
|
2天前
|
Java
Java并发编程中锁的释放
Java并发编程中锁的释放
12 1
|
6天前
|
安全 Java 开发者
深入理解Java内存模型(JMM)及其对并发编程的影响
【6月更文挑战第19天】在Java的世界中,内存模型是构建高效、线程安全应用的基石。本文将通过探讨Java内存模型(JMM)的核心概念和原理,揭示它如何影响并发编程实践。我们将从JMM的基本定义出发,逐步解析它在同步机制、可见性规则以及有序性保证方面的作用。同时,我们也将讨论JMM对现代Java开发中常见的并发模式和框架的影响。最后,文章会提供一些实际的编码建议和最佳实践,帮助开发者更好地利用JMM来设计并发程序。
|
5天前
|
存储 安全 算法
Java并发编程中的线程安全性与性能优化
在Java编程中,特别是涉及并发操作时,线程安全性及其与性能优化是至关重要的问题。本文将深入探讨Java中线程安全的概念及其实现方式,以及如何通过性能优化策略提升程序的并发执行效率。
9 1
|
5天前
|
Java
多线程同步新姿势:Lock接口助你“一统江湖”!
【6月更文挑战第20天】Java多线程高手之路,不仅要懂`synchronized`,还要精通`Lock`接口。`Lock`自Java 5起提供更灵活的同步,包括可中断、超时等待和公平/非公平锁。`ReentrantLock`是重要实现,支持重入,适用于复杂场景。通过显式`lock()`和`unlock()`管理锁,避免异常导致的死锁。`Condition`接口实现精确线程控制,如生产者-消费者模式。掌握这些,你将在Java并发世界中游刃有余。
|
19小时前
|
缓存 监控 安全
深入理解Java中的线程池和并发编程
深入理解Java中的线程池和并发编程

热门文章

最新文章