学会在JAVA项目中使用synchronized关键字

synchronized为java中的关键字,用于保证多线程并发时,在同一时刻最多只有一个线程执行该段代码。

包括两种用法:synchronized 方法和 synchronized 块。

第一个Demo

public class ThreadTest extends Thread {   
    private int threadNo;   
    public ThreadTest(int threadNo) {   
        this.threadNo = threadNo;   
    }   
    public static void main(String[] args) throws Exception {   
        for (int i = 1; i < 10; i++) {   
           new ThreadTest(i).start();   
            Thread.sleep(1);   
        }   
     }   
    
    @Override  
     public synchronized void run() {   
        for (int i = 1; i < 100; i++) {   
            System.out.println("No." + threadNo + ":" + i);   
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }   
     }   
 }  

执行结果

No.1:1
No.2:1
No.3:1
No.4:1
No.5:1
No.6:1
No.7:1
No.8:1
No.9:1........

很显然并没有实现同步操作,代码中在run()方法上面加了synchronized关键字,那为什么没有实现同步呢

首先明确一点,对于一个成员方法加synchronized关键字,这实际上是以这个成员方法所在的对象本身作为对象锁。

这段代码其实就是以ThreadTest类的一个具体对象作为对象锁的。一共十个线程,每个线程持有自己线程对象的那个对象锁。这必然不能产生同步的效果

第二个Demo

public class ThreadTest2 extends Thread{
    private int threadNo; 
    private String lock = "lock";
    public ThreadTest2(int threadNo) {   
        this.threadNo = threadNo;   
    }   
    public static void main(String[] args) throws Exception {   
        for (int i = 1; i < 10; i++) {   
           new ThreadTest2(i).start();   
            Thread.sleep(1);   
        }   
     }   
    
    @Override  
     public  void run() {  
        synchronized (lock) {
             for (int i = 1; i < 100; i++) {   
                    System.out.println("No." + threadNo + ":" + i);   
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }   
        }
     }   
}

执行结果

No.1:1
No.1:2
No.1:3
No.1:4
No.1:5
No.1:6
No.1:7
No.1:8
No.1:9......

很显然这段代码实现了同步,在run()方法中使用了同步代码块,并且以“lock”作为锁

以一个String类型的对象为锁,根据Java方法的传值特点,lock为同一个String对象,所以该对象锁是共享且唯一的

第三个Demo

public class ThreadTest3 extends Thread{
    private int threadNo;   
    public ThreadTest3(int threadNo) {   
        this.threadNo = threadNo;   
    }   
    public static void main(String[] args) throws Exception {   
        for (int i = 1; i < 10; i++) {   
           new ThreadTest3(i).start();   
            Thread.sleep(1);   
        }   
     }   
    
     public static synchronized void dowork(int threadNo) {   
        for (int i = 1; i < 100; i++) {   
            System.out.println("No." + threadNo + ":" + i);   
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }   
     } 
     
     @Override
    public void run() {
        dowork(threadNo);
    }    
}

执行结果

No.1:1
No.1:2
No.1:3
No.1:4
No.1:5
No.1:6
No.1:7
No.1:8............

很显然这段代码也实现了同步~ 他和第一个demo一样是把synchronized关键字放在方法上面,唯一的不一样就是,dowork方法使用了static关键字修饰

由于在JVM中,所有被加载的类都有唯一的类对象

对于同步静态方法,对象锁就是该静态放发所在的类的Class对象,唯一的 ThreadTest3.class对象

虽然在main方法中创建了该类的多个实例,但是它的类对象只有一个,所以对象锁是共享且唯一的

总结

1、 对于同步的方法或者代码块来说,必须获得对象锁才能够进入同步方法或者代码块进行操作;

2、 如果采用method级别的同步,则对象锁即为method所在的对象,如果是静态方法,对象锁即指method所在的
Class对象(唯一);

3、 对于代码块,对象锁即指synchronized(abc)中的abc;

4、 静态方法则一定会同步,非静态方法需在单例模式才生效,推荐用静态方法

顺便说一下:

Java中多线程锁释放的条件:

1、执行完同步代码块,就会释放锁。(synchronized)

2、在执行同步代码块的过程中,遇到异常而导致线程终止,锁也会被释放。(exception)

3、在执行同步代码块的过程中,执行了锁所属对象的wait()方法,这个线程会释放锁,进入对象的等待池。(wait)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Java8张图 11、字符串不变性 12、equals()方法、hashCode()方法的区别 13、...
    Miley_MOJIE阅读 3,731评论 0 11
  • 一:java概述:1,JDK:Java Development Kit,java的开发和运行环境,java的开发工...
    ZaneInTheSun阅读 2,688评论 0 11
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,766评论 18 399
  • 本文出自 Eddy Wiki ,转载请注明出处:http://eddy.wiki/interview-java.h...
    eddy_wiki阅读 2,244评论 0 14
  • 感赏: 昨晚老公接女儿下晚自习回家,女儿要求买零食用书包背去学校,学校不允许带零食去,老公跟女儿说了这样不合适,女...
    贵州六年级庄杰阅读 127评论 0 1