Java并发基础:线程的创建
线程的创建和管理:
1.应用Thread类显式创建、管理线程
2.应用Executor创建并管理线程。
定义任务:
无返回的任务:实现Runnable接口并编写run()方法。
有响应的任务:实现Callable接口并编写call()方法。
如下的火箭发射倒计时任务:
public class Lunch implements Runnable {
int countDown = 9;
int taskCount = 0;
int rocketNum = 9;
final int id = taskCount++;
final static Logger log = Logger.getLogger("lavasoft");
Logger log1 = Logger.getLogger("lavasoft");
Logger log2 = Logger.getLogger("lavasoft.blog");
public Lunch(){
}
public Lunch(int countDown,int rocketNum){
this.countDown=countDown;
this.rocketNum=rocketNum;
}
public void status(int rocketNum){
while(countDown>=0){
if(countDown==0){
log.info("火箭"+rocketNum+"发射成功!");
countDown--;
}else{
log.info("火箭"+rocketNum+"将在"+countDown+"秒后发射!");
countDown--;
}
}
}
@Override
public void run() {
// TODO Auto-generated method stub
status(rocketNum);
}
public static void main(String args[]) throws SecurityException, IOException{
FileHandler fileHandler = new FileHandler("C:/testlog.log");
fileHandler.setLevel(Level.INFO);
fileHandler.setFormatter(new MyLogHander());
ExecutorService exec = Executors.newCachedThreadPool();// 需要多少线程创建多少线程,用多少给多少。
ExecutorService exec2 = Executors.newSingleThreadExecutor();// 保证任意时刻任何线程中都只有唯一的任务在运行。
ExecutorService exec3 = Executors.newFixedThreadPool(2);//指定线程数量。
ExecutorService exec4 = Executors.newScheduledThreadPool(2);//可实现周期性或延迟任务
log.addHandler(fileHandler);
for(int rocketNum=1;rocketNum<10;rocketNum++){
//用Thread类显式创建并管理线程
Thread t1 = new Thread(new Lunch(6,rocketNum));
t1.start();
//用Executor类创建并管理线程
exec.execute(new Lunch(6,rocketNum));
exec2.execute(new Lunch(6,rocketNum));
exec3.execute(new Lunch(6,rocketNum));
exec4.execute(new Lunch(6,rocketNum));
log.info("等待火箭"+rocketNum+"发射");
}
exec.shutdown();
}
}
class MyLogHander extends Formatter{
@Override
public String format(LogRecord record) {
return record.getLevel() + ":" + record.getMessage()+"\n";
}
}
public class Lunch implements Callable<String> {
int countDown = 9;
int taskCount = 0;
int rocketNum = 9;
final int id = taskCount++;
final static Logger log = Logger.getLogger("lavasoft");
Logger log1 = Logger.getLogger("lavasoft");
Logger log2 = Logger.getLogger("lavasoft.blog");
public Lunch(){
}
public Lunch(int countDown,int rocketNum){
this.countDown=countDown;
this.rocketNum=rocketNum;
}
public void status(int rocketNum){
while(countDown>=0){
if(countDown==0){
log.info("火箭"+rocketNum+"发射成功!");
countDown--;
}else{
log.info("火箭"+rocketNum+"将在"+countDown+"秒后发射!");
countDown--;
}
}
}
@Override
public String call() {
// TODO Auto-generated method stub
status(rocketNum);
return "***火箭"+rocketNum+"发射完毕***";
}
public static void main(String args[]) throws SecurityException, IOException{
FileHandler fileHandler = new FileHandler("C:/testlog.log");
fileHandler.setLevel(Level.INFO);
fileHandler.setFormatter(new MyLogHander());
ExecutorService exec = Executors.newCachedThreadPool();// 需要多少线程创建多少线程,用多少给多少。
ExecutorService exec2 = Executors.newSingleThreadExecutor();// 保证任意时刻任何线程中都只有唯一的任务在运行。
ExecutorService exec3 = Executors.newFixedThreadPool(2);//指定线程数量。
ExecutorService exec4 = Executors.newScheduledThreadPool(2);//可实现周期性或延迟任务
log.addHandler(fileHandler);
List<Future<String>> list = new ArrayList<Future<String>>();
for(int rocketNum=1;rocketNum<10;rocketNum++){
//用Thread类显式创建并管理线程
Thread t1 = new Thread(new Lunch(6,rocketNum));
t1.start();
//用Executor类创建并管理线程
exec.execute(new Lunch(6,rocketNum));
exec2.execute(new Lunch(6,rocketNum));
exec3.execute(new Lunch(6,rocketNum));
Future<String> a = exec.submit(new Lunch(6,rocketNum));//call方法会返回一个Future对象,与runnable不同,将任务赋予给线程时,必须用submit方法。
list.add(a);
try {
System.out.println(a.get());
} catch (InterruptedException | ExecutionException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
log.info("等待火箭"+rocketNum+"发射");
}
exec.shutdown();
}
}
class MyLogHander extends Formatter{
@Override
public String format(LogRecord record) {
return record.getLevel() + ":" + record.getMessage()+"\n";
}
}
总结:
Callable接口和Runnable接口相似,区别就是Callable需要实现call方法,而Runnable需要实现run方法;并且,call方法还可以返回任何对象,无论是什么对象,JVM都会当作Object来处理。但是如果使用了泛型,我们就不用每次都对Object进行转换了。
Runnable和Callable都是接口
不同之处:
1.Callable可以返回一个类型V,而Runnable不可以
2.Callable能够抛出checked exception,而Runnable不可以。
3.Runnable是自从java1.1就有了,而Callable是1.5之后才加上去的
4.Callable和Runnable都可以应用于executors。而Thread类只支持Runnable.
上面只是简单的不同,其实这两个接口在用起来差别还是很大的。Callable与executors联合在一起,在任务完成时可立刻获得一个更新了的Future。而Runable却要自己处理
Future接口,一般都是取回Callable执行的状态用的。其中的主要方法:
- cancel,取消Callable的执行,当Callable还没有完成时
- get,获得Callable的返回值
- isCanceled,判断是否取消了
- isDone,判断是否完成
用Executor来构建线程池,应该要做的事:
1).调用Executors类中的静态方法newCachedThreadPool(必要时创建新线程,空闲线程会被保留60秒)或newFixedThreadPool(包含固定数量的线程池)等,返回的是一个实现了ExecutorService接口的ThreadPoolExecutor类或者是一个实现了ScheduledExecutorServiece接口的类对象。
2).调用submit提交Runnable或Callable对象。
3).如果想要取消一个任务,或如果提交Callable对象,那就要保存好返回的Future对象。
4).当不再提交任何任务时,调用shutdown方法。
Java并发基础:线程的创建的更多相关文章
- Java并发基础--线程通信
java中实现线程通信的四种方式 1.synchronized同步 多个线程之间可以借助synchronized关键字来进行间接通信,本质上是通过共享对象进行通信.如下: public class S ...
- Java 并发基础——线程安全性
当线程安全:多个线程访问某个类时,不管运行时环境采用何种调度方式或者这些线程将如何交替执行,并且在主调代码中不需要任何额外的同步或协调,这个类都能表现出正确的行为,那么久称这个类是线程安全的. 在线程 ...
- Java并发基础--线程安全
一.线程安全 1.线程安全的概念 线程安全:某个类被单个线程,或者多个线程同时访问,所表现出来的行为是一致,则可以说这个类是线程安全的. 2.什么情况下会出现线程安全问题 在单线程中不会出现线程安全问 ...
- java并发编程 线程基础
java并发编程 线程基础 1. java中的多线程 java是天生多线程的,可以通过启动一个main方法,查看main方法启动的同时有多少线程同时启动 public class OnlyMain { ...
- Java并发编程:如何创建线程?
Java并发编程:如何创建线程? 在前面一篇文章中已经讲述了在进程和线程的由来,今天就来讲一下在Java中如何创建线程,让线程去执行一个子任务.下面先讲述一下Java中的应用程序和进程相关的概念知识, ...
- Java并发编程:线程和进程的创建(转)
Java并发编程:如何创建线程? 在前面一篇文章中已经讲述了在进程和线程的由来,今天就来讲一下在Java中如何创建线程,让线程去执行一个子任务.下面先讲述一下Java中的应用程序和进程相关的概念知识, ...
- 【转】Java并发编程:如何创建线程?
一.Java中关于应用程序和进程相关的概念 在Java中,一个应用程序对应着一个JVM实例(也有地方称为JVM进程),一般来说名字默认是java.exe或者javaw.exe(windows下可以通过 ...
- java并发基础(五)--- 线程池的使用
第8章介绍的是线程池的使用,直接进入正题. 一.线程饥饿死锁和饱和策略 1.线程饥饿死锁 在线程池中,如果任务依赖其他任务,那么可能产生死锁.举个极端的例子,在单线程的Executor中,如果一个任务 ...
- 2、Java并发编程:如何创建线程
Java并发编程:如何创建线程? 在前面一篇文章中已经讲述了在进程和线程的由来,今天就来讲一下在Java中如何创建线程,让线程去执行一个子任务.下面先讲述一下Java中的应用程序和进程相关的概念知识, ...
- Java 并发基础
Java 并发基础 标签 : Java基础 线程简述 线程是进程的执行部分,用来完成一定的任务; 线程拥有自己的堆栈,程序计数器和自己的局部变量,但不拥有系统资源, 他与其他线程共享父进程的共享资源及 ...
随机推荐
- [cf687c]The Values You Can Make(01背包变形)
题意:给定n个硬币,每个硬币都有面值,求每个能构成和为k的硬币组合中,任意个数相互求和的总额种类,然后将所有硬币组合中最后得到的结果输出. 解题关键:在01背包的过程中进行dp.dp[i][j]表示组 ...
- [413D][搜索]D - Field expansion
http://codeforces.com/contest/799/problem/D 解题关键:因为3^11>100000,所以若只把2单独拿出,最多只需要暴力2^11次,故只需要dfs一下即 ...
- 使用JFileChooser打开文件
-----------------siwuxie095 工程名:TestFileChooser 包名:com.siwuxie095.fi ...
- 【spring boot logback】日志颜色渲染,使用logback-spring.xml自定义的配置文件后,日志没有颜色了
接着spring boot日志logback解析之后,发现使用logback-spring.xml自定义的配置文件后,日志没有颜色了 怎么办? 官网处理日志链接:https://logback.qos ...
- hello markdown
目录 标题 标题1 标题2 标题3 有序列表 无序列表 插入图片 插入链接 粗体.斜体.删除线 引用 表格 代码 目录 新的开始 希望能够坚持下去 cnblogs也支持markdown 之前看了写ma ...
- 线程池和Thread
1.线程池 创建线程需要时间.如果有不同的短任务要完成,就可以事先创建许多线程,在应完成这些任务时发出请求.这个线程数最好在需要更多线程时增加,在需要释放资源时减少.不需要自己创建这样一个列表.该列表 ...
- 微信小程序iPhone X空白兼容
开局一张图…… 看看这空白的地方多丑 ~ 接下来就是见证奇迹的时刻(上代码) //app.js App({ onLaunch: function (ops) { if (ops.scene == 10 ...
- Note: File Recipe Compression in Data Deduplication Systems
Zero-Chunk Suppression 检测全0数据块,将其用预先计算的自身的指纹信息代替. Detect zero chunks and replace them with a special ...
- java 多线程学习笔记(二) -- IO密集型任务
IO密集型是指对IO操作较多的任务.下面以查询一些股票价格任务为例: YahooFinance.java public class YahooFinance { public static doubl ...
- 51nod1118(递推)
题目链接: https://www.51nod.com/onlineJudge/questionCode.html#!problemId=1118 题意: 中文题诶~ 思路: 因为机器人只能往下或者右 ...