java多线程学习-同步(synchronized)
(示例都是网上视频的)
假如两个线程同时调用一个方法输出字符串
public class SynchronizedTest extends Thread {
public static void main(String[] args) {
final SynchronizedTest st = new SynchronizedTest();
new Thread(new Runnable() {
public void run() {
while(true){
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
st.outputStr("this is test");
}
}
}).start();
new Thread(new Runnable() {
public void run() {
while(true){
try {
Thread.sleep(10);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
st.outputStr("okokokokokok");
}
}
}).start();
}
public void outputStr(String str){
for(int i=0;i<str.length();i++){
System.out.print(str.charAt(i));
}
System.out.println();
}
}
输出:
this is test
okokokokokok
this is test
okokokokokok
this is test
okokothis is test
kokokok
tokokokokokok
his is test
okokokokokok
this is test
okokthisokokokok
is test
okothkoiskokokok
is test
okokokothis is test
kokok
this iokokokokokok
s test
okokokothiskokok
is test
okokokokokokthis
is test
okthisok is tokokokoest
可以看出来,一开始输出还没问题,后来输出就乱了。
解决方法就是加上同步方法即在outputStr方法上加上同步关键字synchronized即可
public synchronized void outputStr(String str){
for(int i=0;i<str.length();i++){
System.out.print(str.charAt(i));
}
System.out.println();
}
synchronized :
当然synchronized不仅可以用在方法上,还可以用在对象上。
1.用在String上
public class SynchronizedTest2 {
public static void main(String[] args) {
final SynchronizedTest2 t2 = new SynchronizedTest2();
new Thread(new Runnable() {
public void run() {
while(true){
try {
Thread.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
t2.outputStr("thread-1");
}
}
}).start();
new Thread(new Runnable() {
public void run() {
while(true){
try {
Thread.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
t2.outputStr("thread-2");
}
}
}).start();
}
public void outputStr(String threadName){
String str = "123";
synchronized(str){
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(threadName + " - "+System.currentTimeMillis()/1000);
}
}
}
输出:
thread-1 - 1475373023
thread-2 - 1475373024
thread-1 - 1475373025
thread-2 - 1475373026
thread-1 - 1475373027
可以看出,每一秒只有一个线程在运行,但是也不觉得是thread-1运行下,thread-2运行下,随机的。对字符串同步也可以,但是同步的必须是同一个对象才能互斥。
public void outputStr(String threadName){
String str = "123";
synchronized(threadName){
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(threadName + " - "+System.currentTimeMillis()/1000);
}
}
这次synchronized里面同步的是threadName参数了
输出:
thread-1 - 1475373333
thread-2 - 1475373333
thread-1 - 1475373334
thread-2 - 1475373334
thread-1 - 1475373335
thread-2 - 1475373335
thread-1 - 1475373336
thread-2 - 1475373336
可以看出同一秒中两个线程都运行了,这里面两个线程就没有互斥。
public void outputStr(String threadName){
String str = "123";
synchronized(this){
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(threadName + " - "+System.currentTimeMillis()/1000);
}
}
这时候调用同一SynchronizedTest2对象的线程在这个方法里都会互斥
但是如果把main方法改成这样
public class SynchronizedTest2 {
public static void main(String[] args) {
new Thread(new Runnable() {
public void run() {
SynchronizedTest2 t2 = new SynchronizedTest2();
while(true){
try {
Thread.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
t2.outputStr("thread-1");
}
}
}).start();
new Thread(new Runnable() {
public void run() {
SynchronizedTest2 t2 = new SynchronizedTest2();
while(true){
try {
Thread.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
t2.outputStr("thread-2");
}
}
}).start();
}
public void outputStr(String threadName){
String str = "123";
synchronized(this){
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(threadName + " - "+System.currentTimeMillis()/1000);
}
}
}
即每个线程都使用不同的SynchronizedTest2对象,这时输出:
thread-2 - 1475376542
thread-1 - 1475376542
thread-1 - 1475376543
thread-2 - 1475376543
thread-1 - 1475376544
thread-2 - 1475376544
thread-1 - 1475376545
thread-2 - 1475376545
可以看出此时两个线程并没有互斥。所以同步this只有使用同一个对象才会互斥。
3.用在类的字节码上
把outputStr方法改为
public void outputStr(String threadName){
String str = "123";
synchronized(SynchronizedTest2.class){
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(threadName + " - "+System.currentTimeMillis()/1000);
}
}
这时不管main方法里面的两个线程使用是同一个SynchronizedTest2对象还是不同的对象,在调用这个方法的时候,都能互斥,不会同时运行。可能是因为字节码对象只有一个吧。
其实将synchronized放在方法上,好像和这样同步一样。
synchronized关键字暂时只想到这么多,写的有点乱。
java多线程学习-同步(synchronized)的更多相关文章
- Java多线程学习之synchronized总结
0.概述 synchronized是Java提供的内置的锁机制,来实现代对码块的同步访问,称为内置锁(Intrinsic Lock) .内置锁包括两部分:一个是作为锁的对象的引用,另一个是由这个锁保护 ...
- java多线程学习-同步之线程通信
这个示例是网上烂大街的,子线程循环100次,主线程循环50次,但是我试了很多次,而且从网上找了很多示例,其实多运行几次,看输出结果并不正确.不知道是我转牛角尖了,还是怎么了.也没有大神问,好痛苦.现在 ...
- Java 多线程(Thread) 同步(synchronized) 以及 wait, notify 相关 [实例介绍]
场景描述 有一家很大的商场,在某市有几个商品配送中心,并有几家分店,这家商场经营很多的商品,平时运营情况是这样的: 根据各分店的商品销售情况,给分店配送相应需求量的商品:并上架到分店指让的位置,供客户 ...
- Java多线程学习(二)synchronized关键字(2)
转载请备注地址:https://blog.csdn.net/qq_34337272/article/details/79670775 系列文章传送门: Java多线程学习(一)Java多线程入门 Ja ...
- Java多线程学习(二)synchronized关键字(1)
转载请备注地址: https://blog.csdn.net/qq_34337272/article/details/79655194 Java多线程学习(二)将分为两篇文章介绍synchronize ...
- Java多线程学习(转载)
Java多线程学习(转载) 时间:2015-03-14 13:53:14 阅读:137413 评论:4 收藏:3 [点我收藏+] 转载 :http://blog ...
- java多线程学习笔记——详细
一.线程类 1.新建状态(New):新创建了一个线程对象. 2.就绪状态(Runnable):线程对象创建后,其他线程调用了该对象的start()方法.该状态的线程位于可运行线程池中, ...
- 【转】Java多线程学习
来源:http://www.cnblogs.com/samzeng/p/3546084.html Java多线程学习总结--线程概述及创建线程的方式(1) 在Java开发中,多线程是很常用的,用得好的 ...
- JAVA多线程学习笔记(1)
JAVA多线程学习笔记(1) 由于笔者使用markdown格式书写,后续copy到blog可能存在格式不美观的问题,本文的.mk文件已经上传到个人的github,会进行同步更新.github传送门 一 ...
随机推荐
- oracle远程连接太慢
场景:Oracle11G DB服务器上用sqlplus直接连很快(秒连),但加上服务名后却半天没反应.一台正式.一台测试都出现此问题. sqlplus:xnwz/xnwz 很快 sqlplus:xnw ...
- 使用新浪云(SAE)实现基于mySql和微信公众平台的关键字请求响应服务
本例是作者初次尝试微信公众平台开发之作,实现传统的关键字请求响应功能.即:用户发关键字,公众号通过关键字进行检索, 在mysql数据库中读取与关键字相关的信息,并返回给用户.本例在微信订阅号(开发者模 ...
- 切服务器时请注意robots.txt文件
最近两天切服务器时又出现测试机器忘记改robots.txt文件的情况,以后就用个二级域名做测试,测试的机器也不对robots.txt作限制.
- eclipse默认文件编码
eclipse里 就是在eclipse.ini文件里添加一行-Dfile.encoding=utf-8即可 -startup plugins/org.eclipse.equinox.launcher_ ...
- OFBIZ:启动之StartupLoader
任意一个JAVA程序都是从main()开始启动的,OFBIZ也不例外.OFBIZ的main()位于framework/start/src/org/ofbiz/base/start/Start.java ...
- 在excel 中某一单元格添加指定字符的示例
="select TestSurveyID,'http://www.findoout.cn/survey/viewsurvey.aspx?tid='+CONVERT(varchar(10), ...
- 翻译: TypeScript 1.8 Beta 发布
原文地址:https://blogs.msdn.microsoft.com/typescript/2016/01/28/announcing-typescript-1-8-beta/ 今天,我们发布了 ...
- Linux操作系统学习笔记
关于Linux详细介绍可以参见:http://blog.csdn.net/hguisu/article/details/6122513 关于Linux的常用命令可以参见:http://blog.csd ...
- mybaties中的selectKey和useGeneratedKeys=true
<!-- 账户创建 --><insert id="create" parameterType="Account"> <select ...
- a标签截字
首先要给A标签设置宽度,需要把A标签变为块级元素 display:block:或者 display:inline-block: 然后设置宽度,溢出隐藏,强制不换行这几个属性. a { display: ...