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

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

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

package cn.king;{(String[] args) {new TraditionalThreadSynchronized().foo();}() {// printer须是final的,否则无法编译。这主要是为了保证printer的一致性。final Printer printer = new Printer();new Thread(new Runnable() {() {while(true) {try {Thread.sleep(10);} catch (InterruptedException e) {e.printStackTrace();}/** Cannot refer to a non-final variable printer* inside an inner class defined in a different method* 更多内容可参阅java8 lambda表达式(闭包)相关知识*/printer.output(“123456789”);}}}).start();new Thread(new Runnable() {() {while(true) {try {Thread.sleep(10);} catch (InterruptedException e) {e.printStackTrace();}printer.output(“abcdefghi”);}}}).start();}static class Printer {String _lock = “”;(String name) {int len = name.length();// 同步代码块/* 方法1:* 以this作为锁对象,* 与使用this加锁的代码块或synchronized方法互斥*/// synchronized(this) {/* 方法2:* 以Outputer类的字节码对象(该对象由虚拟机自动创建)作为锁对象,* 与使用Outputer.class加锁的代码块或static synchronized方法互斥*/// synchronized(Outputer.class) {/* 方法3:* 以自定义对象作为锁对象,* 与使用_lock加锁的代码块互斥*/synchronized(_lock) {for(int i=0; i<len; i++) {System.out.print(name.charAt(i));}System.out.println();}}(String name) {int len = name.length();for(int i=0; i<len; i++) {System.out.print(name.charAt(i));}System.out.println();}(String name) {int len = name.length();for(int i=0; i<len; i++) {System.out.print(name.charAt(i));}System.out.println();}}}

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

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

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

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

人创造奇迹常常是在瞬间,

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

相关文章:

你感兴趣的文章:

标签云: