博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
多线程day01
阅读量:5814 次
发布时间:2019-06-18

本文共 4477 字,大约阅读时间需要 14 分钟。

多线程作为Java中很重要的一个知识点,在此还是有必要总结一下的。

一.线程的生命周期及五种基本状态

关于Java中线程的生命周期,首先看一下下面这张较为经典的图:

上图中基本上囊括了Java中多线程各重要知识点。掌握了上图中的各知识点,Java中的多线程也就基本上掌握了。主要包括:

Java线程具有五中基本状态

新建状态(New):当线程对象对创建后,即进入了新建状态,如:Thread t = new MyThread();

就绪状态(Runnable):当调用线程对象的start()方法(t.start();),线程即进入就绪状态。处于就绪状态的线程,只是说明此线程已经做好了准备,随时等待CPU调度执行,并不是说执行了t.start()此线程立即就会执行;

运行状态(Running):当CPU开始调度处于就绪状态的线程时,此时线程才得以真正执行,即进入到运行状态。注:就     绪状态是进入到运行状态的唯一入口,也就是说,线程要想进入运行状态执行,首先必须处于就绪状态中;

阻塞状态(Blocked):处于运行状态中的线程由于某种原因,暂时放弃对CPU的使用权,停止执行,此时进入阻塞状态,直到其进入到就绪状态,才 有机会再次被CPU调用以进入到运行状态。根据阻塞产生的原因不同,阻塞状态又可以分为三种:

1.等待阻塞:运行状态中的线程执行wait()方法,使本线程进入到等待阻塞状态;

2.同步阻塞 -- 线程在获取synchronized同步锁失败(因为锁被其它线程所占用),它会进入同步阻塞状态;

3.其他阻塞 -- 通过调用线程的sleep()或join()或发出了I/O请求时,线程会进入到阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入就绪状态。

死亡状态(Dead):线程执行完了或者因异常退出了run()方法,该线程结束生命周期。

二. Java多线程的创建及启动

【通过继承Thread】

 一个Thread对象只能创建一个线程,即使它调用多次的.start()也会只运行一个的线程。

  【看下面的代码 & 输出结果】

class CTest extends Thread { 4     private int tickte = 20; 5  6     public void run() { 7         while (true) { 8             if (tickte > 0) { 9                 System.out.println(Thread.currentThread().getName() + " 出售票 "10                         + tickte--);11             } else {12                 System.exit(0);13             }14         }15     }16 17 }18 19 public class Demo3 {20     public static void main(String[] args) {21         // new CTest().start();22         // new CTest().start();23         Thread t1 = new CTest();//创建一个线程24         t1.start();25         t1.start();26     }27 }28 29 //30 Thread-0 出售票 2031 Thread-0 出售票 1932 Thread-0 出售票 1833 Thread-0 出售票 1734 Thread-0 出售票 1635 Thread-0 出售票 1536 Thread-0 出售票 1437 Thread-0 出售票 1338 Thread-0 出售票 1239 Thread-0 出售票 1140 Thread-0 出售票 1041 Thread-0 出售票 942 Thread-0 出售票 843 Thread-0 出售票 744 Thread-0 出售票 645 Thread-0 出售票 546 Thread-0 出售票 447 Thread-0 出售票 348 Thread-0 出售票 249 Thread-0 出售票 1

通过调用当前线程对象的名字Thread.currentThread.getName(),根据结果可以看出,只运行了一个线程

这就说明了一个问题,每创建一个Thread对象,只能创建一个线程。

2.实现Runnable接口,并重写该接口的run()方法,该run()方法同样是线程执行体,创建Runnable实现类的实例,并以此实例作为Thread类的target来创建Thread对象,该Thread对象才是真正的线程对象。

class MyRunnable implements Runnable { 2     private int i = 0; 3  4     @Override 5     public void run() { 6         for (i = 0; i < 100; i++) { 7             System.out.println(Thread.currentThread().getName() + " " + i); 8         } 9     }10 }
public class ThreadTest { 2  3     public static void main(String[] args) { 4         for (int i = 0; i < 100; i++) { 5             System.out.println(Thread.currentThread().getName() + " " + i); 6             if (i == 30) { 7                 Runnable myRunnable = new MyRunnable(); // 创建一个Runnable实现类的对象 8                 Thread thread1 = new Thread(myRunnable); // 将myRunnable作为Thread target创建新的线程 9                 Thread thread2 = new Thread(myRunnable);10                 thread1.start(); // 调用start()方法使得线程进入就绪状态11                 thread2.start();12             }13         }14     }15 }

3、使用匿名内部类的方式创建线程

首先回顾下之前的匿名内部类:  

匿名内部类的格式:

  new 接口或者接口名(){
        重写方法
    };
本质:是该类或者接口的子类对象

 

匿名内部类方式使用多线程

1、new Thread(){代码…}.start();
2、new Thread(new Runnable(){代码…}).start();

例子1:继承Thread类的匿名内部类实现多线程

// 一、继承Thread类实现多线程2         new Thread() {3             // 线程的代码4             public void run() {5                 for (int x = 0; x < 100; x++) {6                     System.out.println("Thread" + "--" + x);7                 }8             }9         }.start();// 别忘了启动线程

例子2:继承Runnable类的匿名内部类实现多线程

// 二、继承Runnable类实现多线程 2         new Thread(new Runnable() { 3             // 线程的代码 4             public void run() { 5                 for (int x = 0; x < 100; x++) { 6                     System.out.println("Runnable" + "--" + x); 7                 } 8             } 9 10         })11 12         {13             // 这里的代码为空14         }.start();

由于继承Runnable类实现线程中,start之前的{}为空,这里在继承Thread类中是重写线程的方法的,

 所以,如果两者结合起来的话,会执行Runnable还是Thread?

 

例子3:同时继承Runnable类和Thread类的匿名内部类来实现多线程

 

// 三、两者结合 2  3         new Thread(new Runnable() { 4  5             public void run() { 6                 // 填写继承Ruannble的线程代码 7                 for (int x = 0; x < 100; x++) { 8                     System.out.println("hello" + "--" + x); 9                 }10 11             }12 13         }) {14             // 填写继承Thread类的线程代码15             public void run() {16                 for (int x = 0; x < 100; x++) {17                     System.out.println("world" + "--" + x);18                 }19             }20         }.start();21         //通过运行结果可知道,这里只执行继承Thread类的代码

 

转载地址:http://yhmbx.baihongyu.com/

你可能感兴趣的文章
微信公众号与APP微信第三方登录账号打通
查看>>
onchange()事件的应用
查看>>
Windows 下最佳的 C++ 开发的 IDE 是什么?
查看>>
软件工程师成长为架构师必备的十项技能
查看>>
python 异常
查看>>
百度账号注销
查看>>
Lua语言特色
查看>>
C# 单机Window 程序 sqlite 数据库实现
查看>>
mysql-This version of MySQL doesn’t yet support ‘LIMIT & IN/ALL/ANY/SOME 错误解决
查看>>
BIEE Demo(RPD创建 + 分析 +仪表盘 )
查看>>
Cocos2dx 3.0开发环境的搭建--Eclipse建立在Android工程
查看>>
基本概念复习
查看>>
重构第10天:提取方法(Extract Method)
查看>>
Android Fragment使用(四) Toolbar使用及Fragment中的Toolbar处理
查看>>
解决pycharm在ubuntu下搜狗输入法一直固定在左下角的问题
查看>>
“Info.plist” couldn’t be removed
查看>>
多线程day01
查看>>
react-native 模仿原生 实现下拉刷新/上拉加载更多(RefreshListView)
查看>>
MySQL出现Access denied for user ‘root’@’localhost’ (using password:YES)
查看>>
通过Roslyn构建自己的C#脚本(更新版)(转)
查看>>