本文属原创,转载请注明出处:http://www.cnblogs.com/robinjava77/p/5481608.html  (Robin)

QuoteTest(引用对象技巧)

 import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map; /**
* Created by robin on 2016/4/13.
* 引用型对向操作总结:
* 1.被引用的对象,值改变时,会直接改变引用源的值;
* 2.当引用的对象,改变其引用源时,对其操作,只会改变新的引用源的值,并不会影响之前的引用源的值
* 3.从map中获取的引用不存在时,需要将新的引用put到map中,map中该位置的值,才会被引入
* @author robin
*/
public class QuoteTest { public static void main(String args[]){
Map<String,List<String>> map = new HashMap<String, List<String>>();
for (int i =0;i< 5;i++){
List<String> datalist = new ArrayList<String>();
for (int j=0;j<10;j++){
datalist.add(i*10+""+j);
}
map.put(""+i,datalist);
}
for (List<String> list:map.values()){
System.out.println(listToString(list));
}
System.out.println("----------分隔线1-----------");
List<String> tempList = map.get("3");
tempList.add("avc");
tempList.remove("300");
for (List<String> list:map.values()){
System.out.println(listToString(list));
}
System.out.println("----------分隔线2-----------");
List<String> tempList2 = map.get("2");//tempList 获得map中 key为2的引用
List<String> replaceList = new ArrayList<String>();
tempList2 = replaceList;////tempList 获得其他list的引用,失去map中 key为2的引用,此时对templist2做任何操作,影响的时replaceList引用的区域
tempList2.add("replaceList的值被改变");
for (List<String> list:map.values()){
System.out.println(listToString(list));
}
System.out.println("replaceList的值:"+listToString(replaceList));
System.out.println("----------分隔线3-----------");
List<String> tempList3 = map.get("2");
tempList3 = replaceList;
map.put("2",tempList3);
for (List<String> list:map.values()){
System.out.println(listToString(list));
}
System.out.println("----------分隔线4-----------");
List<String> notExistList = map.get("5");
if(notExistList == null){
notExistList = new ArrayList<String>();
}
notExistList.add("第5行数据添加进来...");
for (List<String> list:map.values()){
System.out.println(listToString(list));
}
System.out.println("----------分隔线5-----------");
List<String> notExistList2 = map.get("6");
if(notExistList2 == null){
notExistList2 = new ArrayList<String>();
}
notExistList2.add("第6行数据添加进来...");
map.put("6",notExistList2);
for (List<String> list:map.values()){
System.out.println(listToString(list));
}
System.out.println("----------分隔线5-----------"); Map<String,Map<String,String>> mapOne = new HashMap<String, Map<String, String>>();
String keyss = "mapTest";
Map<String,String> mapTwo = new HashMap<String, String>();
mapOne.put(keyss,mapTwo);
System.out.println("mapOne的数据:" + mapToString(mapOne));
System.out.println("----------分隔线6-----------");
mapTwo.put("aaa", "aaav");
mapTwo.put("bbb", "bbbv");
mapTwo.put("ccc","cccv");
System.out.println("mapOne的数据:"+mapToString(mapOne));
System.out.println("----------分隔线7-----------");
} private static String listToString(List<String> list){
StringBuilder sb = new StringBuilder("");
for (String s:list){
sb.append("["+s+"] ");
}
return sb.toString();
} private static String mapToString(Map<?,?> map){
StringBuilder sb = new StringBuilder("");
for(Map.Entry entry:map.entrySet()){
sb.append("[key:"+entry.getKey()+";value:"+entry.getValue()+"]");
}
return sb.toString();
} }

---------------------

引用对象易产生的bug:

2016.05.11

关于引用对象,使用不恰当,很容易给自己挖坑,产生非常严重的bug,进而导致整个系统实际业务的崩溃,而且这种bug很难被查出来。(如果日志记录不够详细,分析不够彻底,要找出这种bug,只能靠上帝保佑)

下面先上bug 代码 demo

 import java.util.Iterator;
import java.util.List;
import java.util.Vector; /**
* Created by robin on 2016/5/11.
*
* @author robin
*/
public class QuoteBugDemo { private static List<Integer> publicNums = new Vector<Integer>(); public static void main(String args[]) throws InterruptedException {
initPublicNums();//初始化公共数据源 timeTask(1);//模拟执行定时任务1 timeTask(2);//模拟执行定时任务2 } private static void initPublicNums(){
for (int i =0;i < 10;i++){
publicNums.add(i);
}
} /**
* 这块代码模拟的逻辑:
* 1.每天获取配置好10个的数据源;
* 2.检查这10个数据源,当数据源的数据准备好后,开始执行数据同步;
* 3.从当天的带同步数据源list删除已经同步的数据;
* @param mark
* @throws InterruptedException
*/
private static void timeTask(int mark) throws InterruptedException {
final long start = System.currentTimeMillis();//程序开始运行时间
//每天待同步数据源
List<Integer> dataSources = publicNums;
StringBuffer sb = new StringBuffer("mark("+mark+");公共数据源数目:"+dataSources.size()+";数据源列表[");
for (Integer i:dataSources){
sb.append(i+",");
}
sb.append("]");
System.out.println("日志【"+sb.toString()+"】");
while(true){
long seconds = (System.currentTimeMillis() - start) / 1000;
if(seconds > 15l){
System.out.println("运行超过限定时间:15秒,退出");
break;
}
Iterator<Integer> ite = dataSources.iterator();
while (ite.hasNext()){//
int dataSource = ite.next();
boolean flag = isOk(dataSource);
if(flag){//数据源数据已准备好
System.out.println("对数据源:"+dataSource+"进行数据处理。");//同步数据
ite.remove();//待同步数据源删除该数据源
}
}
if(dataSources.size() != 0){
Thread.sleep(1000);
}else{
break;
}
}
System.out.println("定时任务mark["+mark+"]已经执行完毕");
} /**
* 模拟检查数据源是否OK
* @param dataSource
* @return
*/
private static boolean isOk(int dataSource){
if(dataSource%2 == 0){
return true;
}
return false;
} }

执行结果:

 日志【mark(1);公共数据源数目:10;数据源列表[0,1,2,3,4,5,6,7,8,9,]】
对数据源:0进行数据处理。
对数据源:2进行数据处理。
对数据源:4进行数据处理。
对数据源:6进行数据处理。
对数据源:8进行数据处理。
运行超过限定时间:15秒,退出
定时任务mark[1]已经执行完毕
日志【mark(2);公共数据源数目:5;数据源列表[1,3,5,7,9,]】
运行超过限定时间:15秒,退出
定时任务mark[2]已经执行完毕

定时任务2,执行的时候,数据源只剩1,3,5,7,9。

改进方案:将公共数据源保护起来,仅提供公共数据源的副本:shallow copy和deep copy

核心代码:

 /**
* 改进方案1:获取公共数据源对象的副本
* shallow copy:list中 元素引用 仍然是相同的
* @return
*/
private static List<Integer> getPublicNums(){
List<Integer> clone = new ArrayList<Integer>(publicNums);
return clone;
}

改进后全部代码:

 import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.Vector; /**
* Created by robin on 2016/5/11.
*
* @author robin
*/
public class QuoteBugDemo { private static List<Integer> publicNums = new Vector<Integer>(); public static void main(String args[]) throws InterruptedException {
initPublicNums();//初始化公共数据源 timeTask(1);//模拟执行定时任务1 timeTask(2);//模拟执行定时任务2 } private static void initPublicNums(){
for (int i =0;i < 10;i++){
publicNums.add(i);
}
} /**
* 改进方案1:获取公共数据源对象的副本
* shallow copy:list中 元素引用 仍然是相同的
* @return
*/
private static List<Integer> getPublicNums(){
List<Integer> clone = new ArrayList<Integer>(publicNums);
return clone;
} /**
* 这块代码模拟的逻辑:
* 1.每天获取配置好10个的数据源;
* 2.检查这10个数据源,当数据源的数据准备好后,开始执行数据同步;
* 3.从当天的带同步数据源list删除已经同步的数据;
* @param mark
* @throws InterruptedException
*/
private static void timeTask(int mark) throws InterruptedException {
final long start = System.currentTimeMillis();//程序开始运行时间
//每天待同步数据源
List<Integer> dataSources = getPublicNums();//改进方案1
StringBuffer sb = new StringBuffer("mark("+mark+");公共数据源数目:"+dataSources.size()+";数据源列表[");
for (Integer i:dataSources){
sb.append(i+",");
}
sb.append("]");
System.out.println("日志【"+sb.toString()+"】");
while(true){
long seconds = (System.currentTimeMillis() - start) / 1000;
if(seconds > 15l){
System.out.println("运行超过限定时间:15秒,退出");
break;
}
Iterator<Integer> ite = dataSources.iterator();
while (ite.hasNext()){//
int dataSource = ite.next();
boolean flag = isOk(dataSource);
if(flag){//数据源数据已准备好
System.out.println("对数据源:"+dataSource+"进行数据处理。");//同步数据
ite.remove();//待同步数据源删除该数据源
}
}
if(dataSources.size() != 0){
Thread.sleep(1000);
}else{
break;
}
}
System.out.println("定时任务mark["+mark+"]已经执行完毕");
} /**
* 模拟检查数据源是否OK
* @param dataSource
* @return
*/
private static boolean isOk(int dataSource){
if(dataSource%2 == 0){
return true;
}
return false;
} }

执行结果:

 日志【mark(1);公共数据源数目:10;数据源列表[0,1,2,3,4,5,6,7,8,9,]】
对数据源:0进行数据处理。
对数据源:2进行数据处理。
对数据源:4进行数据处理。
对数据源:6进行数据处理。
对数据源:8进行数据处理。
运行超过限定时间:15秒,退出
定时任务mark[1]已经执行完毕
日志【mark(2);公共数据源数目:10;数据源列表[0,1,2,3,4,5,6,7,8,9,]】
对数据源:0进行数据处理。
对数据源:2进行数据处理。
对数据源:4进行数据处理。
对数据源:6进行数据处理。
对数据源:8进行数据处理。
运行超过限定时间:15秒,退出
定时任务mark[2]已经执行完毕

已达预期。

------------------------------------------

shallow copy 和 deep copy 的区别详见我的另一篇博客:http://www.cnblogs.com/robinjava77/p/5481874.html

引用对象的使用和易产生bug的示例的更多相关文章

  1. struts2:遍历自定义字符串数组,遍历Action实例所引用对象中的数组

    在struts2:OGNL表达式,遍历List.Map集合:投影的使用一文中已经讲述了OGNL遍历List.Map集合等功能. 本文简单写一个遍历数组的示范程序. 1. 遍历自定义字符串数组 < ...

  2. 理解Java的引用对象

    SoftReferenceWeakReference 的特性基本一致, 最大的区别在于 SoftReference会尽可能长的保留引用,不会在GC时就回收对象,而是直到JVM 内存不足时才会被回收(虚 ...

  3. Andorid Binder进程间通信---Binder本地对象,实体对象,引用对象,代理对象的引用计数

    本文參考<Android系统源码情景分析>,作者罗升阳. 一.Binder库(libbinder)代码: ~/Android/frameworks/base/libs/binder --- ...

  4. C++实现引用和被引用对象分离

    上一篇博客简单介绍了C++中引用的底层实现,接下来,通过简单的代码验证如何将引用和被引用对象分离. 代码如下; #include <iostream> #include<string ...

  5. Effective Java 第三版——64. 通过对象的接口引用对象

    Tips 书中的源代码地址:https://github.com/jbloch/effective-java-3e-source-code 注意,书中的有些代码里方法是基于Java 9 API中的,所 ...

  6. 浅谈String中的==和对象中引用对象类型的==

    @Test public void test02() { StringBuffer sb = new StringBuffer(); sb.append('a'); sb.append(11); Sy ...

  7. 使用引用类型变量来访问所引用对象的属性和方法时,Java 虚拟机绑定规则

    通过引用类型变量来访问所引用对象的属性和方法时,Java 虚拟机将采用以下绑定规则: 实例方法与引用变量实际引用的对象的方法进行绑定,这种绑定属于动态绑定,因为是在运行时由 Java 虚拟机动态决定的 ...

  8. Python中的变量引用对象需注意的几点

    Python中的变量引用对象需注意的几点 分类:Python (55)  (0) 普通引用: Python中,变量的作用仅仅是一个标识,只有赋值后才被创建,它可以引用任何类型的对象,而且在引用之前必须 ...

  9. 使用NSHashTable存储引用对象

    NSHashTable 我们使用集合(NSArray,NSMutableArray,NSDictionary,NSMutableDictionary,NSSet,NSMutableSet)存储对象时会 ...

随机推荐

  1. xmlstreaml xml过滤 格式化 报文的发送接收 struct2

    有时候把东西想的过于复杂了,还是思路不清晰啊. seervlet struct2配置过程

  2. CSS布局奇技淫巧:各种居中

    居中是我们使用css来布局时常遇到的情况.使用css来进行居中时,有时一个属性就能搞定,有时则需要一定的技巧才能兼容到所有浏览器,本文就居中的一些常用方法做个简单的介绍. 注:本文所讲方法除了特别说明 ...

  3. (转)如何进行Monkey Test

    如何进行Monkey Test   目录 一 简介 二 测试准备 三 基本命令格式 四 测试Log获取 五 Monkey命令参数介绍 六 保存monkey log以及手机log到sdcard(新增) ...

  4. 在MyEclipse和Eclipse中添加Hibernate开发工具

    一.插件准备 MyEclipse需要的插件:HibernateTools-3.2.4.zip Eclipse需要的插件:jbosstools-4.2.3.Final_2015-03-26_22-41- ...

  5. PHP中的运算符---算术运算符、逻辑运算符、赋值运算符、比较运算符

    1.算术运算符 常见的算术运算符 运算类型 运算符 举例 结果 取反运算 - -$a 返回$a的负值 加法运算 + $a + $b 返回$a与$b的和 减法运算 - $a - $b 返回$a与$b的差 ...

  6. C#参数化执行SQL语句,防止漏洞攻击本文以MySql为例【20151108非查询操作】

    为什么要参数化执行SQL语句呢? 一个作用就是可以防止用户注入漏洞. 简单举个列子吧. 比如账号密码登入,如果不用参数, 写的简单点吧,就写从数据库查找到id和pw与用户输入一样的数据吧 sql:se ...

  7. 在线markdown编辑器

    https://www.zybuluo.com/mdeditor http://mahua.jser.me/

  8. 转-springAOP基于XML配置文件方式

    springAOP基于XML配置文件方式 时间 2014-03-28 20:11:12  CSDN博客 原文  http://blog.csdn.net/yantingmei/article/deta ...

  9. c# equals与==

    对于值类型,如果对象的值相等,则相等运算符 (==) 返回 true,否则返回 false.对于string 以外的引用类型,如果两个对象引用同一个对象,则 == 返回 true.对于 string ...

  10. 使用IntelliJ IDEA 配置Maven(入门)(转)

    原文转自:http://blog.csdn.net/qq_32588349/article/details/51461182 1. 下载Maven 官方地址:http://maven.apache.o ...