• 11、JUC--线程八锁


    线程八锁

    • 一个对象里面如果有多个synchronized方法,某一个时刻内,只要一个线程去调用
      其中的一个synchronized方法了,其它的线程都只能等待,换句话说,某一个时刻
      内,只能有唯一一个线程去访问这些synchronized方法


    • 锁的是当前对象this,被锁定后,其它的线程都不能进入到当前对象的其它的
      synchronized方法


    • 加个普通方法后发现和同步锁无关


    • 换成两个对象后,不是同一把锁了,情况立刻变化。


    • 都换成静态同步方法后,情况又变化


    • 所有的非静态同步方法用的都是同一把锁——实例对象本身,也就是说如果一个实

      例对象的非静态同步方法获取锁后,该实例对象的其他非静态同步方法必须等待获
      取锁的方法释放锁后才能获取锁,可是别的实例对象的非静态同步方法因为跟该实
      例对象的非静态同步方法用的是不同的锁,所以毋须等待该实例对象已获取锁的非
      静态同步方法释放锁就可以获取他们自己的锁。


    • 所有的静态同步方法用的也是同一把锁——类对象本身,这两把锁是两个不同的对
      象,所以静态同步方法与非静态同步方法之间是不会有竞态条件的。但是一旦一个
      静态同步方法获取锁后,其他的静态同步方法都必须等待该方法释放锁后才能获取
      锁,而不管是同一个实例对象的静态同步方法之间,还是不同的实例对象的静态同
      步方法之间,只要它们同一个类的实例对象!

    题目判打印的是one还是two?

    1、

    class Number{
        
        public synchronized void getOne(){
            System.out.println("one");
        }
        
        public synchronized void getTwo(){
            System.out.println("two");
        }
        
    }

    测试:

    public static void main(String[] args) {
        
            Number num = new Number();
            
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num.getOne();
                }
            }).start();
            
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num.getTwo();
                }
            }).start();
            
        }

    两个普通同部方法,两个线程标准打印

    2、

    新增方法让getOne方法让其睡眠三秒

    class Number{
        
        public synchronized void getOne(){
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            System.out.println("one");
        }
        
        public synchronized void getTwo(){
            System.out.println("two");
        }
        
    }

    新增Thread.sleep()方法给getOne()

    此时得到结果的顺序:

    3、

    新增一个普通方法

    class Number{
        
        public synchronized void getOne(){
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            System.out.println("one");
        }
        
        public synchronized void getTwo(){
            System.out.println("two");
        }
        
        public void getThree(){
            System.out.println("Three");
        }
        
    }

    测试类:

    Number num = new Number();
            
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num.getOne();
                }
            }).start();
            
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num.getTwo();
                }
            }).start();
            
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num.getThree();
                }
            }).start();

     

    4、

    两个普通同部方法,两个Number对象

    class Number{
        
        public synchronized void getOne(){
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("one");
        }
        
        public synchronized void getTwo(){
            System.out.println("two");
        }
        
        public void getThree(){
            System.out.println("Three");
        }
    }
         Number num = new Number();
            Number num1 = new Number();
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num.getOne();
                }
            }).start();
            
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num1.getTwo();
                }
            }).start();

    5、

    修改getOne()为静态同部方法

    class Number{
        public static synchronized void getOne(){
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            System.out.println("one");
        }
        
        public synchronized void getTwo(){
            System.out.println("two");
        }
        
        public void getThree(){
            System.out.println("Three");
        }
        
    }
            Number num = new Number();
            
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num.getOne();
                }
            }).start();
            
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num.getTwo();
                }
            }).start();

    6、

    修改两个方法均为静态同部方法

    class Number{
        
        public static synchronized void getOne(){
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            System.out.println("one");
        }
        
        public static synchronized void getTwo(){
            System.out.println("two");
        }
    }
            Number num = new Number();
            //Number num1 = new Number();
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num.getOne();
                }
            }).start();
            
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num.getTwo();
                }
            }).start();

     

    7、

    一个静态同部方法

    一个非静态同部方法

    两个对象

    class Number{
        
        public static synchronized void getOne(){
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("one");
        }
        
        public  synchronized void getTwo(){
            System.out.println("two");
        }
        
        public void getThree(){
            System.out.println("Three");
        }
        
    }
         Number num = new Number();
            Number num1 = new Number();
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num.getOne();
                }
            }).start();
            
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num1.getTwo();
                }
            }).start();

    8、

    两个静态同部方法

    两个对象

    class Number{
        
        public static synchronized void getOne(){
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("one");
        }
        
        public static  synchronized void getTwo(){
            System.out.println("two");
        }
    }
            Number num = new Number();
            Number num1 = new Number();
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num.getOne();
                }
            }).start();
            
            new Thread(new Runnable() {
                @Override
                public void run() {
                    num1.getTwo();
                }
            }).start();

     线程八锁的关键:

    1、非静态方法的锁默认为this,静态方法的锁为对应的Class实例

    2、在某一个时刻内,只能有一个线程持有锁,无论几个线程

  • 相关阅读:
    HackerRank
    HackerRank
    LeetCode "Kth Smallest Element in a BST"
    HackerRank
    HackerRank
    LeetCode "Roman to Integer"
    LeetCode "Integer to Roman"
    LeetCode "Majority Element II"
    HackerRank
    HackerRank
  • 原文地址:https://www.cnblogs.com/Mrchengs/p/10802406.html
Copyright © 2020-2023  润新知