Java多线程(三)关于多线程管理的相关函数说明

接着前面的多线程(二)的内容,下面我们接着来探讨多个线程创建之后,关于线程调度和管理的一些方法。

先来简单介绍下线程调度###

对于计算机的CPU(以单核为例),在任意时刻只能执行一条指令,每个线程只有获得CPU的使用权才能执行指令。当有多个处于可运行状态的线程在等待CPU,JVM的一项任务就是负责线程的调度。JVM按照特定机制为多个线程分配CPU的使用权过程就是线程调度。

线程调度的两种模型:分时调度模型和抢占式调度模型。
① 分时调度模型是指让所有的线程轮流获得cpu的使用权,并且平均分配每个线程占用的CPU的时间片,这个比较好理解。
② JVM采用抢占式调度模型,就是让线程抢夺CPU资源,运行顺序是不确定的,优先权高的线程,会有一定几率优先占用CPU。处于运行状态的线程会一直运行下去,直至它不得不放弃CPU。比如线程运行完毕、线程阻塞、运行被打断。关于线程的多种运行状态详见:Java中的多线程(二)线程的创建及线程的生命周期。当线程被中断时,CPU会保存当前线程的状态,以备此线程被唤醒时继续执行。

线程管理之设置线程优先级###

Java线程优先级共有10个级别,优先级较高的线程会获得较多的运行机会,取值范围是从1到10。如果小于1或大于10,则JDK抛出异常IllegalArgumentException()。Thread类有以下三个静态常量:
① static final int MAX_PRIORITY :值为10,代表最高优先级。
② static final int MIN_PRIORITY :值为1,代表最低优先级。
③ static final int NORM_PRIORITY:值为5,代表默认优先级。

这里要注意的是:
① 并不是说优先级较高的线程一定会在优先级较低的线程之前运行,优先级高这里是指获得较多的运行机会。
② 优先级高的线程会大部分先执行完,并不一定会全部执行完毕。
③ 子线程的优先级是跟父类优先级是一样的。

设置和获取优先级方法:
thread.setPriority(Thread.MIN_PRIORITY)和 thread.getPriority()

线程管理之守护线程###

我们在程序中创建的线程默认都是用户线程(User Thread)。与用户线程对应则是守护线程(Daemon Thread),也可称之为后台线程,它的作用就是为其它线程提供服务的。守护线程使用的情况较少,举例来说:JVM的垃圾(GC)回收线程就是守护线程。

需要注意的是:
① 当所有的用户线程都结束退出的时候,守护线程也就没啥可服务的了,随着线程的结束而结束。如果JVM只剩下守护线程,虚拟机就会退出。
② 守护线程会随时中断,因此不要在如输入输出流,数据库连接等场合使用守护线程。
③ 守护线程并非是JVM内部可提供,我们自己可以根据需要来设定守护线程。可以通过isDaemon和setDaemon方法来判断和设置一个线程为守护线程。
④ 守护线程必须在start方法前设置,否则会抛出IllegalThreadStateException异常。
⑤ 一个守护线程创建的子线程依然是守护线程。

package com.Dan;

public class Main {

    public static void main(String[] args) {
        // write your code here
        DaemonThread daemonThread = new DaemonThread();
        // 设置为守护线程
        daemonThread.setDaemon(true);
        daemonThread.start();
    }
}

class DaemonThread extends Thread {
    public void run() {
        System.out.println(Thread.currentThread().getName() + " start");

        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        System.out.println(Thread.currentThread().getName() + " end");
    }
}
// Thread-0 start
// 子线程需要执行1秒,因为设置被设置为守护线程,因此主线程不会等待子线程执行结束,而是提前退出。
// 所有用户线程都退出了,守护线程也就退出了,因此并没有打印end。

线程管理之常用方法###

一些简单方法#####

public static int activeCount():返回当前线程的线程组中活动线程的数目。
public static Thread currentThread():返回对当前正在执行的线程对象的引用。
public long getId():返回该线程的标识符,线程 ID 是唯一的。
public final void setName(String name):改变线程名称。
public final String getName():返回该线程的名称。
public String toString():返回该线程的字符串表示形式,包括线程名称、优先级和线程组。
public void start():使该线程开始执行,Java 虚拟机调用该线程的 run 方法。

下面是一些比较重要的方法:

sleep()方法 线程睡眠#####

sleep方法作用就是让当前线程休眠,交出CPU,让CPU去执行其他的任务。当前线程是指this.currentThread()返回的线程。sleep方法使线程转到阻塞状态,当睡眠结束后,就转为可运行状态。

package com.Dan;

/**
 * Created by daniel on 17/3/27.
 */
public class 多线程博客三 {
    public static void main(String[] args) {

        System.out.println(Thread.currentThread().getName()+"主线程开始执行。");
        TaskThread thread1 = new TaskThread("线程-A-");
        TaskThread thread2 = new TaskThread("线程-B-");
        thread1.start();
        thread2.start();
        System.out.println(Thread.currentThread().getName()+ "主线程运行结束。");
    }
}

class TaskThread extends Thread{
    private String name;

    public TaskThread(String name) {
        this.name = name;
    }
    @Override
    public void run() {

        System.out.println(Thread.currentThread().getName() + "子线程运行开始。");
        for (int i = 0; i < 5; i++) {
            System.out.println(name + "运行: " + i);
            try {
                sleep((long) Math.random() * 100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        System.out.println(Thread.currentThread().getName() + "子线程运行结束!");
    }
}

// main主线程开始执行。
// Thread-0子线程运行开始。
// Thread-1子线程运行开始。
// 线程-B-运行: 0
// main主线程运行结束。
// 线程-A-运行: 0
// 线程-B-运行: 1
// 线程-A-运行: 1
// 线程-B-运行: 2
// 线程-A-运行: 2
// 线程-B-运行: 3
// 线程-A-运行: 3
// 线程-B-运行: 4
// 线程-A-运行: 4
// Thread-1子线程运行结束。
// Thread-0子线程运行结束。

// sleep方法使当前线程休眠,交出CPU,让CPU去执行其他的任务。这里的主线程,早在子线程开始后就马上结束了。
join()线程加入#####

在当前线程中调用另一个线程的join方法,则当前线程转入阻塞状态,直到另一个进程运行结束,当前线程再由阻塞转为可运行状态。如果调用此方法时,另一个线程已经运行完毕,那就接着运行当前线程。

package com.Dan;

/**
 * Created by daniel on 17/3/27.
 */
public class ThreadJoin {
    
    public static void main(String[] args) {
        
        Thread thread = new Thread(new Runnable() { //匿名对象
            @Override
            public void run() {
                for (int i = 0; i < 10; i++) {
                    System.out.println(Thread.currentThread().getName() + " 子线程 " + i);
                }
            }
        });
        
        thread.start();
        
        for (int i = 0; i < 10; i++) {
            if (i == 5) {
                try {
                    thread.join(); // 此时调用Thread线程
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            System.out.println(Thread.currentThread().getName() + " 主线程 " + i);
        }
    }
}

// main 主线程 0
// main 主线程 1
// main 主线程 2
// main 主线程 3
// main 主线程 4
// Thread-0 子线程 0
// Thread-0 子线程 1
// Thread-0 子线程 2
// Thread-0 子线程 3
// Thread-0 子线程 4
// Thread-0 子线程 5
// Thread-0 子线程 6
// Thread-0 子线程 7
// Thread-0 子线程 8
// Thread-0 子线程 9
// main 主线程 5
// main 主线程 6
// main 主线程 7
// main 主线程 8
// main 主线程 9
yield()线程让步#####

暂停当前正在运行的线程,把运行机会让给其它的线程。这里的暂停,并不是让线程转到阻塞或等待状态,而是返回可运行状态,等待被调度运行。需要注意的是,此时让步的线程是可运行状态,它有可能会被再次运行。

package com.Dan;

/**
 * Created by daniel on 17/3/24.
 */

public class JavaEveryDay0324 extends Thread{

    public static void main(String[] args) {
        JavaEveryDay0324 test1 = new JavaEveryDay0324();
        JavaEveryDay0324 test2 = new JavaEveryDay0324();
        test1.start();
        test2.start();
    }
    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName() + "  1");

        yield();
        System.out.println(Thread.currentThread().getName() + "  2");
    }
}

// Thread-0  1
// Thread-1  1
// Thread-0  2
// Thread-1  2
// 线程test1在输出1后,执行yield()方法进入可运行状态,然后将CPU让给线程test2。同样线程test2运行输出1后,执行yield()方法也进入了可运行状态,将CPU又让给线程test1,线程test1继续执行,打印输出2,然后线程test2执行输出2。

// Thread-0  1
// Thread-0  2
// Thread-1  1
// Thread-1  2
// 也有可能出现这种情况,多运行几次,肯定会有的。

// Thread-0  1
// Thread-1  1
// Thread-1  2
// Thread-0  2
// 又或者是这一种。只要记住它返回的是可运行状态,不是阻塞,也不是等待。那么不同的结果就能解释清楚了。
** interrupt()线程中断信号**#####

interrupt():这里只谈中断信号,中断线程后续博客会更新。它向线程发送一个中断信号,强制结束调用该方法的线程当前状态,让线程在无限等待时(如死锁时)能抛出异常。

package com.Dan;

/**
 * Created by daniel on 17/3/27.
 */
public class ThreadInterrupt {

    public static void main(String[] args) {

        Thread thread = new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(5000000); // 别想醒过来了
                } catch (InterruptedException e) {
                    //e.printStackTrace();
                    System.out.println("抛异常喽吆唉哈咦。");
                }
                for (int i = 0; i < 10; i++) {
                    System.out.println(Thread.currentThread().getName() + " 子线程 " + i);
                }
            }
        });

        thread.start();

        for (int i = 0; i < 6; i++) {
            if (i == 2) {
                thread.interrupt(); // 终止当前线程的状态,并抛出个异常: e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "输出:" + i);
        }
    }
}

// main输出:0
// main输出:1
// 抛异常喽吆唉哈咦。
// main输出:2
// Thread-0 子线程 0
// Thread-0 子线程 1
// Thread-0 子线程 2
// Thread-0 子线程 3
// Thread-0 子线程 4
// Thread-0 子线程 5
// main输出:3
// main输出:4
// Thread-0 子线程 6
// Thread-0 子线程 7
// Thread-0 子线程 8
// Thread-0 子线程 9
// main输出:5

** wait()线程等待**#####

这里先简单介绍下wait方法,我们会在下一篇博客探讨线程同步相关知识的时候,再来探讨它。

在线程通信中,wait方法经常与notify和notifyAll方法一起使用。,他们并不是Thread类的方法,而是Object类的方法。
① wait:当达到某种状态的时候,wait方法让线程进入等待状态,让本线程休眠。线程自动释放其占有的对象锁,并等待notify。直到有其它线程调用对象的notify方法唤醒该线程,才能继续获取对象锁,继续运行。
② notify:唤醒一个正在等待当前对象锁的线程,并让它拿到对象锁。
③ notifyAll:唤醒所有正在等待前对象锁的线程。

需要注意的是:
① 在调用这3个方法的时候,当前线程必须获得这个对象的锁,也就是说这三个方法必须在synchronized(Obj){...}内部。
② 在调用notify方法后,并不会马上释放对象锁,而是在synchronized(){}执行结束的时候,自动释放锁,JVM会随机唤醒一个正在等待当前对象锁的线程,让他获得对象锁,唤醒线程。

** wait()与sleep()方法的区别**#####

① sleep方法是一个静态方法,作用在当前线程上。而wait方法是一个实例方法,并且只能在其他线程调用本实例的notify方法时被唤醒。
② wait只能在线程同步环境中被调用,会释放锁。而sleep不限制使用环境,当在一个Synchronized块中调用Sleep方法时,线程虽然休眠了,但是对象的锁并木有被释放,其他线程无法访问这个对象。
③ sleep必须捕获异常,而wait,notify和notifyAll则不需要。
④ 进入等待状态的线程能够被notify方法唤醒。sleep休眠时间到了,该线程不一定会立即执行,因为其它线程可能正在运行。
⑤ 如果你需要暂停某个线程一段特定的时间,就使用sleep方法。如果你想要实现线程间通信就使用wait方法。

** 关于 “调用yield()方法后,会选择同等优先级的线程继续执行。” 勘误**#####

看到网上有人说调用yield()方法后,会选择同等优先级或更高优先级的线程继续执行。这个观点是错误的。具体执行执行哪个线程是由JVM说了算。请看下面的例子,低优先级的也会被执行。

// 在上面方法基础上重新修改的
package com.Dan;

/**
 * Created by daniel on 17/3/24.
 */

public class JavaEveryDay0324 extends Thread{

    public static void main(String[] args) {
        JavaEveryDay0324 test3 = new JavaEveryDay0324();
        JavaEveryDay0324 test1 = new JavaEveryDay0324();
        JavaEveryDay0324 test2 = new JavaEveryDay0324();

        test3.setPriority(1);
        test2.setPriority(10);
        test1.setPriority(5);
        test3.start();
        test1.start();
        test2.start();
        System.out.println("线程1ID: "+test1.getId()+" 线程2ID: "+test2.getId()+" 线程3ID: "+test3.getId());

    }
    @Override
    public void run() {


        for (int i = 0; i < 3; i++) {
            System.out.println("线程ID为"+Thread.currentThread().getId()+": " + " 的第①次打印");

            yield();
            System.out.println("线程ID为"+Thread.currentThread().getId() +": " +  " 的第②次打印");
        }
    }
}

// 线程ID为10:  的第①次打印
// 线程ID为11:  的第①次打印
// 线程ID为12:  的第①次打印
// 线程1ID: 11 线程2ID: 12 线程3ID: 10
// 线程ID为10:  的第②次打印
// 线程ID为10:  的第①次打印
// 线程ID为11:  的第②次打印
// 线程ID为11:  的第①次打印
// 线程ID为12:  的第②次打印
// 线程ID为12:  的第①次打印
// 线程ID为10:  的第②次打印
// 线程ID为10:  的第①次打印
// 线程ID为11:  的第②次打印
// 线程ID为11:  的第①次打印
// 线程ID为12:  的第②次打印
// 线程ID为12:  的第①次打印
// 线程ID为10:  的第②次打印
// 线程ID为12:  的第②次打印
// 线程ID为11:  的第②次打印

写完喽!ㄟ(▔,▔)ㄏㄟ(▔,▔)ㄏㄟ(▔,▔)ㄏ


知识重在总结和梳理,只有不断地去学习并运用,才能化为自己的东西。当你能为别人讲明白的时候,说明自己已经掌握了。

欢迎转载,转载请注明出处!

如果有错误的地方,或者有您的见解,还请不啬赐教!

喜欢的话,麻烦点个赞!

Java中的多线程(一)多线程基础之进程、线程、并发、并行。
Java中的多线程(二)线程的创建及线程的生命周期

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,736评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,167评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,442评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,902评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,302评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,573评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,847评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,562评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,260评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,531评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,021评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,367评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,016评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,068评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,827评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,610评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,514评论 2 269

推荐阅读更多精彩内容

  • 本文主要讲了java中多线程的使用方法、线程同步、线程数据传递、线程状态及相应的一些线程函数用法、概述等。 首先讲...
    李欣阳阅读 2,377评论 1 15
  • Java多线程学习 [-] 一扩展javalangThread类 二实现javalangRunnable接口 三T...
    影驰阅读 2,921评论 1 18
  • 该文章转自:http://blog.csdn.net/evankaka/article/details/44153...
    加来依蓝阅读 7,257评论 3 87
  • 写在前面的话: 这篇博客是我从这里“转载”的,为什么转载两个字加“”呢?因为这绝不是简单的复制粘贴,我花了五六个小...
    SmartSean阅读 4,650评论 12 45
  • 早上去看了一场电影。 这两天心情一直不好,周末了也一个人在宾馆,怕憋出病来,于是早上起来定了9:55的《木乃伊》。...
    Evan0827阅读 221评论 0 0