2.2.9静态同步synchronized方法与synchronized(class)代码块

synchronized还可以应用在static静态方法上,如果这样写,那是对当前的*.java文件对应的Class类进行持锁。

/**
 * @author wuyoushan
 * @date 2017/4/10.
 */
public class Service {

    synchronized  public static void printA(){
        try{
            System.out.println("线程名称为:"+Thread.currentThread().getName()
                    +"在"+System.currentTimeMillis()+"进入printA");
            Thread.sleep(3000);
            System.out.println("线程名称为:"+Thread.currentThread().getName()
                    +"在"+System.currentTimeMillis()+"离开printB");
        }catch (InterruptedException e){
            e.printStackTrace();
        }
    }

    synchronized  public static void printB(){
        System.out.println("线程名称为:"+Thread.currentThread().getName()
                +"在"+System.currentTimeMillis()+"进入printA");
        System.out.println("线程名称为:"+Thread.currentThread().getName()
                +"在"+System.currentTimeMillis()+"离开printB");
    }
}

 * @author wuyoushan
 * @date 2017/4/4.
 */
public class ThreadA extends Thread{

    @Override
    public void run() {
       super.run();
        Service.printA();
    }
}

/**
 * @author wuyoushan
 * @date 2017/4/4.
 */
public class ThreadB extends Thread{

    @Override
    public void run() {
        super.run();
        Service.printB();
    }
}

/**
 * @author wuyoushan
 * @date 2017/3/20.
 */
public class Run {
    public static void main(String[] args){
      ThreadA a=new ThreadA();
      a.setName("A");
      a.start();
      ThreadB b=new ThreadB();
      b.setName("B");
      b.start();
    }
}

程序的运行结果为:

线程名称为:A在1492993790410进入printA
线程名称为:A在1492993793410离开printA
线程名称为:B在1492993793410进入printB
线程名称为:B在1492993793410离开printB

从运行的结果来看,并没有什么特别之处,都是同步的效果,和将synchronized关键字加到static方法上使用的效果是一样的。其实还是有本子上的不同的,synchronized关键字加到static静态方法是是给Class类上锁的,而synchronized关键字加到非static静态方法上是给对象上锁。

验证不是同一个锁

/**
 * @author wuyoushan
 * @date 2017/4/10.
 */
public class Service {

    synchronized  public static void printA(){
        try{
            System.out.println("线程名称为:"+Thread.currentThread().getName()
                    +"在"+System.currentTimeMillis()+"进入printA");
            Thread.sleep(3000);
            System.out.println("线程名称为:"+Thread.currentThread().getName()
                    +"在"+System.currentTimeMillis()+"离开printA");
        }catch (InterruptedException e){
            e.printStackTrace();
        }
    }

    synchronized  public static void printB(){
        System.out.println("线程名称为:"+Thread.currentThread().getName()
                +"在"+System.currentTimeMillis()+"进入printB");
        System.out.println("线程名称为:"+Thread.currentThread().getName()
                +"在"+System.currentTimeMillis()+"离开printB");
    }

    synchronized  public void printC(){
        System.out.println("线程名称为:"+Thread.currentThread().getName()
                +"在"+System.currentTimeMillis()+"进入printC");
        System.out.println("线程名称为:"+Thread.currentThread().getName()
                +"在"+System.currentTimeMillis()+"离开printC");
    }
}

/**
 * @author wuyoushan
 * @date 2017/4/4.
 */
public class ThreadA extends Thread{

    private Service service;

    public ThreadA(Service service) {
        this.service = service;
    }

    @Override
    public void run() {
       super.run();
        service.printA();
    }
}

/**
 * @author wuyoushan
 * @date 2017/4/4.
 */
public class ThreadB extends Thread{

   private Service service;

    public ThreadB(Service service) {
        this.service = service;
    }

    @Override
    public void run() {
        super.run();
        service.printB();
    }
}

/**
 * @author wuyoushan
 * @date 2017/4/4.
 */
public class ThreadC extends Thread{

   private Service service;

    public ThreadC(Service service) {
        this.service = service;
    }

    @Override
    public void run() {
        super.run();
        service.printC();
    }
}

/**
 * @author wuyoushan
 * @date 2017/3/20.
 */
public class Run {
    public static void main(String[] args){
      Service service=new Service();
      ThreadA a=new ThreadA(service);
      a.setName("A");
      a.start();
      ThreadB b=new ThreadB(service);
      b.setName("B");
      b.start();
      ThreadC c=new ThreadC(service);
      c.setName("C");
      c.start();
    }
}

程序的运行结果为:

线程名称为:B在1492994339480进入printB
线程名称为:B在1492994339480离开printB
线程名称为:A在1492994339482进入printA
线程名称为:C在1492994339482进入printC
线程名称为:C在1492994339482离开printC
线程名称为:A在1492994342482离开printA

异步的原因是持有不同的锁,一个是对象锁,另外一个是Class锁,而Class锁可以对类的所有对象实例起作用

/**
 * @author wuyoushan
 * @date 2017/4/10.
 */
public class Service {

    synchronized  public static void printA(){
        try{
            System.out.println("线程名称为:"+Thread.currentThread().getName()
                    +"在"+System.currentTimeMillis()+"进入printA");
            Thread.sleep(3000);
            System.out.println("线程名称为:"+Thread.currentThread().getName()
                    +"在"+System.currentTimeMillis()+"离开printA");
        }catch (InterruptedException e){
            e.printStackTrace();
        }
    }

    synchronized  public static void printB(){
        System.out.println("线程名称为:"+Thread.currentThread().getName()
                +"在"+System.currentTimeMillis()+"进入printB");
        System.out.println("线程名称为:"+Thread.currentThread().getName()
                +"在"+System.currentTimeMillis()+"离开printB");
    }

}

/**
 * @author wuyoushan
 * @date 2017/4/4.
 */
public class ThreadA extends Thread{

    private Service service;

    public ThreadA(Service service) {
        this.service = service;
    }

    @Override
    public void run() {
       super.run();
        service.printA();
    }
}

/**
 * @author wuyoushan
 * @date 2017/4/4.
 */
public class ThreadB extends Thread{

   private Service service;

    public ThreadB(Service service) {
        this.service = service;
    }

    @Override
    public void run() {
        super.run();
        service.printB();
    }
}

/**
 * @author wuyoushan
 * @date 2017/3/20.
 */
public class Run {
    public static void main(String[] args){
      Service service1=new Service();
      Service service2=new Service();
      ThreadA a=new ThreadA(service1);
      a.setName("A");
      a.start();
      ThreadB b=new ThreadB(service2);
      b.setName("B");
      b.start();
    }
}

程序的运行结果为:

线程名称为:A在1492995107179进入printA
线程名称为:A在1492995110186离开printA
线程名称为:B在1492995110186进入printB
线程名称为:B在1492995110186离开printB

同步synchronized(class)代码块的作用其实和synchronized static方法的作用一样。

/**
 * @author wuyoushan
 * @date 2017/4/10.
 */
public class Service {

    public static void printA(){
        synchronized (Service.class){
            try {
                System.out.println("线程名称为:" + Thread.currentThread().getName()
                        + "在" + System.currentTimeMillis() + "进入printA");
                Thread.sleep(3000);
                System.out.println("线程名称为:" + Thread.currentThread().getName()
                        + "在" + System.currentTimeMillis() + "离开printA");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    public static void printB() {
        synchronized (Service.class) {
            System.out.println("线程名称为:" + Thread.currentThread().getName()
                    + "在" + System.currentTimeMillis() + "进入printB");
            System.out.println("线程名称为:" + Thread.currentThread().getName()
                    + "在" + System.currentTimeMillis() + "离开printB");
        }
    }

}

/**
 * @author wuyoushan
 * @date 2017/4/4.
 */
public class ThreadA extends Thread{

    private Service service;

    public ThreadA(Service service) {
        this.service = service;
    }

    @Override
    public void run() {
       super.run();
        service.printA();
    }
}

/**
 * @author wuyoushan
 * @date 2017/4/4.
 */
public class ThreadB extends Thread{

   private Service service;

    public ThreadB(Service service) {
        this.service = service;
    }

    @Override
    public void run() {
        super.run();
        service.printB();
    }
}

/**
 * @author wuyoushan
 * @date 2017/3/20.
 */
public class Run {
    public static void main(String[] args){
      Service service1=new Service();
      Service service2=new Service();
      ThreadA a=new ThreadA(service1);
      a.setName("A");
      a.start();
      ThreadB b=new ThreadB(service2);
      b.setName("B");
      b.start();
    }
}

程序的运行结果为:

线程名称为:A在1492995444496进入printA
线程名称为:A在1492995447497离开printA
线程名称为:B在1492995447497进入printB
线程名称为:B在1492995447497离开printB

摘选自 java多线程核心编程技术-2.2.9

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 32,644评论 18 399
  • 一:java概述:1,JDK:Java Development Kit,java的开发和运行环境,java的开发工...
    ZaneInTheSun阅读 7,618评论 0 11
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,492评论 11 349
  • Java8张图 11、字符串不变性 12、equals()方法、hashCode()方法的区别 13、...
    Miley_MOJIE阅读 9,157评论 0 11
  • 在年复一年的一日三餐中,怎么吃?吃什么? 会越吃越瘦,再不用借口,腰细了,皮肤更细腻了。 西葫芦才是最刮脂的疏菜,...
    财富健康1阅读 1,391评论 0 0

友情链接更多精彩内容