程序员在开发Java多线程应用程序中,各个线程之间由于要共享资源,必须用到锁机制,可是对于一个新手java程序员来说不清楚java线程怎么加锁?下面来我们就来给大家讲解一下java线程加锁方法。
1.对象锁(同步块)——锁某一个对象
对象锁:顾名思义给对象上锁
当A线程访问一个object的时候,首先会获取该对象的对象锁,然后访问锁定的代码,而B线程访问一个对象object多顶的代码区时,因为线程A之前获取到对象object的对象锁,因此线程B访问同步代码区的时候会被阻塞,直到线程A执行完同步代码块,然后释放对象锁,B线程获取对象object的对象锁,才能进行访问同步代码块
用法:synchronized (this){
//—-同步代码块———
}
this表示当前访问同步代码区的对象
特别注意:只要同步代码区的是同一个对象时才能实现互斥访问的效果
示例:
package com.demo; public class SyncBlock { public static int count; public static Object lock = new Object(); public static void main(String[] args) throws InterruptedException { Thread t1 = new Thread(new MyRunnable(), "线程1"); Thread t2 = new Thread(new MyRunnable(), "线程2"); t1.start(); t2.start(); t1.join(); t2.join(); System.out.println("count=" + count); } static class MyRunnable implements Runnable { @Override public void run() { for (int i = 0; i < 10000; i++) { synchronized(lock) { count++; } } } } }
2.类锁——锁当前类(方法上有static关键字修饰)
类锁:顾名思义给类上锁–>具体就是给类的字节码上锁
当线程A访问类锁锁定的同步代码块时,线程B访问同步代码块,因为A之前已经获取类锁,因此线程B只能A执行完同步代码块后才能获取类锁进行访问
示例:
package com.demo; public class SyncClass { private static int count; public static void main(String[] args) throws InterruptedException { Thread t1 = new Thread(new MyRunnable()); Thread t2 = new Thread(new MyRunnable()); t1.start(); t2.start(); t1.join(); t2.join(); System.out.println(count); } static class MyRunnable implements Runnable { public static synchronized void increase() { count++; } @Override public void run() { for (int i = 0; i < 10000; i++) { increase(); } } } }
结果:同上
3.实例锁——锁的是当前实例(方法上没有static修饰,表示只能锁当前实例)
同一个实力访问的线程才是安全的,不同的线程访问则不安全
示例:
package com.demo; /** * @author dxchen * */ public class SyncInstance { public static int count; public static void main(String[] args) throws InterruptedException { MyRunnable target = new MyRunnable(); Thread t1 = new Thread(target, "线程1"); Thread t2 = new Thread(target, "线程2"); t1.start(); t2.start(); t1.join(); t2.join(); System.out.println(count); } static class MyRunnable implements Runnable { /** * 对count操作的实例方法 */ public synchronized void increase() { count++; } @Override public void run() { for (int i = 0; i < 10000000; i++) { increase(); } } } }
java线程生命周期是怎样的?
下图显示了一个线程完整的生命周期。
1.新建状态
使用 new 关键字和 Thread 类或其子类建立一个线程对象后,该线程对象就处于新建状态。它保持这个状态直到程序 start() 这个线程。
2.就绪状态
当线程对象调用了start()方法之后,该线程就进入就绪状态。就绪状态的线程处于就绪队列中,要等待JVM里线程调度器的调度。
3.运行状态
如果就绪状态的线程获取 CPU 资源,就可以执行 run(),此时线程便处于运行状态。处于运行状态的线程最为复杂,它可以变为阻塞状态、就绪状态和死亡状态。
4.阻塞状态
如果一个线程执行了sleep(睡眠)、suspend(挂起)等方法,失去所占用资源之后,该线程就从运行状态进入阻塞状态。在睡眠时间已到或获得设备资源后可以重新进入就绪状态。可以分为三种:
5.等待阻塞
运行状态中的线程执行 wait() 方法,使线程进入到等待阻塞状态。
6.同步阻塞
线程在获取 synchronized 同步锁失败(因为同步锁被其他线程占用)。
7.其他阻塞
通过调用线程的 sleep() 或 join() 发出了 I/O 请求时,线程就会进入到阻塞状态。当sleep() 状态超时,join() 等待线程终止或超时,或者 I/O 处理完毕,线程重新转入就绪状态。
8.死亡状态:
一个运行状态的线程完成任务或者其他终止条件发生时,该线程就切换到终止状态。
其实java线程是一个动态执行的过程,它也是有一个从产生到死亡的过程,直到最后死亡状态就是线程终止。最后大家如果想要了解更多java入门知识,敬请关注奇Q工具网。
推荐阅读: