转载请注明出处!!https://blog.csdn.net/qq_31842777/article/details/90180668
一、线程的五种状态
1、new,即新建状态。new关键字创建了一个线程对象实例,它仅仅作为一个对象实例存在,JVM没有为其分配CPU时间片timeslice等资源;
2、runnable, 即可运行状态。处于新建状态的线程调用start()方法启动线程后,线程已经得到除CPU时间片外的其他系统资源。该状态的线程位于可运行线程池中,等待JVM的线程调度器进行调度,从而得到获取CPU时间片的机会;
3、running,即运行状态。处于可运行状态的线程获取了CPU时间片,执行程序代码;
4、block,即阻塞状态。因为某种原因,线程放弃CPU时间片,暂时停止运行,此时,JVM调度器不会给它分配任何CPU时间,直接跳过它,除非线程进入runnable状态,才有机会再次获取CPU时间片。大多数情况下,sleep(long n)、suspend()、wait()使线程进入阻塞状态,对应的suspend()/resume()、wait()/notify()方法恢复线程运行。
5、dead,即死亡状态。线程体run()运行结束,或者调用线程对象的stop()方法后,线程终止运行,JVM回收线程占用的资源。死亡的线程不可复生。ps: stop()方法不安全,会破坏线程的状态,已经不推荐使用。
二、线程同步(互斥锁)
1、线程使用同步的原因
Java允许多线程并发控制,当多个线程同时操作一个可共享的资源变量时(数据的增删改查),将会导致数据不准确,相互之间产生冲突,因此加入同步锁以避免在该线程没有完成操作之前,被其他线程的调用, 从而保证了该变量的唯一性和准性。
2、同步方法
即有synchronized关键字修饰的方法,该关键字修饰的方法会自动加上互斥锁,实现同步。如:public synchronized void deposit()。
ps: synchronized关键字也可以修饰静态方法,此时如果调用该静态方法,将会锁住整个类。
3、同步代码块
即有synchronized关键字修饰的语句块, 被该关键字修饰的语句块会自动被加上内置锁,从而实现同步。如synchronized(object){}。
ps: Java多线程同步提供了两类方法:wait()、notify()。wait():当一个线程执行了该方法时,放弃互斥锁,进入互斥锁的等待队列;notify():该方法唤醒互斥锁等待队列中的线程,并进入可运行状态runnable。
4、同步方法、同步代码块区别
同步方法默认使用当前类class的对象作为锁;同步方法的颗粒度更细。
参考示例:https://www.cnblogs.com/xujingyang/p/6565606.html
三、死锁deadlock
**死锁 **指两个或两个以上的进程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去。
死锁产生的4个必要条件:
(1) 互斥条件:一个资源每次只能被一个进程使用;
(2) 请求与保持条件:一个进程因请求资源而阻塞时,对已获得的资源保持不放;
(3) 不剥夺条件: 进程已获得的资源,在末使用完之前,不能强行剥夺;
(4) 循环等待条件: 若干进程之间形成一种头尾相接的循环等待资源关系。
预防:
合理分配资源,避免进程永久占用系统资源;防止进程在等待状态下占用资源。
PS: 指定获取锁的顺序,并强制线程按照指定的顺序获取锁,可以避免死锁发生!(实际破坏了循环等待条件)
如果遭受了冷漠,那就埋头努力吧!