关于Java中锁的总结

多个进程或线程同时(或着说在同一段时间内)访问同一资源会产生并发(线程安全)问题。解决并发问题可以用锁。

java的内置锁:

每个java对象都可以用做一个实现同步的锁,这些锁称为内置锁。线程进入同步代码块或方法的时候会自动获得该锁,在退出同步代码块或方法时会释放该锁。获得内置锁的唯一途径就是进入这个锁保护的同步代码块或方法。java内置锁是一个互斥锁,这就意味着最多只有一个线程能够获得该锁,当线程A尝试去获得线程B持有的内置锁时,线程A必须等待或者阻塞,直到线程B释放这个锁,如果线程B不释放这个锁,那么线程A将永远等待下去。

java的对象锁和类锁:

java的内置锁基本上可以分为对象锁和类锁,对象锁是用于对象实例方法,或者一个对象实例上的,类锁是用于类的静态方法或者一个类的class对象上的。我们知道,类的对象实例可以有很多个,但是每个类只有一个class对象,所以不同对象实例的对象锁是互不干扰的,但是每个类只有一个类锁。有一点必须注意的是,其实类锁只是一个概念上的东西,并不是真实存在的,它只是用来帮助我们理解锁定实例方法和静态方法的区别的。

synchronized只是一个内置锁的加锁机制,当某个方法加上synchronized关键字后,就表明要获得该内置锁才能执行,并不能阻止其他线程访问不需要获得该内置锁的方法。

一、synchronized的用法

1、对象锁:同步方法,是对该对象加锁,其他线程将无法访问需要获取该对象锁的方法和代码块。

public class Test{

    public synchronized void print(){
        System.out.println("hello world!");
    }

}

2、对象锁:同步代码块,这种写法也是锁住该对象,和1效果一样,其他线程将无法访问需要获取该对象锁的方法和代码块。

public class Test{

    public void print(){
        synchronized(this){
            System.out.println("hello world!");
        }
    }

}

3、对象锁:同步代码块,这种写法锁住传入的对象,不影响需要获取当前对象锁的方法和代码块的访问。

public class Test{

    private String a = "test";

    public void print(){
        synchronized(a){
            System.out.println("hello world!");
        }
    }

    public synchronized void print1(){
        System.out.println("123456");
    }

}

执行print()里面的同步代码块,会给对象a加锁,注意不是给Test的对象加锁,也就是说Test对象的其它synchronized方法和代码块不会因为print()而被锁。同步代码块执行完,则释放对a的锁。
为了锁住一个对象的代码块而不影响该对象其它synchronized块的高性能写法:

public class Test{

    private byte[] lock = new byte[0];

    public void print(){
        synchronized(lock){
            System.out.println("hello world!");
        }
    }

    public synchronized void print1(){
        System.out.println("123456");
    }

}

4、类锁:用于静态方法。

public class Test{

    public synchronized static void print(){
        System.out.println("hello world!");
    }

}

效果等同于同步代码块传入该类的class对象。

public class Test{

    public void print(){
        synchronized(Test.class){
            System.out.println("hello world!");
        }
    }

}

类锁修饰方法和代码块的效果和对象锁是一样的,因为类锁只是一个抽象出来的概念,只是为了区别静态方法的特点,因为静态方法是所有对象实例共用的,所以对应着synchronized修饰的静态方法的锁也是唯一的,所以抽象出来个类锁。类锁和对象锁是互不干扰的。同样,线程获得对象锁的同时,也可以获得该类锁,即同时获得两个锁,这是允许的。

二、锁的释放

  一般是执行完毕同步代码块(锁住的代码块)后就释放锁,也可以用wait()方式半路上释放锁。wait()方式就好比蹲厕所到一半,突然发现下水道堵住了,不得已必须出来站在一边,好让修下水道师傅(准备执行notify的一个线程)进去疏通马桶,疏通完毕,师傅大喊一声: "已经修好了"(notify),刚才出来的同志听到后就重新排队。注意啊,必须等师傅出来啊,师傅不出来,谁也进不去。也就是说notify后,不是其它线程马上可以进入封锁区域活动了,而是必须还要等notify代码所在的封锁区域执行完毕从而释放锁以后,其它线程才可进入。

  由于wait()操作而半路出来的同志没收到notify信号前是不会再排队的,他会在旁边看着这些排队的人(其中修水管师傅也在其中)。注意,修水管的师傅不能插队,也得跟那些上厕所的人一样排队,不是说一个人蹲了一半出来后,修水管师傅就可以突然冒出来然后立刻进去抢修了,他要和原来排队的那帮人公平竞争,因为他也是个普通线程。如果修水管师傅排在后面,则前面的人进去后,发现堵了,就wait,然后出来站到一边,再进去一个,再wait,出来,站到一边,直到师傅进去修好后执行notify。这样,一会儿功夫,排队的旁边就站了一堆人,等着notify。

终于,师傅进去,然后修好了,接着notify了,接下来呢?

1. 有一个wait()的人(线程)被通知到。
2. 为什么被通知到的是他而不是另外一个wait()的人?取决于JVM。我们无法预先判断出哪一个会被通知到。也就是说,优先级高的不一定被优先唤醒,等待时间长的也不一定被优先唤醒,一切不可预知!(当然,如果你了解该JVM的实现,则可以预知)。
3. 他(被通知到的线程)要重新排队。
4. 他会排在队伍的第一个位置吗?回答是:不一定。他会排最后吗?也不一定。但如果该线程优先级设的比较高,那么他排在前面的概率就比较大。
5. 轮到他重新进入厕位时,他会从上次wait()的地方接着执行,不会重新执行。恶心点说就是,他会接着拉粑粑,不会重新拉。
6. 如果师傅notifyAll()。则那一堆半途而废出来的人全部重新排队,顺序不可知。

三、Lock的使用

用synchronized关键字可以对资源加锁。用Lock关键字也可以。它是JDK1.5中新增内容。

public class BoundedBuffer {

    final Lock lock = new ReentrantLock();
    final Condition notFull  = lock.newCondition();
    final Condition notEmpty = lock.newCondition(); 

    final Object[] items = new Object[100];
    int putptr, takeptr, count;

    public void put(Object x) throws InterruptedException {
        lock.lock();
        try {
            while (count == items.length)
                notFull.await();
            items[putptr] = x;
            if (++putptr == items.length) putptr = 0;
            ++count;
            notEmpty.signal();
        } finally {
            lock.unlock();
        }
    }

    public Object take() throws InterruptedException {
        lock.lock();
        try {
            while (count == 0)
                notEmpty.await();
            Object x = items[takeptr];
            if (++takeptr == items.length) takeptr = 0;
            --count;
            notFull.signal();
            return x;
        } finally {
            lock.unlock();
        }
    } 

}

(注:这是JavaDoc里的例子,是一个阻塞队列的实现例子。所谓阻塞队列,就是一个队列如果满了或者空了,都会导致线程阻塞等待。Java里的 ArrayBlockingQueue提供了现成的阻塞队列,不需要自己专门再写一个了。)

一个对象的lock.lock()和lock.unlock()之间的代码将会被锁住。这种方式比起synchronize好在什么地方?

简而言之,就是对wait的线程进行了分类。用厕位理论来描述,则是那些蹲了一半而从厕位里出来等待的人原因可能不一样,有的是因为马桶堵了,有的是因为马桶没水了。通知(notify)的时候,就可以喊:因为马桶堵了而等待的过来重新排队(比如马桶堵塞问题被解决了),或者喊,因为马桶没水而等待的过来重新排队(比如马桶没水问题被解决了)。这样可以控制得更精细一些。不像synchronize里的wait和notify,不管是马桶堵塞还是马桶没水都只能喊:刚才等待的过来排队!假如排队的人进来一看,发现原来只是马桶堵塞问题解决了,而自己渴望解决的问题(马桶没水)还没解决,只好再回去等待(wait),白进来转一圈,浪费时间与资源。

Lock与synchronized对应关系:

synchronized wait notify notifyAll
Lock await signal signalAll

注意:不要在Lock方式锁住的块里调用wait、notify、notifyAll。

 四、volatile的使用

  volatile真正解决的问题是 JVM 在-server模式下(注意普通运行模式下没有此问题),线程优先取用自己的线程私有stack中的变量值,而不是公共堆中的值,造成变量值老旧的问题。换句话说,volatile强制要求了所有线程在使用volatile修饰的变量的时候要去公共内存堆中获取值,不可以偷懒使用自己的。volatile绝对不保证原子性,原子性只能用synchronized同步修饰符实现。

  我们知道,在Java中设置变量值的操作,除了long和double类型的变量外都是原子操作,也就是说,对于变量值的简单读写操作没有必要进行同步。这在JVM 1.2之前,Java的内存模型实现总是从主存读取变量,是不需要进行特别的注意的。而随着JVM的成熟和优化,现在在多线程环境下volatile关键字的使用变得非常重要。

  在当前的Java内存模型下,线程可以把变量保存在本地内存(比如机器的寄存器)中,而不是直接在主存中进行读写。这就可能造成一个线程在主存中修改了一个变量的值,而另外一个线程还继续使用它在寄存器中的变量值的拷贝,造成数据的不一致。要解决这个问题,只需要把该变量声明为volatile(不稳定的)即可,这就指示JVM,这个变量是不稳定的,每次使用它都到主存中进行读取。一般说来,多任务环境下各任务间共享的标志都应该加volatile修饰。

  volatile修饰的成员变量在每次被线程访问时,都强迫从共享内存中重读该成员变量的值。而且,当成员变量发生变化时,强迫线程将变化值回写到共享内存。这样在任何时刻,两个不同的线程总是看到某个成员变量的同一个值。

  Java语言规范中指出:为了获得最佳速度,允许线程保存共享成员变量的私有拷贝,而且只当线程进入或者离开同步代码块时才与共享成员变量的原始值对比。这样当多个线程同时与某个对象交互时,就必须要注意到要让线程及时的得到共享成员变量的变化。而volatile关键字就是提示VM:对于这个成员变量不能保存它的私有拷贝,而应直接与共享成员变量交互。

  使用建议:在两个或者更多的线程访问的成员变量上使用volatile。当要访问的变量已在synchronized代码块中,或者为常量时,不必使用。由于使用volatile屏蔽掉了VM中必要的代码优化,所以在效率上比较低,因此一定在必要时才使用此关键字。

时间: 2024-10-12 04:11:00

关于Java中锁的总结的相关文章

java中锁机制

一段synchronized的代码被一个线程执行之前,他要先拿到执行这段代码的权限,在java里边就是拿到某个同步对象的锁(一个对象只有一把锁): 如果这个时候同步对象的锁被其他线程拿走了,他(这个线程)就只能等了(线程阻塞在锁池等待队列中). 取到锁后,他就开始执行同步代码(被synchronized修饰的代码):线程执行完同步代码后马上就把锁还给同步对象,其他在锁池中等待的某个线程就可以拿到锁执行同步代码了.这样就保证了同步代码在统一时刻只有一个线程在执行.   众所周知,在Java多线程编

Java中锁的级别

Java中的锁按等级分可以分为对象锁.方发锁.类锁. java的对象锁和类锁:java的对象锁和类锁在锁的概念上基本上和内置锁是一致的,但是, 两个锁实际是有很大的区别的,对象锁是用于对象实例方法,或者一个对象实例上的,类 锁是用于类的静态方法或者一个类的class对象上的.我们知道,类的对象实例可以有很 多个,但是每个类只有一个class对象,所以不同对象实例的对象锁是互不干扰的,但是 每个类只有一个类锁.但是有一点必须注意的是,其实类锁只是一个概念上的东西,并 不是真实存在的,它只是用来帮助

java中锁的应用

锁作为并发共享数据,保证一致性的工具,在JAVA平台有多种实现(如 synchronized(重量级) 和 ReentrantLock(轻量级)等等 ) .这些已经写好提供的锁为我们开发提供了便利. 1.重入锁 重入锁,也叫做递归锁,指的是同一线程 外层函数获得锁之后 ,内层递归函数仍然有获取该锁的代码,但不受影响.synchronized(重量级) 和 ReentrantLock(轻量级)都属于可重入锁. synchronized 和 Lock的区别 synchronize是重量级锁,使用结束

java中锁的概念/介绍

前言 Java提供了种类丰富的锁,每种锁因其特性的不同,在适当的场景下能够展现出非常高的效率.本文旨在对锁相关源码(本文中的源码来自JDK 8和Netty 3.10.6).使用场景进行举例,为读者介绍主流锁的知识点,以及不同的锁的适用场景. Java中往往是按照是否含有某一特性来定义锁,我们通过特性将锁进行分组归类,再使用对比的方式进行介绍,帮助大家更快捷的理解相关知识.下面给出本文内容的总体分类目录: ? 1. 乐观锁 VS 悲观锁 乐观锁与悲观锁是一种广义上的概念,体现了看待线程同步的不同角

Java中锁分类

锁的分类大致如下:公平锁/非公平锁可重入锁/不可重入锁独享锁/共享锁乐观锁/悲观锁分段锁 1.公平锁/非公平锁公平锁就是严格按照线程启动的顺序来执行的,不允许其他线程插队执行的:而非公平锁是允许插队的. 默认情况下 ReentrantLock 和 synchronized 都是非公平锁.ReentrantLock 可以设置成公平锁. 2.可重入锁/不可重入锁可重入锁指同一个线程可以再次获得之前已经获得的锁,避免产生死锁. ReenTrantLock可以指定是公平锁还是非公平锁.而synchron

JAVA中锁的使用

关系性锁: Lock: 锁可以保证线程的执行是安全的,使线程在执行时,只有执行完一个线程才能执行其他线程.任何时刻只有一个线程才能进入临界区,一旦一个线程封锁了锁对象,其他线程将无法通过lock语句.锁是可重入的,线程可以重复获得持有的锁.锁保持一个持有计数来跟踪锁的嵌套调用.每一次调用lock都要调用unlock来释放锁. 1.基本步骤 Lock myLock = new ReentrantLock(); public void method(){ myLock.lock(); try{ ..

在 Java 中高效使用锁的技巧--转载

竞争锁是造成多线程应用程序性能瓶颈的主要原因 区分竞争锁和非竞争锁对性能的影响非常重要.如果一个锁自始至终只被一个线程使用,那么 JVM 有能力优化它带来的绝大部分损耗.如果一个锁被多个线程使用过,但是在任意时刻,都只有一个线程尝试获取锁,那么它的开销要大一些.我们将以上两种锁称为非竞争锁.而对性能影响最严重的情况出现在多个线程同时尝试获取锁时.这种情况是 JVM 无法优化的,而且通常会发生从用户态到内核态的切换.现代 JVM 已对非竞争锁做了很多优化,使它几乎不会对性能造成影响.常见的优化有以

深入理解Java中的锁(一)

Java中锁的概念 自旋锁 : 是指当一个线程在获取锁的时候,如果锁已经被其他线程获取,那么该线程将循环等待,然后不断判断锁是否能够被成功获取,直到获取到锁才会退出循环. 乐观锁 : 假定没有冲突,在修改数据时如果发现数据和之前获取的不一致,则读最新数据,修改后重试修改 悲观锁 :假定会发生并发冲突,同步所有对数据的相关操作,从读数据就开始上锁 独享锁(写) : 给资源加上写锁,拥有该锁的线程可以修改资源,其他线程不能再加锁(单写) 共享锁(读) : 给资源加上读锁后只能读不能改,其他线程也只能

java中线程安全的定义,实现等

线程安全的定义: 当多个线程访问某个类时,不管运行时环境采用何种调度方式活着这些线程如何交互执行,并且在主调用代码中不需要任何额外的同步或者协同操作,这个类都能表现出正确的行为,那么这就称这个类是线程安全的 线程安全的类中,封装了必要的同步机制,因我们的主调用代码并不需要进一步的采取同步措施 竞态条件: 由于不恰当的执行时序,而出现的不正确的结果,就是竞态条件,举个例子:a线程需要将i的原始值加一赋予i,b线程需要将a线程执行后的i再+1,赋予j,这个时候如果b线程先执行,那么只鹅个结果就是错误