实现线程的两种方式

上一节我们了解了关于线程的一些基本知识,下面我们正式进入多线程的实现环节。实现线程常用的有两种方式,一种是继承Thread类,一种是实现Runnable接口。当然还有第三种方式,那就是通过线程池来生成线程,后面我们还会学习,一步一个脚印打好基础。

Runnable接口:

public interface Runnable {
public abstract void run();
}

Thread类:

public class Thread implements Runnable {
public synchronized void start() {
if (threadStatus != 0)
throw new IllegalThreadStateException();
group.add(this);
boolean started = false;
try {
start0();
started = true;
} finally {
try {
if (!started) {
group.threadStartFailed(this);
}
} catch (Throwable ignore) {
/* do nothing. If start0 threw a Throwable then
it will be passed up the call stack */
}
}
@Override
public void run() {
if (target != null) {
target.run();
}
}
}

上面为Thread类和Runnable类的源码,我们可以看到Thread类也是实现了Runnable接口,即Thread是Runnable的实现,那么他们到底在实现多线程上有什么区别呢?

Thread和Runnable解析:

①Runnable接口:

Runnable接口是java中线程的定义类。所有线程都是通过该接口来实现,该接口中的run()方法为实现方法,即线程所要实现的内容写入该方法里面,当线程启动时会调用该方法。

在大多数情况下,如果只想重写run()方法而不重写其他方法,应使用Runnable接口。

public class ThreadDemo3 {
public static void main(String[] args) {
//new了两个线程对象——s1和s2
//其中两个对象各对应一个内存区域。线程运行过程中运行都是自己内存块中的数据
Shop1 s1 = new Shop1("小武");
s1.start(); Shop1 s2 = new Shop1("小潘");
s2.start();
/*
//实例化了两个线程对象,所以分配了两块内存空间
//执行过程中操作的是自己的内存空间
Shop2 s3 = new Shop2("小武");
s3.run();
Shop2 s4 = new Shop2("小潘");
s4.run();
//实际实例化了两个线程对象
//所以同样分配两个内存空间
Thread t1 = new Thread(new Shop2("小武"));
t1.start(); Thread t2 = new Thread(new Shop2("小潘"));
t2.start();
*/
//创建了两个线程对象,但是使用的是同一个对象——s6
Shop2 s5 = new Shop2("w");
Shop1 s6 = new Shop1("T");
Thread t3 = new Thread(s6);
t3.start(); Thread t4 =new Thread(s6);
t4.start();
}
} /**
* 武大郎卖烧饼(因为业务的拓展,现在可以实现多窗口的出售)
* 要求:每天只卖10个
*
*/
class Shop1 extends Thread{
//private int count = 10;
//使用静态变量可以有效的实现资源共享(因为在内存中只有一份count)
private static int count = 10;
public Shop1(String name) {
super(name);
}
public void run(){
//判断是否已经卖完
while(count>0){
count--;
System.out.println(this.getName() +"卖出了一个烧饼" + ",现在剩余" + count);
}
}
} /**
* 使用接口实现上面的代码
*
*/
class Shop2 implements Runnable{
//私有变量,存储剩余烧饼的个数
private int count = 10;
//存储当前人的姓名
private String name=""; public Shop2(String name) {
this.name = name;
} /**
* 实现销售的方法
*/
public void run(){
//判断是否已经卖完
while(count>0){
count--;
System.out.println(Thread.currentThread().getId() + "、" + this.name +"卖出了一个烧饼" + ",现在剩余" + count);
}
}
}

②Thread类:

Thread类是Runnable接口的实现,jdk给我们提供了一个不用我们去想如何实现线程的方式供我们使用。同样你在继承Thread类的时候也需要重写run()方法来实现你想在线程中实现的内容。

public class Test{
public static void main(String[] args) {
//传统方式——单任务方式
/*
SimpleClass sc1 = new SimpleClass();
sc1.say("Mike"); SimpleClass sc2 = new SimpleClass();
sc2.say("Han Meimei");
*/
//创建一个线程
ThreadClass tc1 = new ThreadClass("Mike");
//启动线程
tc1.start();
//创建一个线程
ThreadClass tc2 = new ThreadClass("Han Meimei");
tc2.start(); }
}
} class SimpleClass{
public void say(String name){
while(true){
System.out.println("Hi,Im " + name);
}
}
}
class ThreadClass extends Thread{
public ThreadClass(String name) {
super(name);
} /**
* 将父类(Thread)的run()方法进行重写
* 在run()方法中包含了需要执行的代码
*/
public void run(){
while(true){
System.out.println("Hi,Im " + this.getName() + "|" + this.getId() + "|" + this.getStackTrace());
}
}
}

Thread类中常用方法:

run():如果该线程时使用独立的Runnable运行对象构造的,则调用该Runnable对象的run方法。否则,该方法不执行任何操作并返回。

sleep(longmillls):在指定的毫秒数内让当前正在执行的线程休眠(暂停执行),此操作受到系统计时器和调度程序精度和准确性的影响 String

yield():暂停当前正在执行的线程对象,并执行其他线程 start():使该线程开始运行,java虚拟机再调用该线程的run方法

join():等待该线程结束

对比:

上面给出了Thread和Runnable的实现,我们能看到在使用Runnable的方式实现线程的过程中:

    Shop1 s6 = new Shop1("T");
Thread t3 = new Thread(s6);
t3.start();

即把Runnable对象(实现了Runnable接口的对象)还是塞进了Thread中让Thread来实现。那么我们可以new 多个Thread来实现同一个Runnbale对象,即实现了资源的共享,比如在售票系统中多名用户对同一种票的抢购。另一方面,java是单继承多实现的,如果我们使用Thread的话意味着该类只能继承Thread,对于程序的扩展不利,而实现Runnbale接口则没有这个顾虑。考虑程序的健壮性,我们应该尽量使用Runnable来实现我们的线程。

run和start

初学多线程我们总是分不清楚run()方法和start()方法的区别,其实我们再看一下上面Thread类的源码就不难发现他们的用法是很容易区分的:

  1. run()方法是线程的实现方法,即你需要线程去做什么事情,那么这些实现的内容写在run()里面,当线程启动时就会调用run()方法继而实现run()内部的代码;

  2. start()方法是线程的启动方法,即如果你new Thread()这样并不算完。你还得new Thread().start()才算启动这个线程,启动完之后线程内部会主动的调用run()方法执行该线程的业务逻辑代码。

java并发编程(二)----创建并运行java线程的更多相关文章

  1. Java并发编程二三事

    Java并发编程二三事 转自我的Github 近日重新翻了一下<Java Concurrency in Practice>故以此文记之. 我觉得Java的并发可以从下面三个点去理解: * ...

  2. 那些年读过的书《Java并发编程实战》和《Java并发编程的艺术》三、任务执行框架—Executor框架小结

    <Java并发编程实战>和<Java并发编程的艺术>           Executor框架小结 1.在线程中如何执行任务 (1)任务执行目标: 在正常负载情况下,服务器应用 ...

  3. Java并发编程里的volatile。Java内存模型核CPU内存架构的对应关系

    CPU内存架构:https://www.jianshu.com/p/3d1eb589b48e Java内存模型:https://www.jianshu.com/p/27a9003c33f4 多线程下的 ...

  4. 【Java并发编程二】同步容器和并发容器

    一.同步容器 在Java中,同步容器包括两个部分,一个是vector和HashTable,查看vector.HashTable的实现代码,可以看到这些容器实现线程安全的方式就是将它们的状态封装起来,并 ...

  5. Java 并发编程(二):如何保证共享变量的原子性?

    线程安全性是我们在进行 Java 并发编程的时候必须要先考虑清楚的一个问题.这个类在单线程环境下是没有问题的,那么我们就能确保它在多线程并发的情况下表现出正确的行为吗? 我这个人,在没有副业之前,一心 ...

  6. Java并发编程(五):Java线程安全性中的对象发布和逸出

    发布(Publish)和逸出(Escape)这两个概念倒是第一次听说,不过它在实际当中却十分常见,这和Java并发编程的线程安全性就很大的关系. 什么是发布?简单来说就是提供一个对象的引用给作用域之外 ...

  7. Java并发编程实战 第16章 Java内存模型

    什么是内存模型 JMM(Java内存模型)规定了JVM必须遵循一组最小保证,这组保证规定了对变量的写入操作在何时将对其他线程可见. JMM为程序中所有的操作定义了一个偏序关系,称为Happens-Be ...

  8. java并发编程JUC第九篇:CountDownLatch线程同步

    在之前的文章中已经为大家介绍了java并发编程的工具:BlockingQueue接口.ArrayBlockingQueue.DelayQueue.LinkedBlockingQueue.Priorit ...

  9. java多线程编程(二创建线程)

    1.概念           因为java是完全面向对象的,所以在java中,我们说的线程,就是Thread类的一个实例对象.所以,一个线程就是一个对象,它有自己字段和方法. 2.创建线程 创建线程有 ...

  10. Java并发编程 (二) 并发基础

    个人博客网:https://wushaopei.github.io/    (你想要这里多有) 一.CPU多级缓存-缓存一致性 1.CPU多级缓存 ​ 上图展示的是CPU高级缓存的配置,数据的读取和存 ...

随机推荐

  1. TCP中的粘包问题,以及用TCP和UDP实现多次聊天

    TCP协议 在连接内多和客户端说几句 #server端 import socket sk = socket.socket() sk.bind(('127.0.0.1',9001)) sk.listen ...

  2. vue集成百度富文本编辑器

    1.前期工作,访问百度富文本官网下载相应的百度富文本文件,根据后端用的技术下载相应的版本,建议下载最新版UTF-8版 (有图有真相,看图) https://ueditor.baidu.com/webs ...

  3. Django rest framework(1)----认证

    目录 Django组件库之(一) APIView源码 Django restframework   (1)  ----认证 Django rest framework(2)----权限 Django ...

  4. 分享基于EF6、Unitwork、Autofac的Repository模式设计

    目录 分享基于EF6.Unitwork.Autofac的Repository模式设计 一.实现的思路和结构图 二.Repository设计具体的实现代码 三.Repository设计的具体的使用 四. ...

  5. python如何将一个多位数数值转换为列表类型

    现在:a = 10,由于暂时没找到更好的方法,且使用下面的方法进行转换. 目标:转化为['10'] 以下为错误尝试: 1.直接转换,提示整型对象不可迭代. 2.先转换为字符串,再转换为列表,发现被分成 ...

  6. Python 3.5学习笔记(第一章)

    本章内容: 1.安装python 3.5 和 PyCharm 社区版 2.第一个python程序 3.变量 4.字符编码 5.用户输入 6.字符串格式化输出 7.if .else .elif 8.fo ...

  7. canvas的width和height设置问题

    最近在学习canvas属性中遇到一个小问题,就是canvas的width和height设置问题 代码如下: <!DOCTYPE html> <html lang="en&q ...

  8. Excel催化剂开源第43波-Excel选择对象Selection在.Net开发中的使用

    Excel的二次开发有一极大的优势所在,可以结合用户的交互进行程序的运行,大量用户的交互,都是从选择对象开始,用户选择了单元格区域.图形.图表等对象,之后再进行程序代码的加工处理,生成用户所需的最终结 ...

  9. Flutter学习笔记(10)--容器组件、图片组件

    如需转载,请注明出处:Flutter学习笔记(10)--容器组件.图片组件 上一篇Flutter学习笔记(9)--组件Widget我们说到了在Flutter中一个非常重要的理念"一切皆为组件 ...

  10. Kafka学习(三)-------- Kafka核心之Cosumer

    了解了什么是kafka( https://www.cnblogs.com/tree1123/p/11226880.html)以后 学习核心api之消费者,kafka的消费者经过几次版本变化,特别容易混 ...