美文网首页Java线程与并发
(六) synchronized关键字

(六) synchronized关键字

作者: 覆水无言 | 来源:发表于2020-03-11 20:39 被阅读0次

    Java多线程目录

    1 synchronized是什么

    synchronized是Java在并发编程中的一个中要关键字,在并发编程中我们会遇到线程安全问题,synchronized就像是一把锁,而打开这把锁的钥匙只能一个线程持有,这样线程中共享变量的访问每次都会只有一个线程进行操作,这样就不会存在线程安全的问题。
    synchronized执行时会在进入代码处获得锁,离开synchronized修饰的代码释放锁。

    2 synchronized的使用

    synchronized同步锁使用主要有一下几个方面:

    1. 使用在代码块上。
    2. 使用在普通方法上。
    3. 使用在静态方法上。
    4. 使用在类上。

    2.1 修饰代码块

    synchronized修改代码块的结构是

    public void test(){
      synchronized(param){
        //TODO
      }
    }
    

    synchronized的锁是加在那,取决于它后面的参数param,如果param是this,这个锁就是加载这个对象上的,如果是其他的的对象,则这个锁是加在其他对象上的。

    public class ThreadOne implements Runnable {
    
        int i = 0;
        private Object object = new Object();
    
        public void run() {
            test();
            test2();
        }
    
        public void test() {
            synchronized (this) {
                i++;
                System.out.println(Thread.currentThread().getName() + " " + i);
            }
        }
    
        public void test2() {
            synchronized (object) {
                i++;
                System.out.println(Thread.currentThread().getName() + " " + i);
            }
        }
    
        public static void main(String[] args) {
            ThreadOne one = new ThreadOne();
            for (int i = 1; i < 100; i++) {
                new Thread(one, "" + i).start();
            }
        }
    }
    
    ------------------------------------------------------------
    
    public class ThreadOne extends Thread {
    
        int i = 0;
        private Object object = new Object();
    
        @Override
        public void run() {
            super.run();
            test();
            test2();
        }
    
        public void test() {
            synchronized (this) {
                i++;
                System.out.println(Thread.currentThread().getName() + " " + i);
            }
        }
    
        public void test2() {
            synchronized (object) {
                i++;
                System.out.println(Thread.currentThread().getName() + " " + i);
            }
        }
    
    
        public static void main(String[] args) {
            for (int i = 1; i < 1000; i++) {
                new ThreadOne( ).start();
            }
        }
    }
    

    上面两个例子可以猜出结果,第二个例子是无法得到正确结果的,因为synchronized的加锁是加在后面参数param上的,所以第一个例子他的锁加载this, object上,这里这个程序中this指ThreadOne, object是ThreadOne类对象中的一个成员变量,这里ThreadOne都是同一个对象,所以锁加在的对象也就只有一个。第二个例子是每次新建了一个ThreadOne类对象,一个ThreadOne对象中一个Object成员变量,这样其实synchronized加的锁都加在了不同的对象上,所以无法达到我们的预期。

    2.2 修饰普通方法

        public synchronized void test() {
            //todo
        }
    

    注意synchronized获得的是当前的对象锁,与synchronized(this)语义相同。多个线程对同一个对象的同步方法访问会竞争这个对象锁,每次只能有一个线程访问这个方法。

    2.3 直接加类锁

    上面synchronized(this)修饰的这个类的对象的锁,如果我们使用

    synchronized(ThreadOne.class){}
    

    则上面这个是类的锁,类锁表示的意思是这个锁对这个类的所有的对象都适用。当多个线程适用这个类时,不管有多少个对象,这里都是同步代码块,都只允许一个线程访问。

    修饰静态方法。

    //与普通方法使用一样。
    public static synchronized test(){
    }
    

    修饰讲台方法时这个锁是类锁,也就是说这个静态方法的同步调用是依据的这个类,而不是这个类的类对象。

    3 分类

    上面介绍了总计下来synchronized分为类对象锁和类锁,

    1. 类对象锁: 锁加到的是类的对象实例,多个线程同时访问同一个对象实例的同步方法时synchronized起作用
    2. 类锁: 类锁是加在类class上的,对该类的所有对象都适用,多线程访问多个该类对象实例的同步方法这个synchronized起作用。

    相关文章

      网友评论

        本文标题:(六) synchronized关键字

        本文链接:https://www.haomeiwen.com/subject/qmxhjhtx.html