synchronized -- 同步加锁的是对象,而不是代码
2015-07-09 18:00
120 查看
很长时间都不理解“同步加锁的是对象,而不是代码”这句话,昨天在看TimerTask源码时候发现源码中TimerTask就是一个典型的最优同步方法实现的类,又结合网上其他文章做了一些研究,总结以下自己的认识。
先说结论:“同步加锁的是对象”是指同步加锁的是同步代码或同步代码块所在的类的实例对象或者是一个指定的其他实例对象,而不是加锁的代码或者类本身。下面继续分析,通过同步的几种实现方法以及每种实现的特性来了解得出这个结论的细节。
同步分两种,即同步方法和同步代码块:
1.同步代码块
由于同一个对象中的所有同步方法同一时间只能有其中一个可以在一个线程中执行,导致同步会影响程序运行效率,所以我们同步所包括的代码内容越少越少,而同步代码块正是可以解决这个问题。下面是TimerTask源码中一个同步代码块的实现:
public abstract class TimerTask implements Runnable {
/* Lock object for synchronization. It's also used by Timer class. */
final Object lock = new Object();
......
/** * Cancels the {@code TimerTask} and removes it from the {@code Timer}'s queue. Generally, it * returns {@code false} if the call did not prevent a {@code TimerTask} from running at * least once. Subsequent calls have no effect. *
* @return {@code true} if the call prevented a scheduled execution * from taking place, {@code false} otherwise. */
public boolean cancel() {
synchronized (lock) {
boolean willRun = !cancelled && when > 0;
cancelled = true;
return willRun;
}
}
.....
}}
同步代码块的实现特点是它只对同步逻辑中的关键运算部分代码进行同步,而不是整个功能方法。
同步代码块又根据加锁对象的不同写法分为三种:
1.1:synchronized (this)(...);
这种方式的加锁对象就是同步代码块所在的类的实例对象。
1.2:对指定的对象加锁,正如上面TimerTask中截取的那段源码
synchronized(obj);
这里的加锁对象就是这个指定的实例对象。
1.3:对当前类的Class的加锁,由于每个类都是Class的一个实例,所以可以对当前类的Class加锁,如下:
Class Foo { public void methodBBB() { synchronized(Foo.class) // class literal(类名称字面常量) } }}} 这里的Foo.class是获得Foo.java这个类的字节码对象即Class,是不同与对Foo.java类的实例对象的。这种写法相当于1.2中的效果,只不过是找了一个特殊的加锁对象而已。
2.同步方法:就是用synchronized关键字直接修饰一个成员方法。
2.1普通成员方法的加锁
public synchronized void methodAAA() { //…. }
2.2静态成员方法的加锁
public synchronized static void methodAAA() // 同步的static 函数 { //…. } } 这种的加锁对象和1.3中一样
先说结论:“同步加锁的是对象”是指同步加锁的是同步代码或同步代码块所在的类的实例对象或者是一个指定的其他实例对象,而不是加锁的代码或者类本身。下面继续分析,通过同步的几种实现方法以及每种实现的特性来了解得出这个结论的细节。
同步分两种,即同步方法和同步代码块:
1.同步代码块
由于同一个对象中的所有同步方法同一时间只能有其中一个可以在一个线程中执行,导致同步会影响程序运行效率,所以我们同步所包括的代码内容越少越少,而同步代码块正是可以解决这个问题。下面是TimerTask源码中一个同步代码块的实现:
public abstract class TimerTask implements Runnable {
/* Lock object for synchronization. It's also used by Timer class. */
final Object lock = new Object();
......
/** * Cancels the {@code TimerTask} and removes it from the {@code Timer}'s queue. Generally, it * returns {@code false} if the call did not prevent a {@code TimerTask} from running at * least once. Subsequent calls have no effect. *
* @return {@code true} if the call prevented a scheduled execution * from taking place, {@code false} otherwise. */
public boolean cancel() {
synchronized (lock) {
boolean willRun = !cancelled && when > 0;
cancelled = true;
return willRun;
}
}
.....
}}
同步代码块的实现特点是它只对同步逻辑中的关键运算部分代码进行同步,而不是整个功能方法。
同步代码块又根据加锁对象的不同写法分为三种:
1.1:synchronized (this)(...);
这种方式的加锁对象就是同步代码块所在的类的实例对象。
1.2:对指定的对象加锁,正如上面TimerTask中截取的那段源码
synchronized(obj);
这里的加锁对象就是这个指定的实例对象。
1.3:对当前类的Class的加锁,由于每个类都是Class的一个实例,所以可以对当前类的Class加锁,如下:
Class Foo { public void methodBBB() { synchronized(Foo.class) // class literal(类名称字面常量) } }}} 这里的Foo.class是获得Foo.java这个类的字节码对象即Class,是不同与对Foo.java类的实例对象的。这种写法相当于1.2中的效果,只不过是找了一个特殊的加锁对象而已。
2.同步方法:就是用synchronized关键字直接修饰一个成员方法。
2.1普通成员方法的加锁
public synchronized void methodAAA() { //…. }
2.2静态成员方法的加锁
public synchronized static void methodAAA() // 同步的static 函数 { //…. } } 这种的加锁对象和1.3中一样
相关文章推荐
- Spark问题
- 关于 AnnotationConfigBeanDefinitionParser are only available on JDK 1.5 and higher
- spring中classpath和classpath*的配置区别
- Delphi XE5 常见错误与解决
- SPOJ QTREE5 lct
- Spring的Maven项目错误
- php基于Snoopy解析网页html的方法
- Lua与c/c++交互
- 同一台机器安装多个python版本
- 《编程导论(Java)》格言录
- 优化JavaScripe 提升首页加载速度的几种方案解析
- php生成固定长度纯数字编码的方法
- 刚学的php分页笔记 未使用MVC分层模式
- Java基础之Format
- MyBatis调用存储过程,含有返回结果集、return参数和output参数
- 图书馆管理系统(C++课程设计)
- VisualStudio2010中创建ASP.Net WebService
- eclipse中hibernate配置*.hbm.xml文件和hibernate.cfg.xml文件使其能自动提示
- java语法基础(总结)
- delphi 调用 webservice (.NET C#版)