java多线程总结四:volatile、synchronized示例
1、synchronized保证同步
先看一个生成偶数的类
- <span style="font-size:16px;">package demo.thread;
- /**
- *这是一个int生成器的抽象类
- *
- */
- public abstract class IntGenerator {
- private volatile boolean canceled = false;
- public abstract int next();
- public void cancel() {
- canceled = true;
- }
- public boolean isCanceled() {
- return canceled;
- }
- }
- </span>
- <span style="font-size:16px;">/*
- * 产生偶数
- */
- class EvenGenerator extends IntGenerator {
- private int currentEvenValue = 0;
- String s = "";
- @Override
- public int next() {
- <span style="color:#ff0000;">synchronized </span>(s) {
- ++currentEvenValue;
- ++currentEvenValue;
- return currentEvenValue;
- }
- }
- // //这样也可以
- // public <span style="color:#ff0000;">synchronized </span>int next() {
- // ++currentEvenValue;
- // ++currentEvenValue;
- // return currentEvenValue;
- // }
- }</span>
注意到在产生偶数是要加同步锁,否则可能线程1刚好执行了一句++currentEvenValue;操作,就被线程2抢去了cpu,此时线程2执行return currentEvenValue;这时返回的就是一个奇数。加synchronized
就是两个线程同时只能一个线程执行synchronized 块的代码。
测试代码:
- <span style="font-size:16px;">package demo.thread;
- import java.util.concurrent.ExecutorService;
- import java.util.concurrent.Executors;
- /*
- * 消费数字
- */
- public class EvenChecker implements Runnable {
- private IntGenerator generator;
- private final int id;
- public EvenChecker(IntGenerator g, int ident) {
- generator = g;
- id = ident;
- }
- public void run() {
- while (!generator.isCanceled()) {
- int val = generator.next();
- if (val % 2 != 0) {//如果不是偶数
- System.out.println(val + " not enen!");
- generator.cancel();
- }
- }
- }
- public static void test(IntGenerator gp, int count) {
- ExecutorService exec = Executors.newCachedThreadPool();
- for (int i = 0; i < count; i++)
- exec.execute(new EvenChecker(gp, i));
- exec.shutdown();
- }
- public static void test(IntGenerator gp) {
- test(gp, 10);
- }
- public static void main(String[] args) {
- test(new EvenGenerator());
- }
- }</span>
分析:如果产生偶数的类未加synchronized,那么测试程序将会出现奇数导致退出程序。
2、volatile表示原子性,可见性。
对于多个线程之间共享的变量,每个线程都有自己的一份拷贝,当线程1改变变量值时,其他线程并不马上知道该变量值改变了,volatile就保证了变量值对各个线程可见,一个线程改变该值,马上其他线程中该值也改变。原子性表明操作不可中断,如基本变量赋值。
代码示例:
- <span style="font-size:16px;">package demo.thread;
- public class VolatileDemo implements Runnable {
- private volatile int i = 0;//volatile设置可见性
- public synchronized int getValue() {
- return i;
- }
- private synchronized void enenIncrement() {
- i++;
- i++;
- }
- @Override
- public void run() {
- while (true)
- enenIncrement();
- }
- public static void main(String[] args) {
- VolatileDemo at = new VolatileDemo();
- new Thread(at).start();
- while (true) {
- int val = at.getValue();
- if (val % 2 != 0) {//出现奇数,退出程序
- System.out.println(val+" is not enen!");
- System.exit(0);
- }
- }
- }
- }
- </span>
注意i++操作并不是原子行操作,getValue() 方法也要加synchronized 。
java多线程总结四:volatile、synchronized示例的更多相关文章
- Java多线程(四)—— synchronized关键字续
1.synchronized原理 在java中,每一个对象有且仅有一个同步锁.这也意味着,同步锁是依赖于对象而存在.当我们调用某对象的synchronized方法时,就获取了该对象的同步锁.例如,sy ...
- Java多线程核心技术(四)Lock的使用
本文主要介绍使用Java5中Lock对象也能实现同步的效果,而且在使用上更加方便. 本文着重掌握如下2个知识点: ReentrantLock 类的使用. ReentrantReadWriteLock ...
- Java多线程(四)java中的Sleep方法
点我跳过黑哥的卑鄙广告行为,进入正文. Java多线程系列更新中~ 正式篇: Java多线程(一) 什么是线程 Java多线程(二)关于多线程的CPU密集型和IO密集型这件事 Java多线程(三)如何 ...
- “全栈2019”Java多线程第四章:设置和获取线程名称
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...
- java多线程之内存可见性-synchronized、volatile
1.JMM:Java Memory Model(Java内存模型) 关于synchronized的两条规定: 1.线程解锁前,必须把共享变量的最新值刷新到主内存中 2.线程加锁时,将清空工作内存中共享 ...
- java多线程中的volatile和synchronized
package com.chzhao; public class Volatiletest extends Thread { private static int count = 0; public ...
- java多线程系列(四)---Lock的使用
Lock的使用 前言:本系列将从零开始讲解java多线程相关的技术,内容参考于<java多线程核心技术>与<java并发编程实战>等相关资料,希望站在巨人的肩膀上,再通过我的理 ...
- java多线程(3)---synchronized、Lock
synchronized.Lock 一.概述 1.出现线程不安全的原因是什么? 如果我们创建的多个线程,存在着共享数据,那么就有可能出现线程的安全问题:当其中一个线程操作共享数据时,还未操作完成,另外 ...
- java多线程系列(四)---ReentrantLock的使用
Lock的使用 前言:本系列将从零开始讲解java多线程相关的技术,内容参考于<java多线程核心技术>与<java并发编程实战>等相关资料,希望站在巨人的肩膀上,再通过我的理 ...
- java多线程:线程同步synchronized(不同步的问题、队列与锁),死锁的产生和解决
0.不同步的问题 并发的线程不安全问题: 多个线程同时操作同一个对象,如果控制不好,就会产生问题,叫做线程不安全. 我们来看三个比较经典的案例来说明线程不安全的问题. 0.1 订票问题 例如前面说过的 ...
随机推荐
- Javascript注意事项二【避免误用parseInt】
parseInt("123abc"); //123parseInt("1.73"); //1parseInt(".123"); //NaN ...
- ACM_基础知识
1. PI值的定义 const double PI = acos(-1.0); 2. system("pause")就是调用从程序中调用系统命令,而"pause" ...
- oracle强制使用索引
select /*+ INDEX(表名,索引名称) */ col_1,...from xxx
- Unity3D4.x之AssetBundle学习笔记
关于AssetBundle AssetBundle可用来将多个资源打包为一个文件,实现动态下载和更新.需要注意的是Unity3D5.x以后对打包方式进行了升级,不用再在依赖关系上伤透脑筋,但是和4.x ...
- Winform- 界面开发之布局控件"WeifenLuo.WinFormsUI.Docking"的使用
布局控件"WeifenLuo.WinFormsUI.Docking"是一个非常棒的开源控件,用过的人都深有体会,该控件之强大.美观.不亚于商业控件.而且控件使用也是比较简单的. 实 ...
- c# 泛型List的定义、作用、用法
定义:List<T>类表示可通过索引访问的对象的强类型列表,提供用于对列表进行搜索.排序和操作的方法. 作用: 泛型最常见的用途是泛型集合 我们在创建列表类时,列表项的数据类型可能是int ...
- 我的第一篇博客 ——【ToDoList】小程序开发
我是一只即将大四的大三狗,这是我的第一篇博客,说来惭愧.今年1月份,学校放寒假的时候开始自学的IOS,放假的时候比较起劲,看了一堆Object-C的视频,然后照着中英文对照的IOS基础开发教程,做了两 ...
- Hooks——钩子概览
(资料源于MSDN,本文仅对其进行翻译.批注.其链接为:http://msdn.microsoft.com/en-us/library/windows/desktop/ms644959%28v=vs. ...
- 【49】了解new_handler的行为
1.使用operator new无法获取内存时,对于旧式编译器,会返回一个null指针.对于新式编译器,会抛出一个异常. 2.考虑下面的需求,当operator new 无法获取内存时,程序员期望获得 ...
- Codeforces Round #200 (Div. 1)D. Water Tree dfs序
D. Water Tree Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/343/problem/ ...