在路上

 找回密码
 立即注册
在路上 站点首页 学习 查看内容

Java多线程基础(三)Java传统线程互斥技术

2017-2-7 13:40| 发布者: zhangjf| 查看: 440| 评论: 0

摘要: Java多线程基础(三)Java传统线程互斥技术 java的线程互斥主要通过synchronized关键字实现。下面的示例代码展示了几种使用synchronized关键字的基本用法。 package cn.king;public class TraditionalThreadSyn ...
Java多线程基础(三)Java传统线程互斥技术

java的线程互斥主要通过synchronized关键字实现。下面的示例代码展示了几种使用synchronized关键字的基本用法。

  1. package cn.king;
  2. public class TraditionalThreadSynchronized {
  3. public static void main(String[] args) {
  4. new TraditionalThreadSynchronized().foo();
  5. }
  6. private void foo() {
  7. // printer须是final的,否则无法编译。这主要是为了保证printer的一致性。
  8. final Printer printer = new Printer();
  9. new Thread(new Runnable() {
  10. @Override
  11. public void run() {
  12. while(true) {
  13. try {
  14. Thread.sleep(10);
  15. } catch (InterruptedException e) {
  16. e.printStackTrace();
  17. }
  18. /* * Cannot refer to a non-final variable printer * inside an inner class defined in a different method * 更多内容可参阅java8 lambda表达式(闭包)相关知识 */
  19. printer.output("123456789");
  20. }
  21. }
  22. }).start();
  23. new Thread(new Runnable() {
  24. @Override
  25. public void run() {
  26. while(true) {
  27. try {
  28. Thread.sleep(10);
  29. } catch (InterruptedException e) {
  30. e.printStackTrace();
  31. }
  32. printer.output("abcdefghi");
  33. }
  34. }
  35. }).start();
  36. }
  37. static class Printer {
  38. String _lock = "";
  39. public void output(String name) {
  40. int len = name.length();
  41. // 同步代码块
  42. /* 方法1: * 以this作为锁对象, * 与使用this加锁的代码块或synchronized方法互斥 */
  43. // synchronized(this) {
  44. /* 方法2: * 以Outputer类的字节码对象(该对象由虚拟机自动创建)作为锁对象, * 与使用Outputer.class加锁的代码块或static synchronized方法互斥 */
  45. // synchronized(Outputer.class) {
  46. /* 方法3: * 以自定义对象作为锁对象, * 与使用_lock加锁的代码块互斥 */
  47. synchronized(_lock) {
  48. for(int i=0; i<len; i++) {
  49. System.out.print(name.charAt(i));
  50. }
  51. System.out.println();
  52. }
  53. }
  54. // 同步方法,相当于synchronized(this){}
  55. public synchronized void output2(String name) {
  56. int len = name.length();
  57. for(int i=0; i<len; i++) {
  58. System.out.print(name.charAt(i));
  59. }
  60. System.out.println();
  61. }
  62. // 静态同步方法,相当于synchronized(Outputer.class){}
  63. public static synchronized void output3(String name) {
  64. int len = name.length();
  65. for(int i=0; i<len; i++) {
  66. System.out.print(name.charAt(i));
  67. }
  68. System.out.println();
  69. }
  70. }
  71. }
复制代码

上面的代码中展示了三种基本的线程互斥实现。下面详述三种方法的实现特点和适用情况。

以this作为锁对象
实现方式
synchronized(this){…}
synchronized实例方法
适用情况
适用于使用类的同一个实例(对象)作为锁对象。下面情况不适用:
若上述代码中第26行和第39行,改为
new Printer().output(“<相应字符串>”);
则无法使用本方法实现线程互斥,而须采用第2种方法。

以Outputer.class作为锁对象
Outputer类的字节码对象由jvm自动加载。
实现方式
synchronized(Outputer.class){…}
static synchronized方法
适用情况
适用于整个类的所有对象都需要互斥访问的情况。

以自定义的对象作为所对象
实现方式
synchronized(<自定义锁对象>){…}
适用情况
同一个类中代码块或者方法的互斥,一般可以用第1种和第2种方法替换。当出现需要在多个类(或者多个类的实例)之间进行互斥控制时,可能需要采用本方法。

来自: http://blog.csdn.net//kingzone_2008/article/details/48166731

最新评论

小黑屋|在路上 ( 蜀ICP备15035742号-1 

;

GMT+8, 2025-7-9 08:14

Copyright 2015-2025 djqfx

返回顶部