创建多线程的方式三:实现Callable接口

实现Callable接口
(有泛型,不加编译器也不会报错)
与使用Runnable相比, Callable功能更强大:
1)相比run()方法没有返回值,需要重写call方法,call方法可以有返回值,分线程做完之后还可以返回一个结果给另外的线程用
2)方法可以抛出异常 ,抛出的异常可以给外面接受,接收之后就知道线程出什么问题。run方法只能使用try,catch的方式解决异常
3)支持泛型的返回值
4) 想使用需要借助FutureTask类,比如获取返回结果

Future接口
1)可以对具体Runnable、Callable任务的执行结果进行取消、查询是否完成、获取结果等
2)FutrueTask是Futrue接口的唯一实现类
3)FutureTask同时实现了Runnable, Future接口。它既可以作为Runnable被线程执行,又可以作为Future得到Callable的返回值

步骤:

  • 1.创建实现Callable接口的实现类
  • 2.实现call方法,将此线程需要做的操作声明在call当中,如果不想返回东西,就return null
  • 3.创建Callable实现类的对象
  • 4.创建FutureTask对象,将创建好的Callable实现类对象作为参数传到FutureTask构造器当中
  • 5.将创建好的FutureTask对象作为参数传给Thread类的构造器,创建Thread类的对象,调用start方法
  • 6.(此步骤可省略)如果对Callable中call的返回值感兴趣,在没有使用泛型的情况下可以用Object对象变量接受FutureTask的对象.get()的结果

说明: 有的时候,一个线程做完以后需要给另外一个线程提供数据,另一个线程拿到数据之后才继续执行,这就需要get方法, get():返回值即为FutureTask构造器参数Callable实现类重写的call的返回值

如何理解实现Callable接口创建多线程的方式比实现Runnable接口创建多线程的方式强大?

  • 1)call()有返回值,分线程执行完后可以给另一个线程返回结果
  • 2)call()可以抛出异常,在调用get方法处就可以捕获异常,根据捕获的异常,可以拿到感兴趣的信息,知道出了什么问题
  • 3)Callable支持泛型,如果用了泛型(泛型不能用基本类型),比如这么干implements Callable< Integer>,那么call的返回值类型就必须写成Integer
    但不能这么写:Integer sum=f.get();此时仍认为是Object类型,因为FutureTask也有泛型,这么干才可以:
    FutureTask< Integer> f = new FutureTask<>(m);
    Integer sum=f.get();
package test0;

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

public class ThreadDemo {
    public static void main(String[] args) {
        myThread m = new myThread();
        //不能直接调用start,它也没有start方法,要想使用要借助FutureTask类(有泛型)(涉及到Future接口)
        FutureTask f = new FutureTask(m);//注意:通过Alt+ctrl+/可以查看它的构造器信息,其中有一个构造器需要传一个Callable对象

        //要想启动线程必须new Thread,前面两种多线程创建方式的实质都是new Thread,因为要去调用start方法
        //利用FutureTask这个类也实现了Runnable接口,所以也可以当做参数传给Thread的构造器

        Thread t=new Thread(f);
        t.start();//启动线程,也可以写成new Thread(f).start();然后就会执行call方法
        
        try {//如果对于call的返回值不感兴趣,就不用调用get(),get的目的只是获取call方法的返回值,和启动线程无关
            Object sum=f.get();//call方法的返回值就是get的返回值
            System.out.println("总和为:"+sum);
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (ExecutionException e) {
            e.printStackTrace();
        }//try,catch都是在主线程中做的
    }
}
class myThread implements Callable{

    //call就相当于run一样
    @Override
    //遍历100以内的偶数,并返回和
    public Object call() throws Exception {
        int sum=0;
        for(int i=1;i<=100;i++){
            System.out.println(i);
            sum+=i;
        }
        return sum;//Object是类层面的,int是基本数据类型,不存在继承关系,这里会自动装箱Integer,把Integer赋给了Object类型,涉及到多态
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值