Java代码质量改进之:同步对象的选择
在Java中,让线程同步的一种方式是使用synchronized关键字,它可以被用来修饰一段代码块,如下:
synchronized(被锁的同步对象) {
// 代码块:业务代码
}
当synchronized被用来修饰代码块的时候表示,如果有多个线程正在执行这段代码块,那么需要等到其中一个线程执行完毕,第二个线程才会再执行它。但是!如果被锁的同步对象没有被正确选择的话,上面的结论是不正确的哦。
到底什么样的对象能够成为一个锁对象(也叫同步对象)?我们在选择同步对象的时候,应当始终注意以下几点:
第一点,需要锁定的对象在多个线程中是可见的、同一个对象
“可见的”这是显而易见的,如果对象不可见,就不能被锁定。“同一个对象”,这理解起来也很好理解,如果锁定的不是同一个对象,那又如何来同步两个对象呢?可是,不见得我们在这上面不会犯错误。为了阐述本建议,我们先模拟一个必须使用到锁的场景:火车站卖火车票。一列火车一共有100张票,一共有3个窗口在同时卖票,代码如下:
package com.zuikc.thread;
public class SynchronizedSample01 {
public static void main(String[] args) {
// 创建
TicketWindow window1 = new TicketWindow("售票窗口1");
TicketWindow window2 = new TicketWindow("售票窗口2");
TicketWindow window3 = new TicketWindow("售票窗口3");
window1.start();
window2.start();
window3.start();
}
}
class TicketWindow extends Thread {
// 共100个座位
static int ticket = 100;
public TicketWindow(String name) {
super(name);
}
@Override
public void run() {
// 模拟卖票
while (ticket > 0) {
System.out.println(this.getName() + "卖出了座位号:" + ticket);
ticket--;
try {
Thread.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
可是,运行之后,我们发现我们的火车票有些座位被卖了多次,比如:

只要多运行几次,我们就会看到不同的结果。但是几乎每次都会有被座位号被卖多次的现象发生。
有同学可能会说,简单:加synchronized锁定同步对象,于是我们修改代码:
class TicketWindow extends Thread {
// 共100个座位
static int ticket = 100;
// 定义被锁的同步对象
Object obj = new Object();
public TicketWindow(String name) {
super(name);
}
@Override
public void run() {
// 想要同步的代码块
synchronized (obj) {
// 模拟卖票
while (ticket > 0) {
System.out.println(this.getName() + "卖出了座位号:" + ticket);
ticket--;
try {
Thread.sleep(10);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}
运行之后,我们发现结果没有任何的改变。为什么呐?
因为在3个线程中,我们锁定的不是同一个对象。
我们看到,被锁的是一个实例变量,如下:
Object obj = new Object();
而存在三个线程,就意味着生成了3个obj,每个线程锁定的是这3个不同的obj对象,所以,同步代码块等于没有被同步。
那应该怎么做呢?最简单的方法是,我们可以把实例变量改成成员变量,即静态变量,如下:
Static Object obj = new Object();
然后,再运行售票代码,就发现可以解决这个问题了。不信,试试看。
第二个注意事项:非静态方法中,静态变量不应作为同步对象
上面刚说完,要修正第一点中的示例,需要将obj变成static。这似乎和本注意事项有矛盾。实际上,第一点中的示例代码仅出于演示的目的,在编写多线程代码时,我们可以遵循这样的一个原则:类型的静态方法应当保证线程安全,非静态方法不需实现线程安全。而如果将syncObject变成static,就相当于让非静态方法具备线程安全性,这带来的一个问题是,如果应用程序中该类型存在多个实例,在遇到这个锁的时候,都会产生同步,而这可能不是我们原先所愿意看到的。
第三点:值类型(基本数据类型)对象不能作为同步对象
实际上,这样的代码也不会通过编译。
值类型在传递另一个线程的时候,会创建一个副本,这相当于每个线程锁定的也是两个对象。故,值类型对象不能作为同步对象。这一点实际也可以归结到第一点中。
第四点,锁定字符串是完全没有必要,而且相当危险的
这整个过程看上去和值类型正好相反。字符串在虚拟机中会被暂存到内存里,如果有两个变量被分配了相同内容的字符串,那么这两个引用会被指向同一块内存。所以,如果有两个地方同时使用了synchronized (“abc”),那么它们实际锁定的是同一个对象,导致整个应用程序被阻滞。
第五点:降低同步对象的可见性
同步对象一般来说,不应该是一个public变量,我们应该始终考虑降低同步对象的可见性,将我们的同步对象藏起来,只开放给自己或自己的子类就够了(需要开放给子类的情况其实也不多见)。
以下是广告时间:最课程(zuikc.com)正在招收Java就业班学员,如果你想学习更多的Java高质量代码编写方面的技巧,请联系我们哦。
Java代码质量改进之:同步对象的选择的更多相关文章
- Java代码质量改进之:使用ThreadLocal维护线程内部变量
在上文中,<Java代码质量改进之:同步对象的选择>,我们提出了一个场景:火车站有3个售票窗口,同时在售一趟列车的100个座位.我们通过锁定一个靠谱的同步对象,完成了上面的功能. 现在,让 ...
- JSP中的Java代码和内置对象
一.JSP中的Java代码 (一)JSP页面中有三种方式嵌入java代码: 1.java的表达式 格式:<%= java表达式 %> 2.java的语句 格式:<% java语句&g ...
- 关于Java多线程的线程同步和线程通信的一些小问题(顺便分享几篇高质量的博文)
Java多线程的线程同步和线程通信的一些小问题(顺便分享几篇质量高的博文) 前言:在学习多线程时,遇到了一些问题,这里我将这些问题都分享出来,同时也分享了几篇其他博客主的博客,并且将我个人的理解也分享 ...
- Android学习笔记_32_通过WebView实现JS代码与Java代码互相通信
webview两种实现方法,覆盖onKeyDown()方法 缓存 WebSettings应用注意的几个问题 1.要实现JS代码与Java代码互相通信,需要通过Android的WebView控件,在视图 ...
- JSP页面java代码报错:Purgoods cannot be resolved to a type
错误提示 : Purgoods cannot be resolved to a type Purgoods不能解析为一个类型 原因 : 缺少引入Purgoods类 页面中引入java类,执行java代 ...
- 0037 Java学习笔记-多线程-同步代码块、同步方法、同步锁
什么是同步 在上一篇0036 Java学习笔记-多线程-创建线程的三种方式示例代码中,实现Runnable创建多条线程,输出中的结果中会有错误,比如一张票卖了两次,有的票没卖的情况,因为线程对象被多条 ...
- 【转】java代码中实现android背景选择的selector-StateListDrawable的应用
原文网址:http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2014/0924/1712.html 下面的代码应该很多人都熟悉: 1 2 3 ...
- Oracle03——游标、异常、存储过程、存储函数、触发器和Java代码访问Oracle对象
作者: kent鹏 转载请注明出处: http://www.cnblogs.com/xieyupeng/p/7476717.html 1.游标(光标)Cursor 在写java程序中有集合的概念,那么 ...
- JAVA之旅(十三)——线程的安全性,synchronized关键字,多线程同步代码块,同步函数,同步函数的锁是this
JAVA之旅(十三)--线程的安全性,synchronized关键字,多线程同步代码块,同步函数,同步函数的锁是this 我们继续上个篇幅接着讲线程的知识点 一.线程的安全性 当我们开启四个窗口(线程 ...
随机推荐
- 加入ffmpeg播放视屏
下面的字反了..,另外没声音 2018-4-28 前段时间已经做的差不多了,音频的pack取出来用openAL播放,并实现了视屏同步播放,并且支持unity 现在的问题就是支持大分辨率视屏播放的问题, ...
- java interface接口的传值方法
A 类 package interface_test; public class A { private IPresenter ip; public A(IPresenter ip) { this.i ...
- Idea2018旗舰版破解方法
完整请参考 https://www.jianshu.com/p/3c87487e7121 1.在hosts文件里添加一行: 0.0.0.0 account.jetbrains.com 2.在Activ ...
- 腾讯云云机安装dockers
云机的配置 首先更新一下源(更新前一直装不了) 下载dockers-ce(社区版) 启动dockers服务 使用hello-world进行测试(由于本地没有hello-world这个镜像,所以dock ...
- Nest.js 守卫
Docs: https://docs.nestjs.com/guards 当调用者具有足够的权限时,特定路由才可用 // app.guard.ts import { CanActivate, Exec ...
- pdf转html插件~~~pdf2htmlEX安装,配置及使用
这是一个将pdf转化为html的服务,开源的. 此功能服务的代码在git上的地址为: https://github.com/coolwanglu/pdf2htmlEX/wiki 安装: 在ubuntu ...
- STL next_permutation 算法原理和自行实现
目标 STL中的next_permutation 函数和 prev_permutation 两个函数提供了对于一个特定排列P,求出其后一个排列P+1和前一个排列P-1的功能. 这里我们以next_pe ...
- windows服务安装 System.IO.FileLoadException
报错如下: System.IO.FileLoadException: 未能加载文件或程序集“file:///D:\WindowsService\bin\Debug\WindowsService.exe ...
- vue路由动态加载
注意:是动态加载不是动态路由 解决的问题: 动态配置菜单栏的路由参数--实现菜单级的权限控制 问题成因: 在vue实例化的时候vuex.vue-router 就需要加载完毕,无法使用异步的方式从服务器 ...
- Prometheus监控学习记录
官方文档 Prometheus基础文档 从零开始:Prometheus 进阶之路:Prometheus —— 技巧篇 进阶之路:Prometheus —— 理解篇 prometheus的数据类型介绍 ...