java synchronized同步静态方法和同步非静态方法的异同(淘宝面试问过此问题)
2014-02-21 15:29
429 查看
synchronized关键字有两种用法,一种是只用于方法的定义中,另外一种是synchronized块,我们不仅可以使用synchronized来同步一个对象变量,你也可以通synchronizedl来同步类中的静态方法和非静态方法。
synchronized块的语法如下:
[java] view
plaincopyprint?
public void method()
{
synchronized(表达式)
{
}
}
Java代码
public void method()
{
synchronized(表达式)
{
}
}
第一种:非静态方法的同步
从java相关语法可以知道使用synchronized关键字来定义方法就会锁定类中所用使用synchroniezd关键字定义的静态方法和非静态方法,但是这有点不好理解,如果要synchronized块,来达到这样的效果,就不难理解为什么会产生这种效果了,如果使用synchronized来锁定类中所有的同步非静态方法,只需要使用this作为synchronized块的参数传入synchronized块中,代码如下:
通过synchronized块来同步非静态方法
在上面的代码中的method1使用了synchronized块,method2方法是用了synchronized关键字来定义方法,如果使用同一个Test实例时,这两个方法只要有一个在执行,其他的方法都会因未获得同步锁而被堵塞。除了使用this作为synchronized块的参数,也可以使用Test.this作为synchronized块的参数来达到同样的效果。
[java] view
plaincopyprint?
public class Test
{
public void method1()
{
synchronized(this)
{
}
}
public synchronized void method2()
{
}
}
Java代码
public class Test
{
public void method1()
{
synchronized(this)
{
}
}
public synchronized void method2()
{
}
}
在内类中使用synchronized块中,this只表示内类,和外类(OuterClass)没有关系。但是内类中的非静态方法和外类的非静态方法也可以同步。如果在内类中加个方法method3也可以使和Test里面的2个方法同步,代码如下:
[java] view
plaincopyprint?
public class Test
{
class InnerClass
{
public void method3()
{
synchronized(Test.this){
}
}
}
}
Java代码
public class Test
{
class InnerClass
{
public void method3()
{
synchronized(Test.this){
}
}
}
}
上面InnerClass的method3方法与Test的method1和method2方法在同一时间内只能有一个方法执行。
synchronized块不管是正确执行完,还是因为程序出错因异常退出synchronized块,当前的synchronized块所持有的同步锁都会自动释放,因此在使用synchronized块不必担心同步锁的问题。
二、静态方法的同步
由于在调用静态方法时,对象实例不一定被创建,因此,就不能使用this来同步静态方法,而必须使用Class对象来同步静态方法。代码如下:
[java] view
plaincopyprint?
public class Test{
pubic static void method1(){
synchronized(Test.class){
}
}
public static synchronized void method2(){
}
}
Java代码
public class Test{
pubic static void method1(){
synchronized(Test.class){
}
}
public static synchronized void method2(){
}
}
在同步静态方法时可以使用类的静态字段class来得到class对象,在上例中method1和method2方法只有一个方法执行,除了使用class字段可以得到class对象,还可以通过实例的getClass()方法获取class对象,代码如下:
[java] view
plaincopyprint?
public class Test{
public static Test test;
public Test(){
test=this;
}
public static void method1(){
synchronized(test.getClass()){
}
}
}
Java代码
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类中方法同步。
[java] view
plaincopyprint?
public class Test1{
public static void method1(){
synchronized(Test.class){
}
}
}
Java代码
public class Test1{
public static void method1(){
synchronized(Test.class){
}
}
}
注意:在使用synchronized块来同步方法时,非静态方法可以通过this来同步,而静态方法必须使用class对象来同步,但是非静态方法也可以通过使用class来同步静态方法。但是静态方法中不能使用this来同步非静态方法。这点在使用synchronized块需要注意。
技术社区群:336866986
synchronized块的语法如下:
[java] view
plaincopyprint?
public void method()
{
synchronized(表达式)
{
}
}
Java代码
public void method()
{
synchronized(表达式)
{
}
}
第一种:非静态方法的同步
从java相关语法可以知道使用synchronized关键字来定义方法就会锁定类中所用使用synchroniezd关键字定义的静态方法和非静态方法,但是这有点不好理解,如果要synchronized块,来达到这样的效果,就不难理解为什么会产生这种效果了,如果使用synchronized来锁定类中所有的同步非静态方法,只需要使用this作为synchronized块的参数传入synchronized块中,代码如下:
通过synchronized块来同步非静态方法
在上面的代码中的method1使用了synchronized块,method2方法是用了synchronized关键字来定义方法,如果使用同一个Test实例时,这两个方法只要有一个在执行,其他的方法都会因未获得同步锁而被堵塞。除了使用this作为synchronized块的参数,也可以使用Test.this作为synchronized块的参数来达到同样的效果。
[java] view
plaincopyprint?
public class Test
{
public void method1()
{
synchronized(this)
{
}
}
public synchronized void method2()
{
}
}
Java代码
public class Test
{
public void method1()
{
synchronized(this)
{
}
}
public synchronized void method2()
{
}
}
在内类中使用synchronized块中,this只表示内类,和外类(OuterClass)没有关系。但是内类中的非静态方法和外类的非静态方法也可以同步。如果在内类中加个方法method3也可以使和Test里面的2个方法同步,代码如下:
[java] view
plaincopyprint?
public class Test
{
class InnerClass
{
public void method3()
{
synchronized(Test.this){
}
}
}
}
Java代码
public class Test
{
class InnerClass
{
public void method3()
{
synchronized(Test.this){
}
}
}
}
上面InnerClass的method3方法与Test的method1和method2方法在同一时间内只能有一个方法执行。
synchronized块不管是正确执行完,还是因为程序出错因异常退出synchronized块,当前的synchronized块所持有的同步锁都会自动释放,因此在使用synchronized块不必担心同步锁的问题。
二、静态方法的同步
由于在调用静态方法时,对象实例不一定被创建,因此,就不能使用this来同步静态方法,而必须使用Class对象来同步静态方法。代码如下:
[java] view
plaincopyprint?
public class Test{
pubic static void method1(){
synchronized(Test.class){
}
}
public static synchronized void method2(){
}
}
Java代码
public class Test{
pubic static void method1(){
synchronized(Test.class){
}
}
public static synchronized void method2(){
}
}
在同步静态方法时可以使用类的静态字段class来得到class对象,在上例中method1和method2方法只有一个方法执行,除了使用class字段可以得到class对象,还可以通过实例的getClass()方法获取class对象,代码如下:
[java] view
plaincopyprint?
public class Test{
public static Test test;
public Test(){
test=this;
}
public static void method1(){
synchronized(test.getClass()){
}
}
}
Java代码
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类中方法同步。
[java] view
plaincopyprint?
public class Test1{
public static void method1(){
synchronized(Test.class){
}
}
}
Java代码
public class Test1{
public static void method1(){
synchronized(Test.class){
}
}
}
注意:在使用synchronized块来同步方法时,非静态方法可以通过this来同步,而静态方法必须使用class对象来同步,但是非静态方法也可以通过使用class来同步静态方法。但是静态方法中不能使用this来同步非静态方法。这点在使用synchronized块需要注意。
技术社区群:336866986
相关文章推荐
- java静态方法同步问题
- day23--电影院买票问题解决 同步代码块 同步方法 静态同步方法的引入
- 学习总结:java面试方法重写详解,静态代码块和构造方法执行顺序问题
- 总结面试中问到的一个问题构造函数、静态代码块、和调用的方法的加载顺序
- 【最近面试遇到的一些问题】多线程有几种实现方法,都是什么?同步有几种实现方法,都是什么
- JAVA静态方法的同步问题~~~~~~~~~~~~~
- java synchronized同步静态方法和同步非静态方法的异同
- 看似简单的问题 静态方法和实例化方法的区别
- VS2010的调试问题:类中的静态构造方法和引用类中静态方法
- 在多线程中使用静态方法是否有线程安全问题
- 几个问题(八)-------使用静态工厂方法的好处和坏处
- 静态main方法内创建本身类对象不会重复初始化的问题
- 一个猥琐的方法:解决模板类静态成员重定义的问题
- 记录Java中方法同步问题
- 多线程下生产者消费者问题的五种同步方法实现
- 如何实现多个线程同步 (2013-11-10 12:07:24)转载▼ 标签: it 在编写一个类时,如果该类中的代码可能运行于多线程环境下,那么就要考虑同步的问题,Java实现线程同步的方法很多
- 线程同步问题 解决同步问题的方法 synchronized方法 同步代码块
- 多线程中使用静态方法是否有线程安全问题
- 面试问题java Object类有哪些方法
- 关于js中post方法遇到的同步问题(取不到json里的数据)