JAVA多线程(四)

线程安全与同步

我们在上一节,用代码简单的介绍了银行挂号的那个案例,当我们的index 越来越多的时候,会出现数据不一致的问题,那么我们如何解决呢?

初始Synchronized关键字
Synchronized 关键字 提供了排他机制,也就是在同一时间执行某些操作,
1.Synchronized 关键字提供了一种锁机制,能够确保共享变量的互斥访问,从而防止数据不一致的情况,
2.Synchronized 关键字 包含了两个JVM指令 一个,monitor enter和monitor exit,它能够保证在任何时候,线程执行到Monitor enter成功都必须从主内存中获取数据,而不是从缓存中,而执行了monitor exit 指令 之后 是将缓存中的数据刷新到主内存中,
3.monitor exit退出之前必定有一个monitor enter。

Synchronized关键字的用法
1.同步方法
public synchronized void sync(){
}
public synchronized static void sync(){
}
2.同步代码块
private final Object mutex = new Object();
public void Sync(){
synchronized(mutex){
}
}
Synchronized 提供了一种互斥机制,其实准确的来讲 应该是某线程获取了与mutex 关联的Monitor锁。

使用jdk/bin目录下的JConsole工具监控线程运行情况
JAVA多线程(四)_第1张图片
JConsole

1.选中要建立连接的本地连接或者远程连接,然后点击连接,按钮进入JConsole控制台
2.选中程序中创建的某个线程,查看当前线程的状态


JAVA多线程(四)_第2张图片
JConsole

JDK命令javap 对Myclass 进行反编译,输出了大量的JVM指令,其中Monitor enter 和Monitor exit是成对出现的,
每一个对象都与Monitor 相关联,一个Monitor锁 只能在同一个时间被一个Monitor 所获得,在一个线程尝试 获得与对象关联的Monitor 会发生如一下事情:

monitor enter

1.如果Monitor 的计数为0 ,则意味着该monitor的Lock还没有被获得,某个线程获得之后将立即对该计数器加一,从此该线程就是这个monitor,
2.如果一个已近拥有该monitor所有权的线程重入,则会导致该monitor计数器累计。
3.如果monitor已近被其他线程所拥有,当前尝试获取monitor锁的线程会陷入阻塞,直到monitor计数器变为0,才能再次尝试获取monitor的所有权,

monitor Exit

释放对Monitor 锁的所有权,想要释放的前提是,你已经获取了所有权,就是将monitor 计数器减一,如果计数器的结果为0,那么该线程将不再拥有该monitor 锁的所有权。

使用Synchronized需要注意的问题

1.与Monitor关联的对象不能为空
意思就是说你Synchronized() 括号中的对象不能为null

private final Object obj = null;
public void syncMethod(){
    Synchronized(obj){
    }
}

2.synchronized作用域太大
由于synchronized 存在排他性,如果synchronized 的作用域太大,则代表着效率非常低,甚至丧失并发的优势...

public class Task implements Runnable
 {
      @Override
      public synchronized void run(){
        
  }
}

synchronized 对整个逻辑单元 都加了锁,从而丧失了并发的能力,

3.不同的monitor 锁相同的方法

public class Task implements Runnable
{
private final Object obj = new Object();
   @Override
    public void run(){
        synchronized (obj){
            //doSomething();
     }
  }
}

public static void main(String [] args){
     for(int i = 0 ;;i<5i++){
      new Thread(Task::new).start();  
  } 
}

上述构造了五个线程同时也构造了五个Runnable实例,Runnable作为逻辑执行单元传递给Thread; Synchronized根本互斥不了与之对应的作用域。上述的代码每个线程争抢的monitor关联都是独立的,因此不可能起到互斥的作用。
4.多个锁交叉导致死锁

private final Object o1 = new Object();
private final Object 02 = new Object();
private void write(){
 synchronized(o1){
      synchronized(o2){
  }
  }
}

public void read(){
     synchronized(o2){
      synchronized(o1){
  }
  }
}
This Monitor 和Class Monitor 的区别

This Monitor

public class ThisMonitor{
  public synchronized void test1() throws InterruptedException{
          TimeUnit.SECOND.Sleep(10); 
  }
public void test2() throws InterruptedException{
      synchronized(this){
            //doSomething
        }
  }
}

上述代码中,test1方法保持方法同步的方式,test2方法保持代码块的方式,他俩关联的Monitor锁的对象都是一样的。与之关联的对象都是ThisMonitor对象的引用

ClassMonitor :有两个类方法(静态方法),分别使用synchronized对其进同步

public class ClassMonitor{
  public static synchronized void staticTest1(){
  }
 public static void staticTest2(){
   synchronized (ClassMonitor.class){
}
  }
}

上述静态方法staticTest1 保持方法同步的方式,staticTest2保持代码块同步的方式,使用ClassMonitor.class 的实例引用作为monitor

程序死锁的原因 既诊断

1.交叉锁导致死锁。
2.内存原因
3.数据库锁
4.死循环引起的死锁

如何诊断程序发生死锁?
打开jstack工具或者jconsole工具 一般交叉锁引起的死锁都会进入阻塞状态,可借助工具来发现。

你可能感兴趣的:(JAVA多线程(四))