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、在某一个时刻内,只能有一个线程持有锁,无论几个线程

原文地址:https://www.cnblogs.com/Mrchengs/p/10802406.html

时间: 2024-11-09 00:29:56

11、JUC--线程八锁的相关文章

(三)juc高级特性——虚假唤醒 / Condition / 按序交替 / ReadWriteLock / 线程八锁

8. 生产者消费者案例-虚假唤醒 参考下面生产者消费者案例: /* * 生产者和消费者案例 */ public class TestProductorAndConsumer { public static void main(String[] args) { Clerk clerk = new Clerk(); Productor pro = new Productor(clerk); Consumer cus = new Consumer(clerk); new Thread(pro, "生产

【转】线程八锁

package com.java.juc; /** * 题目:判断打印 "one" or "two" * * 1.两个普通同步方法,两个线程 ,标准打印,打印?// one two * 2.新增Thread.sleep(3000) 给getOne() 打印? // 3s 后打印 one two * 3.新增普通方法 getThreee 打印?// 先打印three 三秒后打印 one two * 4.两个普通同步方法,两个number对象,打印? // two 3s

9.线程八锁

线程八锁 1.当两个并发线程访问同一个对象object中的这个synchronized(this)同步代码块时,一个时间内只能有一个线程得到执行.另一个线程必须等待当前线程执行完这个代码块以后才能执行该代码块. 2.当一个线程访问一个实例对象的一个synchronized(this)同步代码块时,另一个线程仍然可以访问该实例对象中的synchronized(this)同步代码块. 3.当一个线程访问一个实例对象的一个synchronized(this)同步代码块时,其他线程对该对象中 /*所有其

java多线程 -- 线程八锁

一个对象里面如果有多个synchronized方法,某一个时刻内,只要一个线程去调用其中的一个synchronized方法了,其它的线程都只能等待,换句话说,某一个时刻内,只能有唯一一个线程去访问这些synchronized方法锁的是当前对象this,被锁定后,其它的线程都不能进入到当前对象的其它的synchronized方法 加个普通方法后发现和同步锁无关 换成两个对象后,不是同一把锁了,情况立刻变化. 都换成静态同步方法后,情况又变化 所有的非静态同步方法用的都是同一把锁--实例对象本身,也

线程八锁

锁的是当前对象this,被锁定后,其他线程都不能进入到当前对象的其他的synchronized方法. 所有的非静态同步方法用的都是同一把锁 -- 实例对象本身 所有的静态同步方法用的也是同一把锁 -- 类对象本身 /* * 题目:判断打印的 "one" or "two" ? * * 1. 两个普通同步方法,两个线程,标准打印, 打印? //one two * 2. 新增 Thread.sleep() 给 getOne() ,打印? //one two * 3. 新增

java多线程系类:JUC线程池:03之线程池原理(二)(转)

概要 在前面一章"Java多线程系列--"JUC线程池"02之 线程池原理(一)"中介绍了线程池的数据结构,本章会通过分析线程池的源码,对线程池进行说明.内容包括:线程池示例参考代码(基于JDK1.7.0_40)线程池源码分析(一) 创建"线程池"(二) 添加任务到"线程池"(三) 关闭"线程池" 转载请注明出处:http://www.cnblogs.com/skywang12345/p/3509954.h

Java - "JUC" ReentrantLock获取锁

[Java并发编程实战]-----"J.U.C":ReentrantLock之一简介 ReentrantLock介绍 ReentrantLock是一个可重入的互斥锁,又被称为"独占锁". 顾名思义,ReentrantLock锁在同一个时间点只能被一个线程锁持有:而可重入的意思是,ReentrantLock锁,可以被单个线程多次获取.ReentrantLock分为"公平锁"和"非公平锁".它们的区别体现在获取锁的机制上是否公平.

Java - "JUC线程池" 架构

Java多线程系列--"JUC线程池"01之 线程池架构 概要 前面分别介绍了"Java多线程基础"."JUC原子类"和"JUC锁".本章介绍JUC的最后一部分的内容--线程池.内容包括:线程池架构图线程池示例 转载请注明出处:http://www.cnblogs.com/skywang12345/p/3509903.html 线程池架构图 线程池的架构图如下: 1. Executor 它是"执行者"接口,

Java - "JUC线程池" ThreadPoolExecutor原理解析

Java多线程系列--"JUC线程池"02之 线程池原理(一) ThreadPoolExecutor简介 ThreadPoolExecutor是线程池类.对于线程池,可以通俗的将它理解为"存放一定数量线程的一个线程集合.线程池允许若个线程同时允许,允许同时运行的线程数量就是线程池的容量:当添加的到线程池中的线程超过它的容量时,会有一部分线程阻塞等待.线程池会通过相应的调度策略和拒绝策略,对添加到线程池中的线程进行管理." ThreadPoolExecutor数据结构