实现线程的两种方式

上一节我们了解了关于线程的一些基本知识,下面我们正式进入多线程的实现环节。实现线程常用的有两种方式,一种是继承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. XTOJ 1267:Highway(树的直径)***

    http://202.197.224.59/OnlineJudge2/index.php/Problem/read/id/1267 题意:给出一棵树,每条树边有权值,现在要修建n-1条边,边的权值为边 ...

  2. POJ 1966:Cable TV Network(最小点割集)***

    http://poj.org/problem?id=1966 题意:给出一个由n个点,m条边组成的无向图.求最少去掉多少点才能使得图中存在两点,它们之间不连通. 思路:将点i拆成a和b,连一条a-&g ...

  3. POJ 2914:Minimum Cut(全局最小割Stoer-Wagner算法)

    http://poj.org/problem?id=2914 题意:给出n个点m条边,可能有重边,问全局的最小割是多少. 思路:一开始以为用最大流算法跑一下,然后就超时了.后来学习了一下这个算法,是个 ...

  4. 利用HtmlAgilityPack插件写的一个抓取指定网页的图片 第一次写 很乱 随便看看就行

    public partial class Form1 : Form { /// <summary> /// 存放图片地址 /// </summary> List<stri ...

  5. Azure中国CDN全球覆盖功能初探

    在不久前的4月初,Azure中国官网上简短地发布了其CDN中“标准版 Zone 2”功能.一开始笔者尚有些摸不着头脑,这个“Zone 2”具体指的是什么呢?好在后来官网更新了信息描述如下: 这下就比较 ...

  6. java高并发系列-第1天:必须知道的几个概念

    java高并发系列-第1天:必须知道的几个概念 同步(Synchronous)和异步(Asynchronous) 同步和异步通常来形容一次方法调用,同步方法调用一旦开始,调用者必须等到方法调用返回后, ...

  7. python对Excel的读取

    在python自动化中,经常会遇到对数据文件的操作,比如添加多名员工,但是直接将员工数据写在python文件中,不但工作量大,要是以后再次遇到类似批量数据操作还会写在python文件中吗? 应对这一问 ...

  8. 【最小生成树之Kruskal算法】

    看完之后推荐再看一看[最小生成树之Prim算法]-C++ 定义:一个有 n 个结点的连通图的生成树是原图的极小连通子图,且包含原图中的所有 n 个结点,并且有保持图连通的最少的边.最小生成树可以用kr ...

  9. CentOS 7.3 配置静态ip

    镜像:CentOS-7-x86_64-DVD-1511.iso 1.修改.查看虚拟机的网段 1.1.查看虚拟机网段 编辑-> 虚拟机网络编辑器,修改的需要管理员权限 选择NAT模式 点击 NAT ...

  10. 校园表白墙、微信表白墙、校园墙 微信小程序 JAVA 开发记录与分享

    目录 最新版表白墙博客地址 1.微信小程序前台展示 2.功能介绍 3.后台管理 4.后端语言采用 JAVA 开发 5.体验此微信小程序 扫描下方二维码 6.如何联系我或需要源码进行联系 最新版表白墙博 ...