一、当两个并发线程访问同一个对象object中的这个synchronized(this)同步代码块时,一个时间内只能有一个线程得到执行。另一个线程必须等待当前线程执行完这个代码块以后才能执行该代码块。

二、然而,当一个线程访问object的一个synchronized(this)同步代码块时,另一个线程仍然可以访问该object中的非synchronized(this)同步代码块。

三、尤其关键的是,当一个线程访问object的一个synchronized(this)同步代码块时,其他线程对object中所有其它synchronized(this)同步代码块的访问将被阻塞。

四、第三个例子同样适用其它同步代码块。也就是说,当一个线程访问object的一个synchronized(this)同步代码块时,它就获得了这个object的对象锁。结果,其它线程对该object对象所有同步代码部分的访问都被暂时阻塞。

五、以上规则对其它对象锁同样适用.

举例说明:

一、当两个并发线程访问同一个对象object中的这个synchronized(this)同步代码块时,一个时间内只能有一个线程得到执行。另一个线程必须等待当前线程执行完这个代码块以后才能执行该代码块。

  1. package ths;
  2. public class Thread1 implements Runnable {
  3. public void run() {
  4. synchronized (this) {
  5. for (int i = 0; i < 5; i++) {
  6. System.out.println(Thread.currentThread().getName()
  7. + " synchronized loop " + i);
  8. }
  9. }
  10. }
  11. public static void main(String[] args) {
  12. Thread1 t1 = new Thread1();
  13. Thread ta = new Thread(t1, "A");
  14. Thread tb = new Thread(t1, "B");
  15. ta.start();
  16. tb.start();
  17. }
  18. }
package ths;

public class Thread1 implements Runnable {
public void run() {
synchronized (this) {
for (int i = 0; i < 5; i++) {
System.out.println(Thread.currentThread().getName()
+ " synchronized loop " + i);
}
}
} public static void main(String[] args) {
Thread1 t1 = new Thread1();
Thread ta = new Thread(t1, "A");
Thread tb = new Thread(t1, "B");
ta.start();
tb.start();
}
}

结果:
A synchronized loop 0
A synchronized loop 1
A synchronized loop 2
A synchronized loop 3
A synchronized loop 4
B synchronized loop 0
B synchronized loop 1
B synchronized loop 2
B synchronized loop 3
B synchronized loop 4

二、然而,当一个线程访问object的一个synchronized(this)同步代码块时,另一个线程仍然可以访问该object中的非synchronized(this)同步代码块。

  1. package ths;
  2. public class Thread2 {
  3. public void m4t1() {
  4. synchronized (this) {
  5. int i = 5;
  6. while (i-- > 0) {
  7. System.out
  8. .println(Thread.currentThread().getName() + " : " + i);
  9. try {
  10. Thread.sleep(500);
  11. } catch (InterruptedException ie) {
  12. }
  13. }
  14. }
  15. }
  16. public void m4t2() {
  17. int i = 5;
  18. while (i-- > 0) {
  19. System.out.println(Thread.currentThread().getName() + " : " + i);
  20. try {
  21. Thread.sleep(500);
  22. } catch (InterruptedException ie) {
  23. }
  24. }
  25. }
  26. public static void main(String[] args) {
  27. final Thread2 myt2 = new Thread2();
  28. Thread t1 = new Thread(new Runnable() {
  29. public void run() {
  30. myt2.m4t1();
  31. }
  32. }, "t1");
  33. Thread t2 = new Thread(new Runnable() {
  34. public void run() {
  35. myt2.m4t2();
  36. }
  37. }, "t2");
  38. t1.start();
  39. t2.start();
  40. }
  41. }
package ths;

public class Thread2 {
public void m4t1() {
synchronized (this) {
int i = 5;
while (i-- > 0) {
System.out
.println(Thread.currentThread().getName() + " : " + i);
try {
Thread.sleep(500);
} catch (InterruptedException ie) {
}
}
}
} public void m4t2() {
int i = 5;
while (i-- > 0) {
System.out.println(Thread.currentThread().getName() + " : " + i);
try {
Thread.sleep(500);
} catch (InterruptedException ie) {
}
}
} public static void main(String[] args) {
final Thread2 myt2 = new Thread2();
Thread t1 = new Thread(new Runnable() {
public void run() {
myt2.m4t1();
}
}, "t1");
Thread t2 = new Thread(new Runnable() {
public void run() {
myt2.m4t2();
}
}, "t2");
t1.start();
t2.start();
}
}

结果:
t1 : 4
t2 : 4
t1 : 3
t2 : 3
t1 : 2
t2 : 2
t1 : 1
t2 : 1
t1 : 0
t2 : 0

三、尤其关键的是,当一个线程访问object的一个synchronized(this)同步代码块时,其他线程对object中所有其它synchronized(this)同步代码块的访问将被阻塞。

//修改Thread2.m4t2()方法:

  1. public void m4t2() {
  2. synchronized (this) {
  3. int i = 5;
  4. while (i-- > 0) {
  5. System.out
  6. .println(Thread.currentThread().getName() + " : " + i);
  7. try {
  8. Thread.sleep(500);
  9. } catch (InterruptedException ie) {
  10. }
  11. }
  12. }
  13. }
	public void m4t2() {
synchronized (this) {
int i = 5;
while (i-- > 0) {
System.out
.println(Thread.currentThread().getName() + " : " + i);
try {
Thread.sleep(500);
} catch (InterruptedException ie) {
}
}
} }

结果:
t1 : 4
t1 : 3
t1 : 2
t1 : 1
t1 : 0
t2 : 4
t2 : 3
t2 : 2
t2 : 1
t2 : 0

四、第三个例子同样适用其它同步代码块。也就是说,当一个线程访问object的一个synchronized(this)同步代码块时,它就获得了这个object的对象锁。结果,其它线程对该object对象所有同步代码部分的访问都被暂时阻塞。

//修改Thread2.m4t2()方法如下:

  1. public synchronized void m4t2() {
  2. int i = 5;
  3. while (i-- > 0) {
  4. System.out.println(Thread.currentThread().getName() + " : " + i);
  5. try {
  6. Thread.sleep(500);
  7. } catch (InterruptedException ie) {
  8. }
  9. }
  10. }
	public synchronized void m4t2() {
int i = 5;
while (i-- > 0) {
System.out.println(Thread.currentThread().getName() + " : " + i);
try {
Thread.sleep(500);
} catch (InterruptedException ie) {
}
}
}

结果:
t1 : 4
t1 : 3
t1 : 2
t1 : 1
t1 : 0
t2 : 4
t2 : 3
t2 : 2
t2 : 1
t2 : 0

五、以上规则对其它对象锁同样适用:

  1. package ths;
  2. public class Thread3 {
  3. class Inner {
  4. private void m4t1() {
  5. int i = 5;
  6. while (i-- > 0) {
  7. System.out.println(Thread.currentThread().getName()
  8. + " : Inner.m4t1()=" + i);
  9. try {
  10. Thread.sleep(500);
  11. } catch (InterruptedException ie) {
  12. }
  13. }
  14. }
  15. private void m4t2() {
  16. int i = 5;
  17. while (i-- > 0) {
  18. System.out.println(Thread.currentThread().getName()
  19. + " : Inner.m4t2()=" + i);
  20. try {
  21. Thread.sleep(500);
  22. } catch (InterruptedException ie) {
  23. }
  24. }
  25. }
  26. }
  27. private void m4t1(Inner inner) {
  28. synchronized (inner) { // 使用对象锁
  29. inner.m4t1();
  30. }
  31. }
  32. private void m4t2(Inner inner) {
  33. inner.m4t2();
  34. }
  35. public static void main(String[] args) {
  36. final Thread3 myt3 = new Thread3();
  37. final Inner inner = myt3.new Inner();
  38. Thread t1 = new Thread(new Runnable() {
  39. public void run() {
  40. myt3.m4t1(inner);
  41. }
  42. }, "t1");
  43. Thread t2 = new Thread(new Runnable() {
  44. public void run() {
  45. myt3.m4t2(inner);
  46. }
  47. }, "t2");
  48. t1.start();
  49. t2.start();
  50. }
  51. }
package ths;

public class Thread3 {
class Inner {
private void m4t1() {
int i = 5;
while (i-- > 0) {
System.out.println(Thread.currentThread().getName()
+ " : Inner.m4t1()=" + i);
try {
Thread.sleep(500);
} catch (InterruptedException ie) {
}
}
} private void m4t2() {
int i = 5;
while (i-- > 0) {
System.out.println(Thread.currentThread().getName()
+ " : Inner.m4t2()=" + i);
try {
Thread.sleep(500);
} catch (InterruptedException ie) {
}
}
}
} private void m4t1(Inner inner) {
synchronized (inner) { // 使用对象锁
inner.m4t1();
}
} private void m4t2(Inner inner) {
inner.m4t2();
} public static void main(String[] args) {
final Thread3 myt3 = new Thread3();
final Inner inner = myt3.new Inner();
Thread t1 = new Thread(new Runnable() {
public void run() {
myt3.m4t1(inner);
}
}, "t1");
Thread t2 = new Thread(new Runnable() {
public void run() {
myt3.m4t2(inner);
}
}, "t2");
t1.start();
t2.start();
}
}

结果:

尽管线程t1获得了对Inner的对象锁,但由于线程t2访问的是同一个Inner中的非同步部分。所以两个线程互不干扰。
t1 : Inner.m4t1()=4
t2 : Inner.m4t2()=4
t1 : Inner.m4t1()=3
t2 : Inner.m4t2()=3
t1 : Inner.m4t1()=2
t2 : Inner.m4t2()=2
t1 : Inner.m4t1()=1
t2 : Inner.m4t2()=1
t1 : Inner.m4t1()=0
t2 : Inner.m4t2()=0

现在在Inner.m4t2()前面加上synchronized:

  1. private synchronized void m4t2() {
  2. int i = 5;
  3. while (i-- > 0) {
  4. System.out.println(Thread.currentThread().getName()
  5. + " : Inner.m4t2()=" + i);
  6. try {
  7. Thread.sleep(500);
  8. } catch (InterruptedException ie) {
  9. }
  10. }
  11. }
	private synchronized void m4t2() {
int i = 5;
while (i-- > 0) {
System.out.println(Thread.currentThread().getName()
+ " : Inner.m4t2()=" + i);
try {
Thread.sleep(500);
} catch (InterruptedException ie) {
}
}
}

结果:

尽管线程t1与t2访问了同一个Inner对象中两个毫不相关的部分,但因为t1先获得了对Inner的对象锁,所以t2对Inner.m4t2()的访问也被阻塞,因为m4t2()是Inner中的一个同步方

最新文章

  1. PHP基础知识之常量
  2. pthread_attr_setdetachstate
  3. wordpress换域名后无法登陆的解决方案
  4. maven之ssh项目搭建
  5. HDU 4633 Who&#39;s Aunt Zhang (Polya定理+快速幂)
  6. 我的Python成长之路---第六天---Python基础(20)---2016年2月20日(晴)
  7. java 之 Spring
  8. UVA12186
  9. LVS,Keepalived,HAproxy区别与联系
  10. Yaf 完全精通
  11. android-audioRecord
  12. Splay详解
  13. JavaServlet的文件上传和下载
  14. ubuntu 间简单相互通信
  15. as3.0用了视频组件,导致视频打开后就全屏,加一下代码就行
  16. vue ui之 iview 事件拦截
  17. mysql 添加用户
  18. centos7 安装 openvswitch
  19. (转载)Android学习笔记⑨——android.content.ActivityNotFoundException异常处理
  20. arangodb 安装简单试用

热门文章

  1. 【算法笔记】B1054 求平均值
  2. 洛谷 P3275 [SCOI2011]糖果
  3. Android 调整图标和字体大小
  4. nginx 反向代理导致的session丢失的问题
  5. layui 数据返回但是table表格未渲染出来的问题
  6. spark第三篇:Cluster Mode Overview 集群模式预览
  7. 用poi替换ppt中的文字和图片
  8. oracle 基础知识(九)----SQL解析
  9. linux运维基础之跟我一起学正则表达式(一)
  10. linux 运维基础测验题