java同步技术的几种方法探讨
2014-11-26 12:31
351 查看
1.对象实例的锁
[java]
view plaincopyprint?
class Test{ //第一种同步方式 public synchronized void f1(){ //do something here } //第二种同步方式 public void f2(){ synchronized(this){ //do something here } } }
上面的f1()和f2()效果一致, synchronized取得的锁都是Test某个实列(this)的锁. 比如: Test t = new Test();
线程A调用t.f2()时, 线程B无法进入t.f1(),直到t.f2()结束。解说:这也就是同步方法,那这时synchronized锁定的是哪个对象呢?他锁定的是调用这个同步方法对象。也就是说,当一个对象P1在不同的线程中执行这个同步方法时,他们(同一个实例里的同步方法们)之间会形成互斥,达到同步的效果。但是这个对象所属的Class所产生的另一对象P2却能够任意调用这个被加了synchronized关键字的方法。
那个拿到了P1对象锁的线程,才能够调用P1的同步方法,而对P2而言,P1这个锁和他毫不相干,程式也可能在这种情形下摆脱同步机制的控制,造成数据混乱。
作用: 同一个实例的同步方法在多线程中访问时会有同步效果.
2. class的锁
class Test{
final static byte[] o= new byte[0]; //可以为静态,也可以非静态,看你的需求咯
//同步方式3
public static synchronized void f1(){
//do something here
}
//同步方式4
public static void f2(){
synchronized(Test.class){
//do something here
}
}
//同步方式5
public static void f3(){
try {
synchronized (Class.forName("Test")){
//do something here
}
} catch (ClassNotFoundException ex) {
}
}
//同步方式6
public static void g(){
synchronized(o){
//do something here
}
}
}
解说:
上面f1(),f2(),f3(),g()效果一致
f1(),f2(),f3()中synchronized取得的锁都是Test.class的锁.
g()是自己产生一个对象o,利用o的锁做同步
作用: 多线程中访问此类或此类任一个实例的同步方法时都会同步.(同一个类中多个同步方法是否回受影响呢?比如P1访问了同步方法a1(), 那么P2还能访问同步方法a2()吗?)
注:零长度的byte数组对象创建起来将比任何对象都经济――查看编译后的字节码:生成零长度的byte[]对象只需3条操作码,而Objectlock = new Object()则需要7行操作码。
3.静态方法同步
[java]
view plaincopyprint?
class Test{ private static int v = 0; public static void f1(){ //do something, 但函数中没用用到v } public synchronized static void f2(){ //do something, 函数中对v进行了读/写. } }
多线程中使用Test的某个实列时,
(1) f1()是线程安全的,不需要同步
(2) f2()这个静态方法中使用了函数外静态变量,所以需要同步.
更有一些技巧能够让我们对共享资源的同步访问更加安全:
1. 定义private 的instance变量+他的 get方法,而不要定义public/protected的instance变量。假如将变量定义为public,对象在外界能够绕过同步方法的控制而直接取得他,并改变他。这也是JavaBean的标准实现方式之一。
2. 假如instance变量是个对象,如数组或ArrayList什么的,那上述方法仍然不安全,因为当外界对象通过get方法拿到这个instance对象的引用后,又将其指向另一个对象,那么这个private变量也就变了,岂不是很危险。 这个时候就需要将get方法也加上synchronized同步,并且,只返回这个private对象的clone()――这样,调用端得到的就是对象副本的引用了。
[java]
view plaincopyprint?
class Test{ //第一种同步方式 public synchronized void f1(){ //do something here } //第二种同步方式 public void f2(){ synchronized(this){ //do something here } } }
class Test{ //第一种同步方式 public synchronized void f1(){ //do something here } //第二种同步方式 public void f2(){ synchronized(this){ //do something here } } }
上面的f1()和f2()效果一致, synchronized取得的锁都是Test某个实列(this)的锁. 比如: Test t = new Test();
线程A调用t.f2()时, 线程B无法进入t.f1(),直到t.f2()结束。解说:这也就是同步方法,那这时synchronized锁定的是哪个对象呢?他锁定的是调用这个同步方法对象。也就是说,当一个对象P1在不同的线程中执行这个同步方法时,他们(同一个实例里的同步方法们)之间会形成互斥,达到同步的效果。但是这个对象所属的Class所产生的另一对象P2却能够任意调用这个被加了synchronized关键字的方法。
那个拿到了P1对象锁的线程,才能够调用P1的同步方法,而对P2而言,P1这个锁和他毫不相干,程式也可能在这种情形下摆脱同步机制的控制,造成数据混乱。
作用: 同一个实例的同步方法在多线程中访问时会有同步效果.
2. class的锁
class Test{
final static byte[] o= new byte[0]; //可以为静态,也可以非静态,看你的需求咯
//同步方式3
public static synchronized void f1(){
//do something here
}
//同步方式4
public static void f2(){
synchronized(Test.class){
//do something here
}
}
//同步方式5
public static void f3(){
try {
synchronized (Class.forName("Test")){
//do something here
}
} catch (ClassNotFoundException ex) {
}
}
//同步方式6
public static void g(){
synchronized(o){
//do something here
}
}
}
解说:
上面f1(),f2(),f3(),g()效果一致
f1(),f2(),f3()中synchronized取得的锁都是Test.class的锁.
g()是自己产生一个对象o,利用o的锁做同步
作用: 多线程中访问此类或此类任一个实例的同步方法时都会同步.(同一个类中多个同步方法是否回受影响呢?比如P1访问了同步方法a1(), 那么P2还能访问同步方法a2()吗?)
注:零长度的byte数组对象创建起来将比任何对象都经济――查看编译后的字节码:生成零长度的byte[]对象只需3条操作码,而Objectlock = new Object()则需要7行操作码。
3.静态方法同步
[java]
view plaincopyprint?
class Test{ private static int v = 0; public static void f1(){ //do something, 但函数中没用用到v } public synchronized static void f2(){ //do something, 函数中对v进行了读/写. } }
class Test{ private static int v = 0; public static void f1(){ //do something, 但函数中没用用到v } public synchronized static void f2(){ //do something, 函数中对v进行了读/写. } }
多线程中使用Test的某个实列时,
(1) f1()是线程安全的,不需要同步
(2) f2()这个静态方法中使用了函数外静态变量,所以需要同步.
更有一些技巧能够让我们对共享资源的同步访问更加安全:
1. 定义private 的instance变量+他的 get方法,而不要定义public/protected的instance变量。假如将变量定义为public,对象在外界能够绕过同步方法的控制而直接取得他,并改变他。这也是JavaBean的标准实现方式之一。
2. 假如instance变量是个对象,如数组或ArrayList什么的,那上述方法仍然不安全,因为当外界对象通过get方法拿到这个instance对象的引用后,又将其指向另一个对象,那么这个private变量也就变了,岂不是很危险。 这个时候就需要将get方法也加上synchronized同步,并且,只返回这个private对象的clone()――这样,调用端得到的就是对象副本的引用了。
相关文章推荐
- java 同步的几种探讨
- java中有几种方法可以实现一个线程?用什么关键字修饰同步方法? stop()和suspend()方法为何不推荐使用?
- 聊聊java中几种不同的同步方法
- java开发线程篇1:java中有几种方法可以实现一个线程?用什么关键字修饰同步方法? stop()和suspend()方法为何不推荐使用?
- 【Java面试题】23 java中有几种方法可以实现一个线程?用什么关键字修饰同步方法? stop()和suspend()方法为何不推荐使用?
- java 多线程的同步几种方法
- java多线程有几种实现方法?线程之间如何同步
- java中有几种方法可以实现一个线程?用什么关键字修饰同步方法? stop()和suspend()方法为何不推荐使用
- java中有几种方法可以实现一个线程?用什么关键字修饰同步方法 stop()和suspend()方法为何不推荐使用?
- java有几种方法可以实现一个线程?用什么关键字修饰同步方法?
- Java中有几种方法实现一个线程,用什么关键字修饰同步方法?stop()和suspend()方法为何不提倡使用?
- java中同步的几种方法
- java的几种同步方法
- java 总结几种线程异步转同步的方法
- java基础问题---java中有几种方法可以实现一个线程?用什么关键字修饰同步方法? stop()和suspend()方法为何不推荐使用
- 【Java面试题】26 多线程有几种实现方法?同步有几种实现方法? 当一个线程进入一个对象的一个synchronized方法后,其它线程是否可进入此对象的其它方法?
- java多线程有几种实现方法?线程之间如何同步
- 四、java多线程核心技术——synchronized同步方法与synchronized同步快
- java中有几种方法可以实现一个线程?用什么关键字修饰同步
- 多线程有几种实现方法?同步有几种实现方法?静态同步方法和非静态有何不同?—— Java经典面试题系列