概括总结
在这个版本中,把提交订单的方法设置成public static synchronized
的,与仅仅是设置public synchronized
的,在提交订单的性能上有什么差别?答案是:没有明显区别(是同样的差)。
008版本更新说明
复制出了一个文件,如下图,其中Version008Static
类中的提交订单方法是public static synchronized
的,Version008Normal
类中的提交订单方法是public synchronized
的,其他没有差别。
测试结果
统计10次测试之后的平均值之后:
调用Version008Normal
类中的提交订单方法时,每秒钟可以提交的订单数为:9
调用Version008Static
类中的提交订单方法时,每秒钟可以提交的订单数为:9
两者是一样的。当然,由于9.9
向下取整也是9,9.0
向下取整也是9,所以从这个值来估算的话,最高还是有可能存在不超过10%的误差。
那么可以对比另一个值:
调用Version008Normal
类中的提交订单方法时,提交每个订单平均耗时的纳秒数:102392535
调用Version008Static
类中的提交订单方法时,提交每个订单平均耗时的纳秒数:101756665
性能差别为:(102392535 - 101756665) / 102392535 * 100% = 0.62%,也没有什么明显差别。
【备注】:不同的机器上的测试结果会不一样,以上测试结果仅供参考。
为什么没有明显差别
当一个方法被标记为public static synchronized
时,它的含义是:同一时间只有一个线程能访问属于这个类的任意一个static synchronized
方法。
- 这里锁的是
Class
对象,同一个JVM中,同一个类只会有同一个Class对象。 - 什么叫"属于这个类"?大范围为说,类中的一切都是属于这个类的,不过我们现在讨论的是
同步
这个小范围,所以可以认为static synchronized
标记的方法就是属于这个类的。
当一个方法被标记为public synchronized
时,它的含义是:同一时间只有一个线程能访问属于这个类的的当前这个实例对象的任意一个synchronized
方法。
- 这里锁的是普通的类的实例对象,同一个JVM中,理论上可以存在无数个同一个类的不同的实例对象。
- 什么叫"属于这个对象"?从
同步
的角度来说,就是标记为synchronized
但是没有被标记为static
的方法。
在这次测试中,上锁的Class
对象只有一个,就是Version008Static.class
,上锁的实例对象也只有一个,就是从Version008Normal.class
new 出来的一个实例,而类中又都只有一个synchronized
的方法,不存在多个线程同时调用同一个方法的情况,两种测试都相当于被转换成了单线程测试,因此没有差别。
补充说明
在正常项目中开发时,不可能把一个提交订单的方法设置为public static
的,所以这一版本的测试仅仅只是个示例,只是出于好奇想看看结果是怎样的,没有任何参考价值。从下版本开始,将不再使用public static
方法。
源码
008版本的github源码在这里,
如果不知道怎样运行项目,请参考这里