阅读 201

callbale 和runnable 区别

callbale 和runnable 区别

Callable接口:

1
2
3
public interface Callable<V> {
    V call() throws Exception;
}

Runnable接口:

1
2
3
public interface Runnable {
    public abstract void run();
}

相同点

  1. 两者都是接口;(废话)

  2. 两者都可用来编写多线程程序;

  3. 两者都需要调用Thread.start()启动线程;

不同点

  1. 两者最大的不同点是:实现Callable接口的任务线程能返回执行结果;而实现Runnable接口的任务线程不能返回结果;

  2. Callable接口的call()方法允许抛出异常;而Runnable接口的run()方法的异常只能在内部消化,不能继续上抛;

注意点

  • Callable接口支持返回执行结果,此时需要调用FutureTask.get()方法实现,此方法会阻塞主线程直到获取‘将来’结果;当不调用此方法时,主线程不会阻塞!

Callable工作的Demo:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
package com.callable.runnable;
 
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;
 
/**
 * Created on Jason 2018/10/12.
 */
public class CallableImpl implements Callable<String> {
 
    public CallableImpl(String acceptStr) {
        this.acceptStr = acceptStr;
    }
 
    private String acceptStr;
 
    @Override
    public String call() throws Exception {
        // 任务阻塞 1 秒
        Thread.sleep(1000);
        return this.acceptStr + " append some chars and return it!";
    }
 
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        Callable<String> callable = new CallableImpl("my callable test!");
        FutureTask<String> task = new FutureTask<>(callable);
        long beginTime = System.currentTimeMillis();
        // 创建线程
        new Thread(task).start();
        // 调用get()阻塞主线程,反之,线程不会阻塞
        String result = task.get();
        long endTime = System.currentTimeMillis();
        System.out.println("hello : " + result);
        System.out.println("cast : " + (endTime - beginTime) / 1000 " second!");
    }
}

测试结果:

1
2
3
4
hello : my callable test! append some chars and return it!
cast : 1 second!
 
Process finished with exit code 0

Runnable工作的Demo:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
package com.callable.runnable;
 
/**
 * Created on Jason 2018/10/12.
 */
public class RunnableImpl implements Runnable {
 
    public RunnableImpl(String acceptStr) {
        this.acceptStr = acceptStr;
    }
 
    private String acceptStr;
 
    @Override
    public void run() {
        try {
            // 线程阻塞 1 秒,此时有异常产生,只能在方法内部消化,无法上抛
            Thread.sleep(1000);
        catch (InterruptedException e) {
            e.printStackTrace();
        }
        // 最终处理结果无法返回
        System.out.println("hello : " this.acceptStr);
    }
 
    public static void main(String[] args) {
        Runnable runnable = new RunnableImpl("my runable test!");
        long beginTime = System.currentTimeMillis();
        new Thread(runnable).start();
        long endTime = System.currentTimeMillis();
        System.out.println("cast : " + (endTime - beginTime) / 1000 " second!");
    }
}

测试结果:

1
2
3
4
cast : 0 second!
hello : my runable test!
 
Process finished with exit code 0

比较了两者后,发现Callable在很多特殊的场景下还是很有用的!最后留点代码,加深对Callable的认识!

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
package com.inte.fork;
 
/**
 * Created on Jason 2018/10/12.
 */
 
import java.util.*;
import java.util.concurrent.*;
 
import static java.util.Arrays.asList;
 
public class Sums {
 
    static class Sum implements Callable<Long> {
        private final long from;
        private final long to;
 
        Sum(long from, long to) {
            this.from = from;
            this.to = to;
        }
 
        @Override
        public Long call() {
            long acc = 0;
            for (long i = from; i <= to; i++) {
                acc = acc + i;
            }
            System.out.println(Thread.currentThread().getName() + " : " + acc);
            return acc;
        }
    }
 
    public static void main(String[] args) throws Exception {
        ExecutorService executor = Executors.newFixedThreadPool(3);
        List<Future<Long>> results = executor.invokeAll(asList(
                new Sum(010), new Sum(0, 1_000), new Sum(0, 1_000_000)
        ));
        executor.shutdown();
 
        for (Future<Long> result : results) {
            System.out.println(result.get());
        }
    }
}

我只是偶尔安静下来,对过去的种种思忖一番。那些曾经的旧时光里即便有过天真愚钝,也不值得谴责。毕竟,往后的日子,还很长。不断鼓励自己, 天一亮,又是崭新的起点,又是未知的征程(上校9) 逆水行舟,不进,则退!

来源https://www.cnblogs.com/lixiaochong/p/14789654.html

文章分类
后端
版权声明:本站是系统测试站点,无实际运营。本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 XXXXXXo@163.com 举报,一经查实,本站将立刻删除。
相关推荐