java基础-多线程一
什么是线程
说到线程就不得不说下进程了, 大家都知道,许许多多的线程组合在一起就成了一个进程,进程是由操作系统进行资源操作的一个最小的单位,线程则是比进程更小的实际执行操作的单位;每个线程都有自己的堆栈及变量空间。
就好比回家,各种回家线路组成了一个回家的进程,每条线路则代表一个单一的线程
线程的生命周期
- 新创建(New)
- 就绪(Ready)
- 运行(Running)
- 阻塞(Blocked)
- 销毁(Dead)
线程的实现方式
继承Thread重写run方法
public class MyThread extends Thread{ private String threadName;
// 控制循环的数量
private int num = 1000; // 定义商品总数
private int goodsNum = 15; MyThread(String name) {
this.threadName = name;
} @Override
public void run() {
for (int i = 0; i< num; i++) {
if (this.goodsNum > 0) {
this.goodsNum -- ;
System.out.println(threadName + "购买了一个,剩余" + this.goodsNum);
}
}
} public static void main (String[] args) { MyThread mt = new MyThread("线程1");
MyThread mt1 = new MyThread("线程2");
mt.start();
mt1.start();
}
}
继承Thread需要重写run方法,方法体中就是线程具体的运行逻辑,下面就是代码运行的结果:

有趣的是2个线程都执行了15次,为啥会出现这样的问题?回到代码中
我们是通过new了2个类对象来创建的线程;创建的时候,2个线程都被分配了一样的资源就会导致出现这样 的问题。
优化如下:public class MyThread2{
class TestThread extends Thread { private int goodsNumber = 15; @Override
public void run() {
for (int i = 0; i <= 16; i++) {
if (this.goodsNumber > 0) {
System.out.println(Thread.currentThread().getName()
+ "购买了一个,剩余:" + (this.goodsNumber --));
}
}
}
} public static void main(String[] args) throws InterruptedException {
// 开启一个线程任务
TestThread tt = new MyThread2().new TestThread();
// 将任务分配给2个线程去执行
Thread t1 = new Thread(tt, "线程1");
Thread t2 = new Thread(tt, "线程2");
t1.start();
t2.start();
}
}
结果:

备注:直接new一个线程调用run方法并不会创建一个新线程去执行,只是普通对象的方法调用。切记使用start来启动线程
实现Runnable接口
根据查看Runnable的源码,Runnable只提供一个run方法,并没有start方法,所以Runnable的启动执行还是需要借助Thread类
public class MyThread3 implements Runnable { private int num = 5; @Override
public void run() {
for (int i = 0; i< 10; i++) {
if (this.num > 0) {
System.out.println(Thread.currentThread().getName() + "num left:" + this.num --);
}
}
} public static void main(String[] args) {
// 创建线程任务
MyThread3 m = new MyThread3();
// 将任务分配给3个线程去执行
new Thread(m, "Thread1").start(); // 线程start后,会放置线程等待队列,等待CPU调度,通过Thread调用run方法
new Thread(m, "Thread2").start();
new Thread(m, "Thread3").start();
}
}
Callable和Future的使用
Callable和Runnable有点类似,都是只提供了一个对外接口,不同的是Callable有返回值。
一般Callable搭配ExecutorService来使用;
通过查看源码可以知道ExecutorService可以用来启动Runnable和Callable
// 启动Callable线程
<T> Future<T> submit(Callable<T> task);
// 启动Runnable线程
Future<?> submit(Runnable task);
Future就是对于具体的Runnable或者Callable任务的执行结果进行取消、查询是否完成、获取结果。必要时可以通过get方法获取执行结果,该方法会阻塞直到任务返回结果。
package com.hervey.thread; import java.util.concurrent.*; /**
* @program: thread
* @description: Callable线程
* @author: hetangyuese
* @create: 2019-08-26 14:51
**/
public class ThreadCallable { private int num = 10; /**
* 实现Callable的线程
*/
class TestCallable implements Callable<String> {
@Override
public String call() throws Exception {
Thread.sleep(5000);
return "hello";
}
} /**
* 实现Runnable的线程
*/
class TestRunnable implements Runnable {
@Override
public void run() {
int sleepTime = 1000;
try {
Thread.sleep(sleepTime);
} catch (InterruptedException e) {
e.printStackTrace();
} System.out.println("start");
}
} public static void main(String[] args) throws ExecutionException, InterruptedException {
// 创建线程池 数量为10
ExecutorService es = Executors.newFixedThreadPool(10);
// Callable线程可以有返回值
Future<String> future = es.submit(new ThreadCallable().new TestCallable());
// 在调用get方法的时候会阻塞线程池知道获取到了才会继续执行
System.out.println(future.get());
try {
// 用于取消线程任务,传参表示是否取消正在执行的线程
future.cancel(true);
if (!future.isCancelled()) {
// 此处通过设置超时时间,在一秒后如果future还未获取到返回值则直接结束
System.out.println(future.get(10000, TimeUnit.MILLISECONDS));
}
} catch (TimeoutException e) {
e.printStackTrace();
}
// isDone方法判断future是否完成即有没有获取到返回值
System.out.println(future.isDone()); // 执行Runnable线程
es.submit(new ThreadCallable().new TestRunnable()); if (!es.isShutdown()) {
es.shutdown();
}
}
}
java基础-多线程一的更多相关文章
- java基础---->多线程之Runnable(一)
java线程的创建有两种方式,这里我们通过简单的实例来学习一下.一切都明明白白,但我们仍匆匆错过,因为你相信命运,因为我怀疑生活. java中多线程的创建 一.通过继承Thread类来创建多线程 pu ...
- java基础---->多线程之synchronized(六)
这里学习一下java多线程中的关于synchronized的用法.我来不及认真地年轻,待明白过来时,只能选择认真地老去. synchronized的简单实例 一. synchronized在方法上的使 ...
- java基础---->多线程之wait和notify(八)
这里学习一下java多线程中的关于wait方法和notify方法的用法.命运不是风,来回吹,命运是大地,走到哪你都在命运中. wait和notify方法的使用 一.wait与notify的简单实例 i ...
- java基础---->多线程之ThreadLocal(七)
这里学习一下java多线程中的关于ThreadLocal的用法.人时已尽,人世还长,我在中间,应该休息. ThreadLocal的简单实例 一.ThreadLocal的简单使用 package com ...
- java基础---->多线程之interrupt(九)
这里我们通过实例来学习一下java多线程中关于interrupt方法的一些知识.执者失之.我想当一个诗人的时候,我就失去了诗,我想当一个人的时候,我就失去了我自己.在你什么也不想要的时候,一切如期而来 ...
- java基础---->多线程之yield(三)
yield方法的作用是放弃当前的CPU资源,将它让给其它的任务去占用CPU执行时间.但放弃的时间不确定,有可能刚刚放弃,马上又获得CPU时间片.今天我们通过实例来学习一下yield()方法的使用.最是 ...
- java基础---->多线程之Daemon(五)
在java线程中有两种线程,一种是用户线程,另一种是守护线程.守护线程是一种特殊的线程,当进程中不存在非守护线程了,则守护线程自动销毁.今天我们通过实例来学习一下java中关于守护线程的知识.我是个平 ...
- java基础---->多线程之priority(四)
线程的priority能告诉调度程序其重要性如何,今天我们通过实例来学习一下java多线程中的关于优先级的知识.我从没被谁知道,所以也没被谁忘记.在别人的回忆中生活,并不是我的目的. java多线程的 ...
- Java基础-进程与线程之Thread类详解
Java基础-进程与线程之Thread类详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.进程与线程的区别 简而言之:一个程序运行后至少有一个进程,一个进程中可以包含多个线程 ...
随机推荐
- C语言和内存
1.程序的运行 对cpu来说,内存只是一个存放指令和数据的地方,具体的运算在cpu内完成. 1.寄存器(Register) 是CPU内部非常小.非常快速的存储部件,它的容量很有限,对于32位的CPU, ...
- ES 22 - Elasticsearch中如何进行日期(数值)范围查询
目录 1 范围查询的符号 2 数值范围查询 3 时间范围查询 3.1 简单查询示例 3.2 关于时间的数学表达式(date-math) 3.3 关于时间的四舍五入 4 日期格式化范围查询(format ...
- 主机cpu突然飙高,如何快速排查问题
[问题发现] 使用zabbix软件监控服务器时发现cpu突然异常,在业务主机上使用top命令查看系统的整体运行情况,使用top命令后发现mysqld占用CPU特别高,初步判断可能是mysqld出现问题 ...
- java性能优化--字符串优化处理
String对象 String对象是java中重要的数据类型,在大部分情况下我们都会用到String对象.其实在Java语言中,其设计者也对String做了大量的优化工作,这些也是String对象的特 ...
- Java_转换流和缓冲流
今日内容介绍 转换流 缓冲流 1 转换流 在学习字符流(FileReader.FileWriter)的时候,其中说如果需要指定编码和缓冲区大小时,可以在字节流的基础上,构造一个InputStreamR ...
- 从零开始react实战:云书签-1 react环境搭建
总览篇:react 实战之云书签 本篇是实战系列的第一篇,主要是搭建 react 开发环境,在create-react-app的基础上加上如下功能: antd 组件库按需引入 ,支持主题定制 支持 l ...
- 10分钟了解分布式CAP、BASE理论
CAP理论 2000年7月,Eric Brewer教授提出CAP猜想:2年后,Seth Gilbert和Nancy Lynch从理论上证明了CAP:之后,CAP理论正式成为分布式计算领域的公认定理. ...
- Appium+python自动化(二十七)-让你在手机找到溜冰一样的感觉666,溜得飞起来 - 低级滑动(超详解)
简介 随着现在智能手机的普及和应用,小到五六岁或者更小的娃娃,老至七八十岁老头老太太都是智能手机的用户,基本上达到每个人都在用,每次在地铁或者公交上,就看看到这样的场面,手指不停地在手机屏幕上来来回回 ...
- Oracle GoldenGate mysql To Kafka上车记录
一.前言 首先要学习一下ogg的所有进程,看着这张图来学习 Manager进程是GoldenGate的控制进程,运行在源端和目标端上.它主要作用有以下几个方面:启动.监控.重启Goldengate ...
- Eclipse+CXF框架开发Web服务实战
一. 说明 采用CXF框架开发webservice. 所用软件及版本如下. 操作系统:Window XP SP3. JDK:JDK1.6.0_07,http://www.oracle.com/ ...