多线程二——常用方法、生命周期、互斥锁、死锁与释放锁

线程的终止

1、当线程完成任务后,就会自动退出

2、还可以通过使用变量来控制run方法退出的方式停止线程。

案例:

启动一个线程,然后在main线程中去停止子线程(**我们直线在子线程上设置一个变量,直接让循环开关关闭 **)

/**
 * @author 喂S别闹
 * @create 2021/11/10-8:59
 * @Version 1.0
 * @Description: 线程的退出
 */
public class ThreadExit {
    public static void main(String[] args) {
        T t = new T();
        t.start();

        //如果希望主线程去控制t线程的终止,那么我们就需要去修改loop变量
        //让t退出run方法,从而 终止t线程的循环

        //让主线程休眠10S,再通知退出
        try {
            System.out.println("主线程休眠10S");
            Thread.sleep(10 * 1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        t.setLoop(false);

    }

}

class T extends Thread {
    int count = 0;
    //设置一个控制变量
    private boolean loop = true;

    @Override
    public void run() {
        while (loop) {
            try {
                Thread.sleep(50);//休眠50ms
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("thread...运行中" + ++count);
        }
    }

    public void setLoop(boolean loop) {
        this.loop = loop;
    }
}

线程的常用方法

第一组常规方法以及Interrupt的用法

  • setName : 设置线程的名称
  • getName
  • start:使该线程开始执行;Java虚拟机底层调用该线程的start()方法
  • run:调用线程对象的run方法
  • setPriority :更改线程的优先级
  • getPriority:获取线程的优先级
  • sleep:在指定的毫秒数内让当前正在执行的线程休眠(暂停执行)
  • interrupt:中断线程

注意细节:

1、start底层会创建新的线程,调用run,run就是一个简单的方法调用,不会启动新的线程

2、线程优先级的范围

3、interrupt,中断线程,但并没有真正的结束线程。所以一般用于中断正在休眠的线程

4、sleep:线程的静态方法,使当前线程休眠

Interrupt的用法

/**
 * @author 喂S别闹
 * @create 2021/11/17-11:30
 * @Version 1.0
 * @Description: Interrup中断线程的使用
 */
public class ThreadInterrupt {
    public static void main(String[] args) throws InterruptedException {
        T1 t1 = new T1();
        t1.setName("喂S别闹");
        t1.setPriority(Thread.MIN_PRIORITY);
        t1.start();
        for (int i = 0; i < 5; i++) {
            Thread.sleep(1000);
            System.out.println("主线程" + i);
        }
        System.out.println(t1.getName() + "线程的优先级是:" + t1.getPriority());
        t1.interrupt(); //当执行到这里的时候就会中断T1线程的休眠
    }
}

class T1 extends Thread {
    @Override
    public void run() {
        while (true) {
            for (int i = 0; i < 100; i++) {
                System.out.println(Thread.currentThread().getName() + "线程运行..." + i);
            }
            try {
                System.out.println(Thread.currentThread().getName() + "休眠~~");
                Thread.sleep(20000);
            } catch (InterruptedException e) {
                //当该线程执行到一个interrupt方法时,就会catch一个异常,可以加入自己的业务代码
                //InterruptedException 是捕获到一个中断异常
                e.printStackTrace();
            }
        }
    }
}

常用方法第二组yield与join

1、yield:线程的礼让。让出 CPU,让其他线程去执行,但礼让的时间不确定,所有也不一定礼让成功。(t1线程和t2线程,如果t1线程执行yield,那么就是t1线程主动礼让t2线程先执行,但是如果CPU运行得快,有足够时间,那么礼让就会不成功

2、Join;线程的插队。插队的线程一旦插入成功,则肯定会先执行插入的线程所有任务(比如有t1和t2两个线程在交替执行,如果在t1线程中执行t2.join()方法,那么CPU就会直接会把t2线程完,然后再执行t1

例子练习

题目:

1、主线程每隔1s,输出hi,一共10次

2、当输出5次hi的时候,启动一个子线程(要求实现Runnable),每隔1s输出Hello,等该线程输出10次hello后退出

3、主线程继续输出hi,直到主线程退出

比如:

hi1
hi2
hi3
hi4
hi5
子线程结束
hello1
hello2
。。。
hello10
hi6
..
hi10
主线程结束

代码:

/**
 * @author 喂S别闹
 * @create 2021/11/17-14:17
 * @Version 1.0
 * @Description: 主线程和子线程交替使用的练习
 * 1、主线程每隔1s,输出hi,一共10次
 * <p>
 * 2、当输出5次hi的时候,启动一个子线程(要求实现Runnable),每隔1s输出Hello,等该线程输出10次hello后退出
 * <p>
 * 3、主线程继续输出hi,直到主线程退出
 */
public class ThreadMethodExercise {
    public static void main(String[] args) throws InterruptedException {
      /*  T3 t3 = new T3();
        Thread t4 = new Thread(t3);*/
        Thread t3 = new Thread(new T3());


        for (int i = 1; i <= 10; i++) {
            System.out.println("Hi" + i);
            if (i == 5) {//说明主线程输出了5次
                t3.start(); //启动子线程 输出Hello
                t3.join(); //立即将t3子线程,插入到main线程,让t3先执行
            }
        }
    }
}

class T3 implements Runnable {
    private int count = 0;

    @Override
    public void run() {
        while (true) {
            System.out.println("Hello" + ++count);
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            if (count == 10) {
                break;
            }
        }
    }
}

常用方法:用户线程和守护线程

1、用户线程:也叫工程线程,当线程的任务执行完或通知方式结束

2、守护线程:一般是为工作线程服务的,当所有的用户线程结束,守护线程就自动结束。

3、常见的守护线程:垃圾回收机制

案例:测试如何将一个线程设置成守护线程

/**
 * @author 喂S别闹
 * @create 2021/11/17-15:38
 * @Version 1.0
 * @Description: 将一个线程设置成为守护线程
 */
public class ThreadMethod03 {
    public static void main(String[] args) throws InterruptedException {
        MyDaemonThread myDaemonThread = new MyDaemonThread();
        //如果希望当主线程结束之后,子线程就会自动结束
        //只需将子线程设为守护线程即可
        myDaemonThread.setDaemon(true);
        myDaemonThread.start();
        for (int i = 1; i <= 10; i++) {
            System.out.println("主线程");
            Thread.sleep(1000);
        }
    }
}

class MyDaemonThread extends Thread {
    @Override
    public void run() {
        while (true) {
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("哈哈哈哈");
        }
    }
}

线程的生命周期(中间有个图很重要)

JDK中用Thread.State枚举表示了线程的几种状态

在官方文档里,是有6种线程状态的

1、new:尚未开启线程处于此状态

2、Runnable :在Java虚拟机中执行的线程处于此状态(注意runnable状态不是正在运行状态 ,只是表示可以运行了,具体运行还需要看线程调度器来控制)

3、Blocked:被阻塞等待监视器锁定的线程处于此状态

4、waiting:正在等待另一个线程执行特定动作的线程处于此状态

5、Time_waiting:正在等待另一个线程执行动作达到指定等待时间的线程处于此状态(超时等待

6、Terminated:已退出的线程处于此状态

图很重要

image-20211117161753868

例子:

/**
 * @author 喂S别闹
 * @create 2021/11/17-16:18
 * @Version 1.0
 * @Description: 线程的状态展示
 */
public class ThreadState {
    public static void main(String[] args) throws InterruptedException {
        T t = new T();
        System.out.println(t.getName() + "状态" + t.getState());
        t.start();

        while (Thread.State.TERMINATED != t.getState()) {
            System.out.println(t.getName() + "状态" + t.getState());
            Thread.sleep(500);
        }

        System.out.println(t.getName() + "状态" + t.getState());
    }
}

class T extends Thread {
    @Override
    public void run() {
        while (true) {
            for (int i = 0; i < 10; i++) {
                System.out.println("hi " + i);
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            break;
        }
    }
}

Synchronized关键词

线程同步机制

1、在多线程编程中,一些敏感数据不允许被多个线程同时访问,此时就使用同步访问技术,保证数据在任何同一时刻,最多有一个线程访问,以保证数据的完整性。

2、简单说:线程同步,即当有一个线程在对内存进行操作时,其他线程都不可以对这个内存地址进程操作,直到该线程完成操作,其他线程才能对该内存地址进行操作。

同步具体方法--Synchronized

1、同步代码块
synchronized(对象){//得到对象的锁,才能操作同步代码
//需要被同步代码;
}

2、synchronized还可以放在方法声明中,表示整个方法-为同步方法
    public synchronized void m(String name){
    //需要被同步的代码
}

例子

public class SellTicket {
    public static void main(String[] args) { 	
        //测试同步
        SellTicket03 sellTicket03 = new SellTicket03();
        new Thread(sellTicket03).start(); //第一个线程
        new Thread(sellTicket03).start(); //第2个线程
        new Thread(sellTicket03).start(); //第3个线程
    }
}

//实现runnable接口,使用synchronized实现线程同步
class SellTicket03 implements Runnable {

    private int Ticketnum = 100; //让多个线程共享
    private boolean loop = true;

    public synchronized  void sell(){ //同步方法,在同一时刻,只能有一个线程来执行run方法
        if (Ticketnum <= 0) {
            System.out.println("售票结束了");
            loop=false;
            return;
        }
        //休眠
        try {
            Thread.sleep(50);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("窗口 " + Thread.currentThread().getName() + " 卖出了一张票 " + "还剩" + (--Ticketnum) + "张票");

    }

    @Override
    public void run() {
        while (loop) {
            sell();
        }
    }
}

分析同步原理

image-20211118142814437

互斥锁

基本介绍:

1、Java中,引入了对象互斥锁的概念,来保证共享数据操作的完整性

2、每个对象都对应于一个可称为“互斥锁”的标记,这个标记用来保证在任一时刻,只能有一个线程访问该对象

3、关键字synchronized来与对象的互斥锁联系。当某个对象用synchronized修饰时,表明该对象在任一时刻只能由一个线程访问

4、同步的局限性:导致程序的执行效率要降低

5、同步方法(非静态的类)的锁可以是this,也可以是其他对象(但必须是同一对象

6、同步方法(静态的类)的锁为当前类本身

使用互斥锁来解决售票问题

可以在代码块上加锁

public class SellTicket {
    public static void main(String[] args) {

        //测试同步
        SellTicket03 sellTicket03 = new SellTicket03();
        new Thread(sellTicket03).start(); //第一个线程
        new Thread(sellTicket03).start(); //第2个线程
        new Thread(sellTicket03).start(); //第3个线程
    }
}

//实现runnable接口,使用synchronized实现线程同步
class SellTicket03 implements Runnable {

    private int Ticketnum = 100; //让多个线程共享
    private boolean loop = true;
    Object object = new Object();

    public /*synchronized*/ void sell() { //同步方法,在同一时刻,只能有一个线程来执行run方法
        synchronized (/*this*/ object) {
            if (Ticketnum <= 0) {
                System.out.println("售票结束了");
                loop = false;
                return;
            }
            //休眠
            try {
                Thread.sleep(50);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("窗口 " + Thread.currentThread().getName() + " 卖出了一张票 " + "还剩" + (--Ticketnum) + "张票");

        }
    }

    @Override
    public  void run() {

        while (loop) {
            sell();
        }
    }
}

静态类方法

public class SellTicket {
    public static void main(String[] args) {
        //测试同步
        SellTicket03 sellTicket03 = new SellTicket03();
        new Thread(sellTicket03).start(); //第一个线程
        new Thread(sellTicket03).start(); //第2个线程
        new Thread(sellTicket03).start(); //第3个线程
    }
}

//实现runnable接口,使用synchronized实现线程同步
class SellTicket03 implements Runnable {

    private int Ticketnum = 100; //让多个线程共享
    private boolean loop = true;

    //同步方法静态的类的锁为当前类的本身
    //1、 public synchronized static void m1(){} 锁是在SellTicket03.class
//    2、如果在静态方法中,实现同步代码块,就是类名.class
//    public static void m2(){
//        synchronized (SellTicket03.class){
//            System.out.println("1");
//        }
//    }
    public synchronized static void m1(){

    }
    public static void m2(){
        synchronized (SellTicket03.class){
            System.out.println("1");
        }
    }

互斥锁注意事项

1、同步方法如果没有使用static修饰:默认锁对象为this

2、如果方法使用了static修饰,默认锁对象:当前类.class

3、实现:

  • 需要先分析上锁业务代码
  • 选择同步代码块还是同步方法
  • 要求多个线程的锁对象为同一个即可。

线程的死锁

介绍

多个线程都占用了对方的锁资源,但不肯相互让步,导致了死锁,在平时编程时是我们一定要避免的。

释放锁

释放锁的操作

1、当前线程的同步方法、同步代码块执行结束

2、当前线程在同步代码块、同步方法中遇到break、return

3、当前线程在同步代码块、同步方法中出现了未处理的Error或Exception,导致异常结束

4、当前线程在同步代码块、同步方法中执行了线程对象的wait()方法,当前线程暂停,并释放锁

不会释放锁的操作

1、线程执行同步代码块或同步方法时,程序调用Thread.sleep()、Thread.yield()方法暂停当前线程的执行,不会释放锁。

2、线程执行同步代码块时,其他线程调用了该线程的suspend()方法将该线程挂起,该线程不会释放锁。

注意:尽量避免用suspend()和resume()来控制线程,方法不再推荐使用

posted @ 2021-11-18 20:09  喂s别闹  阅读(128)  评论(0编辑  收藏  举报