java 并发原子性与易变性  具体介绍请參阅thinking in java4 21.3.3

thinking in java 4免费下载:http://download.csdn.net/detail/liangrui1988/7580155

package org.rui.thread.volatiles;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
/**
* 假设你盲目地应用原子性慨念。那么就会看到在以下程序中的getValue符合上面的描写叙述
*
* 可是,该程序将找到奇数值并终止。虽然return i确实是原子性操作。可是缺少同步使得其数值能够在处于不稳定的中间状态时被读取。
* 除此之外,因为i也不是volatile的,因此还存在可视性问题
* @author lenovo
*
*/
public class AtomicityTest implements Runnable { private int i=0;
public int getValue(){return i;}
private synchronized void evenIncrement(){i++;i++;}
@Override
public void run() {
while(true)
evenIncrement(); } public static void main(String[] args) {
ExecutorService exec=Executors.newCachedThreadPool();
AtomicityTest at=new AtomicityTest();
exec.execute(at);
while(true)
{
int val=at.getValue();
if(val%2!=0)
{
System.out.println(val);
System.exit(0);
}
}
} }
package org.rui.thread.volatiles;
/**
* 考虑一些更简单的事情。一个产生序列数字的类,
* 每当nextSerial-Number被调用时,它必须向调用者返回唯一的值
* @author lenovo
*
*/
public class SerialNumberGenerator { private static volatile int serialNumber=0;
public static int nextSerialNumber()
{
return serialNumber++;//非线程安全
}
}
package org.rui.thread.volatiles;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
/**
* 为了測试SerialNumberGenerator 我们须要不会耗尽内存的集(SET) 。
* 以防须要花费非常长的时间来探測问题。 这里所看到的的circularSet重用了存储int数值的内存,
* 并假设在你生成序列数时,产生数值覆盖冲突的可能性极小。add 和contains 方法都 是synchronized,以防示线程冲突
* @author lenovo
*
*/
//重用存储所以我们不耗尽内存
//reuses storage so we don't run out of memory
class CircularSet
{
private int[] array;
private int len;
private int index=0; public CircularSet(int size){
array=new int[size];
len=size;
//初始化一个值而不是生产
//这个初始全是-1和serialNumberGenerator 不同样。其后取serialNumberGenerator.next...存入
//by the serialNumberGenerator;
for(int i=0;i<size;i++)
array[i]=-1;
} //add
public synchronized void add(int i)
{
array[index]=i;
//wrap index and write over old elements; 将指数和写在旧元素;
index=++index % len;
//System.out.println(index+" : len :"+len); } //contains
public synchronized boolean contains(int val)
{
for(int i=0;i<len;i++)
{
System.out.println(array[i]+" == "+val);
if(array[i]==val)
return true; }
return false;
} }
///////////////////////////////////////////////
public class SerialNumberChecker {
private static final int SIZE=10;
private static CircularSet serials=new CircularSet(1000);
private static ExecutorService exec=Executors.newCachedThreadPool(); static class SerialChecker implements Runnable
{
@Override
public void run() {
while(true)
{
//自增长
int serial=SerialNumberGenerator.nextSerialNumber();
if(serials.contains(serial))
{
//反复
System.out.println("Duplicate: "+serial);
System.exit(0);
}
serials.add(serial);
}
}
} ////
public static void main(String[] args) throws NumberFormatException, InterruptedException {
String[] arg=new String[]{"10000"}; for(int i=0;i<SIZE;i++)
{
exec.execute(new SerialChecker());
//stop after n seconds if there 's an argument 停止在n秒后假设有一个论点
if(arg.length>0)
{
TimeUnit.SECONDS.sleep(new Integer(arg[0]));
System.out.println("没有反复检測");
System.exit(0);
}
}
}
} /**
* 通过创建多个任务来竞争序列数,你将发现这些任务终于会得到反复的序列数,假设你执行的时间足够长的话
* 为了解决问题,nextSerialNumber 前面加入 了synchronized关健字
*/

最新文章

  1. ubuntu声音系统
  2. Integer与int的区别
  3. Expression Template(表达式模板,ET)
  4. IOS开发基础知识--碎片33
  5. 【poj1984】 Navigation Nightmare
  6. BestCoder Round #83
  7. C#_数据转换 实用方法
  8. vs2010 vc++ 统一修改所有工程的目录配置
  9. Powerdesigner逆向工程从sql server数据库生成pdm (转载)
  10. 免费公测:RDS只读实例
  11. PHP基础示例:商品信息管理系统v1.1
  12. PHP 字符串变量
  13. Arthas:线上问题排查工具
  14. linux 压缩文件或文件夹
  15. MySQL数据库SQL修改数据规范
  16. windows下Anaconda3配置TensorFlow深度学习库
  17. linux系统下安装两个或多个tomcat
  18. C#线程篇---线程池如何管理线程(6完结篇)
  19. ehcache.xml的配置详解和示例
  20. Websocket、长连接、循环连接

热门文章

  1. poj--1488--TEX Quotes(水题)
  2. linux 不常用命令及命令组合
  3. python 深浅拷贝小记
  4. 超好用的谷歌浏览器、Sublime Text、Phpstorm、油猴插件合集
  5. 编程里的API是什么意思?
  6. CentOS下部署巡风步骤详解
  7. PreferenceActivity、PreferenceFragment使用
  8. 慢慢人生路,学点Jakarta基础-深入剖析Java的接口和抽象类
  9. 47.Express文件上传
  10. 带你底层看Sqoop如何转换成MapReduce作业运行的(代码程序)