synchronized关键字的语义
上一篇文章,我们讲到,如果发生了多个线程共同访问一个全局变量的时候,就会发生各种意料之外的情况。其实现实生活中有很多这样的例子。我举一个例子。
一群人都要过河,但是河面上只有一只独木舟,除了船夫,一次只能带一个人。所有到达河边的人都想往船上抢,难免把船搞翻了。为了解决这个问题,我们可以在河边上设一个售票处,谁先抢到票,谁就可以上船,没有抢到票的,就只能等待下一次,船返回来,再去抢下一张票。
好了,在多线程编程中,我们也可以引入这样一个售票处,让线程先去抢票,抢到票的,就可以使用这只小船,抢不到的,就继续等待。这个售票处,就是 synchronized 了。
synchronized 方法
当一个方法加上synchronized 关键字以后,就只能让一个线程来执行这个方法了。只让一个线程的意思并不是只把这个方法指定给某个固定的线程,而是说一次只能有一个线程来调用这个函数。
我们把昨天的那个程序改一下,就很清楚了:
public class TestTwo {
public int total = 0;
public synchronized void incTotal() {
total += 1;
}
public static void main(String[] args) throws Exception{
TestTwo test = new TestTwo();
Thread t1 = new Thread() {
public void run() {
for (int i = 0; i < 5_000; i++) {
test.incTotal();
}
}
};
Thread t2 = new Thread() {
public void run() {
for (int i = 0; i < 5_000; i++) {
test.incTotal();
}
}
};
t1.start();
t2.start();
t1.join();
t2.join();
System.out.println(test.total);
}
}
现在不论你执行多少次,最后的结果一定会是10000了,这是因为我们把加1的操作用synchronized 保护起来了。一旦一个线程进入到了 incTotal 以后,其他的线程就不能再进入了。这样,我们就可以保证这个加法是完整而且独立的,其他的线程完全不能打扰到它了。
synchronized block
有时候,我们使用synchronized 修饰一个方法,会显得太重了一些。往往需要使用这种互斥进行保护的只是几个语句,而不是一个方法。那我们还可以使用语句块,它的语法是这样的:
synchronized(object) {
// do something
}
例如,上面的例子,我们还可以这样写:
public class TestThree {
public int total = 0;
public static void main(String[] args) throws Exception{
TestThree test = new TestThree();
Thread t1 = new Thread() {
public void run() {
for (int i = 0; i < 5_000; i++) {
synchronized (test) {
test.total += 1;
}
}
}
};
Thread t2 = new Thread() {
public void run() {
for (int i = 0; i < 5_000; i++) {
synchronized (test) {
test.total += 1;
}
}
}
};
t1.start();
t2.start();
t1.join();
t2.join();
System.out.println(test.total);
}
}
看,我们在这里就不用再定义一个synchronized方法了。而是在一个对象上加上这个互斥就可以了。
实际上,这里都不定要使用 test 这个对象。例如,我可以用一个完全不相干的对象来做互斥:
Object o = new Object();
synchronized(o) {
do_something...
}
只要这个变量是两个线程都能访问就可以了。
- 部署Microsoft .NET Framework 3.0[翻译]
- 性能优化总结(六):预加载、聚合SQL应用实例
- 性能优化总结(五):CSLA服务端如何使用多线程的解决方案
- 编辑-发布-开发分离:git作为NoSQL数据库
- 性能优化总结(四):预加载的设计
- 在Expression Blend中使用XAML建立3D应用程序
- 使用 Asp.net Future May 2007 开发Silverlight应用
- Rafy 领域实体框架设计 - 重构 ORM 中的 Sql 生成
- RePractise前端篇: 前端演进史
- 性能优化总结(三):聚合SQL在GIX4中的应用
- ASP.NET AJAX 控件开发基础
- 听我说说我的博客: 月访问量过万的个人IT博客的技术史
- TransactionScope和Enterprise Libray 3.0 Data Access Application Block
- 《Python Web开发 - 测试驱动方法》阅后感
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- 渗透测试之API测试技巧
- PC(C/S架构)客户端测试笔记
- 手把手教你使用Python开发飞机大战小游戏,4万字超详细讲解!
- Android通过URL文件下载及文件名提取
- PyTorch4:模块总览&torch.utils.data
- express处理文件上传
- 《机器学习》-- 第七章 朴素贝叶斯
- 【DIY数字仪表】RT-Thread移植touchgfx使用sd卡升级固件和图片资源(4)
- Transformers中的Beam Search高效实现
- 多边形裁剪图片升级啦!Cocos Creator !
- 使用curl进行服务器测试
- 打卡群刷题总结0805——不同的二叉搜索树
- 利用STS临时密钥服务快速搭建直传页面的实践
- Clickhouse在大数据分析平台-留存分析上的应用
- Rancher 高可用部署