JDK 5.0 新增解决线程安全 Callable接口和线程池
在jdk5.0后又新增了两种解决线程安全的问题
一: 实现Callable接口,
实现接口步骤:
1: 创建一个实现Callable接口的实现类
2: 实现Callable接口中的call()方法, 讲此线程需要做的操作声明再这个方法中
3: 创建Callable 接口实现类的对象
4: 将创建对象传递到FutureTask构造器中,创建FutureTask对象
5: 将Future的对象作为参数传递到Thread类中 并调用thread的start()方法
6: 通过get()获取call方法中的返回值
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask; public class CallableTestTwo {
public static void main(String[] args) {
CallableT callableT = new CallableT(); // 3 : 实例化 实现Callable接口的类的对象
FutureTask futureTask = new FutureTask(callableT); // 4: 传递此对象到FutureTask 中 并实例化
new Thread(futureTask).start(); // 5: 生产对象传递到Thread 中并调用start()方法来启动线程 try { // 6: get获取call方法中抛出的信息 get方法必须搭配try来使用
Object val = futureTask.get();
System.out.println("总和是: "+val);
} catch (InterruptedException e) {
e.printStackTrace();
} catch (ExecutionException e) {
e.printStackTrace();
}
}
} class CallableT implements Callable { // 1: 实现Calable接口
@Override
public Object call() throws Exception { // 2 :实现接口Callable接口中的call()方法
int num = 0;
for (int i = 0; i <= 100; i++) {
System.out.println(i);
num += i;
}
return num;
}
}
如何理解实现Callable接口的方式比创建多线程和实现Runnable接口的方式强大?
如何理解实现Callable接口的方式比创建多线程和实现Runnable接口的方式强大?
1: call方法可以有返回值
2: call方法可以抛出异常被外面的操作捕获, 获取异常信息
3: Callable是支持泛型的
二: 线程池
JDK 5.0起提供了线程池相关API:ExecutorService 和 Executors
ExecutorService:真正的线程池接口。常见子类ThreadPoolExecutor Executors:工具类、线程池的工厂类,用于创建并返回不同类型的线程池
ExecutorService 不能直接手动建立洗澡池子(线程)池子, 需要用工具一个铁湫之类的工具(Executors)来帮助建立线程.
Executors工具类的方法
Executors.newCachedThreadPool():创建一个可根据需要创建新线程的线程池
Executors.newFixedThreadPool(n); 创建一个可重用固定线程数的线程池
Executors.newSingleThreadExecutor() :创建一个只有一个线程的线程池
Executors.newScheduledThreadPool(n):创建一个线程池,它可安排在给定延迟后运 行命令或者定期地执行。
线程池创建的步骤:
1: ExecutorService 使用工具Executors创建线程池
2: 得到的对象使用对应线程的方法submit()或者execute() 操作对应线程
3: shutdown()停止线程
eg:
public class ExecutorsTest {
public static void main(String[] args) {
// 提供指定线程池的数量
// ExecutorService service = Executors.newFixedThreadPool(10);
//
//执行指定的线程的操作,需要提供实现Runnable或者Callable接口的实现的对象
service.submit(); // 适合使用于Callable
service.execute(); // 适合适用于Runnable
//关闭线程连接
// service.shutdown();
ExecutorService executorService = Executors.newFixedThreadPool(10);
executorService.execute(new RunnableT());
executorService.submit(new RunnableT());
executorService.shutdown();
}
}
import java.util.concurrent.*;
public class CallableTestThree {
public static void main(String[] args) {
ExecutorService executorService = Executors.newFixedThreadPool(10);
RunnableTT runnableTT = new RunnableTT();
CallableTT callableTT = new CallableTT();
executorService.submit(callableTT); // submit可以对Callable 和Runnable接口的实现对象进行操作
executorService.submit(runnableTT);
//executorService.execute(callableTT); // execute()不可以获取 callable接口的实现对象 因为他只能对Runnable接口的实现对象进行操作
executorService.shutdown();
}
}
class RunnableTT implements Runnable{
@Override
public void run() {
System.out.println("这是取偶数");
for (int i = 0; i <= 100 ; i++) {
if(i % 2 == 0){
System.out.println(i);
}
}
}
}
class CallableTT implements Callable{
@Override
public Object call() throws Exception {
System.out.println("这是取奇数");
for (int i = 0; i <= 100 ; i++) {
System.out.println(i);
}
return null;
}
}
Executors工具的操作线程方法一般分为两种
submit(): 一般操作Callable
一般主要是操作Callable接口的线程对象, 偶尔可以用于Runnable接口实现的对象,推荐用Callable接口实现的对象
用来操作实现Callable 接口和Runnable接口的线程对象 , 两者都可以
execute(): 一般操作Runnable
只可以操作实现Runnable的接口线程对象
多线程可以理解为一个公交车 一个人就是线程, 你平时是自己走路快还是坐公交快和便捷呢?
肯定是坐公交,你不需要管线程池的问题 只需要管你自己就好了, 因为池子可以装多个线程公交可以装多个人
JDK 5.0 新增解决线程安全 Callable接口和线程池的更多相关文章
- 创建线程的方式三:实现Callable接口 --- JDK 5.0新增
/** * 创建线程的方式三:实现Callable接口. --- JDK 5.0新增 * * * 如何理解实现Callable接口的方式创建多线程比实现Runnable接口创建多线程方式强大? * 1 ...
- 实现Callable接口创建线程
创建执行线程有四种方式: 实现implements接口创建线程 继承Thread类创建线程 实现Callable接口,通过FutureTask包装器来创建线程 使用线程池创建线程 下面介绍通过实现Ca ...
- 通过Callable接口创建线程
通过Callable接口创建线程 一.前言 Java中创建线程的方式有四中,前两种在前面我已经详细介绍过了(Runnable和Thread),不清楚的朋友们可看这里: Java多线程之线程的启动以及J ...
- 创建线程的方式三:实现Callable接口。 --- JDK 5.0新增
如何理解实现Callable接口的方式创建多线程比实现Runnable接口创建多线程方式强大? call()可以有返回值的.call()可以抛出异常,被外面的操作捕获,获取异常的信息Callable是 ...
- 使用Callable接口创建线程和使用线程池的方式创建线程
1.使用Callable接口的方式实现多线程,这是JDK5.0新增的一种创建多线程的方法 package com.baozi.java2; import java.util.concurrent.Ca ...
- 多线程----Thread类,Runnable接口,线程池,Callable接口,线程安全
1概念 1.1进程 进程指正在运行的程序.确切的来说,当一个程序进入内存运行,即变成一个进程,进程是处于运行过程中的程序,并且具有一定独立功能. 任务管理器中: 1.2线程 线程是进程中的一个执行单元 ...
- Callable接口实现线程
public class CallableDemo { public static void main(String[] args) throws Exception, ExecutionExcept ...
- Java之创建线程的方式三:实现Callable接口
import java.util.concurrent.Callable;import java.util.concurrent.ExecutionException;import java.util ...
- 创建线程的方式三:实现Callable接口-----JDK5.0 新增
package com.yhqtv.java2; /* * 创建线程的方式三:实现Callable接口-----JDK5.0 新增 * * 如何理解实现Callable接口的方式创建多线程比实现Run ...
随机推荐
- PyTorch笔记之 squeeze() 和 unsqueeze()
1.squeeze() 函数 squeeze() 用来去掉向量的一个维度,只有维度为 1 的那一维才能去掉 example: 初始化1个向量shape为(1,2,3)的向量 import torch ...
- 第017讲:函数 - Python的乐高积木
0. 你有听说过DRY吗? me:不知道 参考答案: 1. 都是重复一段代码,为什么我要使用函数(而不使用简单的拷贝黏贴)呢? me:函数可以设置参数. 参考答案:0) 可以降低代码量(调用函数只需要 ...
- VS2015-MFC基础教程-应用程序工程中文件的组成结构
VS2015应用程序向导生成框架程序后,我们可以在之前设置的Location下看到此文件夹中包含了几个文件和一个以工程名命名的子文件夹,这个子文件夹中又包含了若干个文件和一个res文件夹,创建工程时的 ...
- vue.js(19)--vue中子组件调用父组件的方法
子组件是不能直接使用父组件中的方法的,需要进行事件绑定(v-on:自定义方法名="父组件方法名"),然后在子组件中再定义一个方法,使用this.$emit('自定义方法名')语句完 ...
- Nginx针对前端静态资源的缓存处理
当用户上报一个线上的bug后,开发者修改前端代码的bug上新后,用户反映问题依旧存在的问题...这种情况是不是曾经遇到过,这个问题跟浏览器的缓存机制有很大关系(强制缓存和协商缓存,这里我就不介绍具体的 ...
- ReactiveCocoa 之 优雅的 RACCommand
RACCommand 是一个在 ReactiveCocoa 中比较复杂的类,大多数使用 ReactiveCocoa 的人,尤其是初学者并不会经常使用它. 在很多情况下,虽然使用 RACSignal 和 ...
- SCRUM REPORT DIRECTORY
Alpha sprint scrum 1 scrum 2 scrum 3 scrum 4 scrum 5 scrum 6 scrum 7 scrum 8 scrum 9 scrum 10 Beta s ...
- MySQL的删除语句
虽然现在数据库空间越来越大,但处理数据时候还是有要删除的时候,以下整理了一些最常用的删除语句. 分成两种 一个是删除指定数据,另一个删除所有数据. 一.删除指定数据 DELETE FROM 表名 WH ...
- dotnet ef执行报错, VS 2019发布时配置项中的Entity Framework迁移项显示不出来
VS 2019发布时配置项中的Entity Framework迁移项显示不出来 dotnet ef dbcontext list --json “无法执行,因为找不到指定的命令或文件.可能的原因包括: ...
- tmux 操作简版
操作session: 操作window: 操作pane: 原文