`
Iam42
  • 浏览: 275818 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

java synchronized同步静态方法和同步非静态方法的异同

    博客分类:
  • Java
阅读更多

-------------------转自CSDN lonely_fireworks 的博客-----------------------

synchronized关键字有两种用法,一种是只用于方法的定义中,另外一种是synchronized块,我们不仅可以使用synchronized来同步一个对象变量,你也可以通synchronizedl来同步类中的静态方法和非静态方法。

synchronized块的语法如下:

  1. public void method()  
  2. {  
  3.     synchronized(表达式)  
  4.      {  
  5.      }  
  6.   
  7. }  
public void method()
{
    synchronized(表达式)
     {
     }

}

第一种:非静态方法的同步

从java相关语法可以知道使用synchronized关键字来定义方法就会锁定类中所用使用synchroniezd关键字定义的静态方法和非静态方法,但是这有点不好理解,如果要synchronized块,来达到这样的效果,就不难理解为什么会产生这种效果了,如果使用synchronized来锁定类中所有的同步非静态方法,只需要使用this作为synchronized块的参数传入synchronized块中,代码如下:

  通过synchronized块来同步非静态方法

在上面的代码中的method1使用了synchronized块,method2方法是用了synchronized关键字来定义方法,如果使用同一个Test实例时,这两个方法只要有一个在执行,其他的方法都会因未获得同步锁而被堵塞。除了使用this作为synchronized块的参数,也可以使用Test.this作为synchronized块的参数来达到同样的效果。

 

  1. public class Test  
  2. {  
  3.   public void method1()  
  4.   {  
  5.     synchronized(this)  
  6.      {  
  7.   
  8.      }  
  9.   }  
  10.   
  11.  public synchronized void method2()  
  12.   {  
  13.   
  14.   }  
  15. }  
public class Test
{
  public void method1()
  {
    synchronized(this)
     {

     }
  }

 public synchronized void method2()
  {

  }
}

 

在内类中使用synchronized块中,this只表示内类,和外类(OuterClass)没有关系。但是内类中的非静态方法和外类的非静态方法也可以同步。如果在内类中加个方法method3也可以使和Test里面的2个方法同步,代码如下:

 

  1. public class Test  
  2. {  
  3.   class InnerClass  
  4.   {  
  5.     public void method3()  
  6.      {  
  7.        synchronized(Test.this){  
  8.   
  9.         }  
  10.       }  
  11.    }  
  12. }  
public class Test
{
  class InnerClass
  {
    public void method3()
     {
       synchronized(Test.this){

        }
      }
   }
}

 

 

上面InnerClass的method3方法与Test的method1和method2方法在同一时间内只能有一个方法执行。

 

synchronized块不管是正确执行完,还是因为程序出错因异常退出synchronized块,当前的synchronized块所持有的同步锁都会自动释放,因此在使用synchronized块不必担心同步锁的问题。

二、静态方法的同步

由于在调用静态方法时,对象实例不一定被创建,因此,就不能使用this来同步静态方法,而必须使用Class对象来同步静态方法。代码如下:

  1. public class Test{  
  2.   
  3.  pubic static void method1(){  
  4.    synchronized(Test.class){  
  5.     }  
  6.  }  
  7.   public static synchronized void method2(){  
  8.   
  9.    }  
  10. }  
public class Test{

 pubic static void method1(){
   synchronized(Test.class){
    }
 }
  public static synchronized void method2(){

   }
}


 

 

在同步静态方法时可以使用类的静态字段class来得到class对象,在上例中method1和method2方法只有一个方法执行,除了使用class字段可以得到class对象,还可以通过实例的getClass()方法获取class对象,代码如下:

  1. public class Test{  
  2.  public static Test test;  
  3.  public Test(){  
  4.  test=this;  
  5.  }  
  6.  public static void method1(){  
  7.  synchronized(test.getClass()){  
  8.  }  
  9.  }  
  10. }  
public class Test{
 public static Test test;
 public Test(){
 test=this;
 }
 public static void method1(){
 synchronized(test.getClass()){
 }
 }
}

 

在上面的代码中,我们通过一个public的静态对象得到Test的一个实例,并通过这个实例的getClass方法获取一个class对象(注意一个类的所有实例通过getClass方法得到的都是同一个Class对象)。我们也可以通过class使不同类的静态方法同步,代码如下:

 

Test类中的方法和Test1类中方法同步。

  1. public class Test1{  
  2.  public static void method1(){  
  3.   synchronized(Test.class){  
  4.    }  
  5.  }  
  6. }  
public class Test1{
 public static void method1(){
  synchronized(Test.class){
   }
 }
}

 

注意:在使用synchronized块来同步方法时,非静态方法可以通过this来同步,而静态方法必须使用class对象来同步,但是非静态方法也可以通过使用class来同步静态方法。但是静态方法中不能使用this来同步非静态方法。这点在使用synchronized块需要注意。

分享到:
评论

相关推荐

    华为JAVA面试题

    静态方法中不能直接调用非静态变量,因为非静态变量是属于实例的,而静态方法属于类。反过来,实例方法可以调用静态变量。 4. 继承与接口的区别 继承和接口都是面向对象程序设计中实现代码复用的重要机制。 - ...

    java面试宝典

    77、简述synchronized和java.util.concurrent.locks.Lock的异同 ? 18 78、abstract class Name { private String name; public abstract boolean isStupidName(String name) {}}这有何错误? 18 79、public class ...

    Java面试宝典2013_PDF版.pdf

    54. 简述synchronized和java.util.concurrent.locks.Lock的异同? synchronized是内置的Java语言关键字,Lock是java.util.concurrent.locks包下的一个接口。synchronized在执行完或者遇到异常时会自动释放锁,而...

    Java程序员面试宝典——2012版

    不可以直接从静态方法内部发出对非静态方法的调用,因为静态方法属于类级别,而非静态方法属于对象级别,调用非静态方法需要先创建对象。 #### 14. `Integer`与`int` `Integer`是`int`的包装类,它提供了额外的...

    最经典的Java面试宝典

    不可以直接从静态方法中调用非静态方法,因为非静态方法依赖于对象的存在。 **14、`Integer`与`int`的区别** `Integer`是`int`的封装类型,可以存储null,用于集合和其他需要对象的地方。`int`是基本数据类型。 **...

    2021年Java大厂面试题整理大全

    4. 静态与非静态:区分静态成员和非静态成员的使用场景。 5. 内存管理:了解堆内存、栈内存以及垃圾回收机制。 二、面向对象 1. 接口与抽象类:比较接口和抽象类的异同,理解它们在设计模式中的应用。 2. 多态性:...

    一些Java面试题库

    Java接口仅包含抽象方法和常量,不允许有任何实现代码,而C++的虚类可以有成员变量和方法实现。Java中的类可以实现多个接口,但C++的类只能单继承。接口提供了更大的灵活性,适用于完全不同的类之间共享行为,而虚类...

    Java面试宝典2011版

    - 不可以直接调用非静态方法,因为静态方法是在任何对象实例化之前就已经存在的,而调用非静态方法需要先创建对象实例。 **14. `Integer`与`int`的区别** - `int`是基本数据类型,而`Integer`是`int`的包装类,它...

    java 面试宝典

    - Java中有多种方法实现线程,关键字synchronized用于同步控制。 - sleep()和wait()都可以使当前线程暂停执行,但wait()通常与notify()或notifyAll()一起使用以实现线程间的通信。 3. Java多线程编程 - 启动一个...

    Java面试宝典2022版(超长版).docx

    - **不可以**: 静态方法不能直接调用非静态方法,因为静态方法是在类加载时初始化的,而非静态方法依赖于对象的存在。 #### 14. `Integer`与`int`的区别 - **`int`**: 基本数据类型。 - **`Integer`**: 包装类,...

    Java基础知识方面的面试题

    - 继承:一个类可以继承另一个类的属性和方法,实现代码复用。 - 多态:同一个接口,使用不同的实例而表现出不同的行为。 5. **Java中的数据类型有哪些?** Java有两大类数据类型:基本数据类型(如int、char、...

    java面试宝典(最新版)

    不可以直接调用非静态方法,因为静态方法在类加载时就已经存在,而非静态方法依赖于具体的对象实例。 #### 14. `Integer`与`int`的区别 - `Integer`是`int`的封装类,可以为null。 - `int`是基本数据类型,不可为...

    Java面试宝典2012最新版

    26. 内部类分为静态内部类和非静态内部类,它们在访问外部类成员时有不同的规则和限制。 27. 内部类可以引用包含类的成员,但要遵循相应的访问权限。 28. 匿名内部类可以实现接口或继承类,但不可以有显式构造函数。...

    Java面试宝典2010版

    - **详细解析**:由于静态方法是属于类级别的,而非静态方法属于对象级别,因此静态方法无法直接调用非静态方法,除非先创建了一个对象。 ##### 14、Integer与int的区别 - **答案**: - `int`是基本数据类型。 -...

    java 面试题

    Java中的多态主要通过方法重载(overloading)和方法重写(overriding)来实现。此外,使用接口和抽象类也是实现多态的一种方式。 #### 24\. abstractclass和interface有什么区别? - **抽象类**:可以有默认方法...

Global site tag (gtag.js) - Google Analytics