Java 8的这个新特性,你用了吗?
今天写的一个方法,如下,实现的功能就是将数据库中的表字段获取,存在map 中,将前端传过来的数据存在map 中。
/**
* 给表中添加数据
* @param object 数据
*/
protected void fillData(String type,Object object){
JSONArray jsonArr = changeObjetToJsonArray(object);
if(jsonArr==null || jsonArr.isEmpty()){
return;
}
//遍历jsonarray
GetDateByTableName getDateByTableName=new GetDateByTableName(testNumber,imsi);
for(int i=0;i<jsonArr.size();i++){
List<String[]> rowList=new ArrayList<>();
JSONObject jsonItem=jsonArr.optJSONObject(i);
if(jsonItem!=null){
String typeName=BaseDataPreHandle.typeNamePreHandle(type, jsonItem);
TypeForTableName typeForTableName=Enum.valueOf(TypeForTableName.class,typeName.toUpperCase());
String tableName=typeForTableName.getTableName();
/**mapTableColumns.computeIfAbsent(tableName,k -> getFieldByTableName(tableName));*/
if(mapTableColumns.get(tableName)==null){
mapTableColumns.put(tableName,getFieldByTableName(tableName));
}
String[] row= getDateByTableName.getDatasbyName(tableName,jsonItem,mapTableColumns.get(tableName));
if(mapTableValues.get(tableName)!=null){
rowList=mapTableValues.get(tableName);
}
rowList.add(row);
mapTableValues.put(tableName,rowList);
}
}
}
异味?
然后我写完用sonarLint 检测发现两个异味,竟然提示我不要这么写,而是提示使用Map.computeIfAbsent()方法,当时很纳闷
使用了lambda表达式,然后我就看了一下computeIfAbsent()源码,如下
default V computeIfAbsent(K key,
Function<? super K, ? extends V> mappingFunction) {
Objects.requireNonNull(mappingFunction);
V v;
if ((v = get(key)) == null) {
V newValue;
if ((newValue = mappingFunction.apply(key)) != null) {
put(key, newValue);
return newValue;
}
}
return v;
}
我看这个方法的意思是如果get(key)为null,那么就创建一个value值,这个value值就是lambda表达式的返回值,并且这个返回值不为null,才会将这个这个值存到map对应的key 中。再来看一下我最开始写的代码的需求
if(mapTableColumns.get(tableName)==null){
mapTableColumns.put(tableName,getFieldByTableName(tableName));
}
如果map中这个key 对应的值不存在,就将新值赋给这个key。和上面computeIfAbsent() 的方法是一致的,因为我可以保证新值不为null。简单的说就是,map中key值存在就不更新,不存在就更新。所以修改为如下:
mapTableColumns.computeIfAbsent(tableName,k -> getFieldByTableName(tableName));
再接着看我后面的代码,单独拿出来
String[] row= getDateByTableName.getDatasbyName(tableName,jsonItem,mapTableColumns.get(tableName));
if(mapTableValues.get(tableName)!=null){
rowList=mapTableValues.get(tableName);
}
rowList.add(row);
mapTableValues.put(tableName,rowList);
这个的意思是,有一个list集合,如果map.get(key)不为空,就将这个值赋给list集合,如果这个值为空就跳过,之后,将list添加一条数据,然后将新的list集合添加到这个map对应的key 值中。简单的说就是更新map 对应key 的值。不管map.get(key)是否为空。
那这样可以使用computeIfAbsent()方法吗,显然是不可以的,因为源代码很清楚,只有当get(key)为空时才会进行更新,不为空的话直接就返回了。
所以computeIfAbsent() 方法的用法总结:
只有在当前 Map 中 key 对应的值不存在或为 null 时
才调用 mappingFunction
并在 mappingFunction 执行结果非 null 时
将结果跟 key 关联.
mappingFunction 为空时 将抛出空指针异常
修改
那么问题来了,现在我的这段代码snoarLint 检测出异味,改怎么修改呢? 这让我想到了computeIfPresent()方法,看了一下源码
default V computeIfPresent(K key,
BiFunction<? super K, ? super V, ? extends V> remappingFunction) {
Objects.requireNonNull(remappingFunction);
V oldValue;
if ((oldValue = get(key)) != null) {
V newValue = remappingFunction.apply(key, oldValue);
if (newValue != null) {
put(key, newValue);
return newValue;
} else {
remove(key);
return null;
}
} else {
return null;
}
}
看源码的意思是:当get(key)值存在时,并且新值不为空,就会更新。否则就返回null。
这与我的需求不相符啊还是,我的需求是,不管key值是否为空,都需要进行更新,我这个更新还是在原来的值上增加数据。
那现在怎样消除异味呢?
实在没办法我使用了computeIfAbsent()和computeIfPresent()两种方法结合。先用computeIfAbsent()方法,保证一定可以获取到get(key) 再用computeIfPresent() 方法,保证一定对get(key)更新。如下:
String[] row= getDateByTableName.getDatasbyName(tableName,jsonItem,mapTableColumns.get(tableName));
mapTableValues.computeIfAbsent(tableName,k->new ArrayList<String[]>());
List<String[]> rowList=mapTableValues.get(tableName);
rowList.add(row);
mapTableValues.computeIfPresent(tableName,(k,v) -> rowList);
最后再检测,消除异味。虽然这段代码看上去没有原来那么好理解,但是我觉得还是个人对这个方法熟悉与否,put和get方法大家比较熟悉,所以一眼就能看懂,但是computeIfAbsent和computeIfPresent 方法在java 8 中才有,大家用的比较少,所以才显得陌生。
最后附上这个方法改好的代码
/**
* 给表中添加数据
* @param object 数据
*/
protected void fillData(String type,Object object){
JSONArray jsonArr = changeObjetToJsonArray(object);
if(jsonArr==null || jsonArr.isEmpty()){
return;
}
//遍历jsonarray
GetDateByTableName getDateByTableName=new GetDateByTableName(testNumber,imsi);
for(int i=0;i<jsonArr.size();i++){
JSONObject jsonItem=jsonArr.optJSONObject(i);
if(jsonItem!=null){
String typeName=typeNamePreHandle(type, jsonItem);
TypeForTableName typeForTableName=Enum.valueOf(TypeForTableName.class,typeName.toUpperCase());
String tableName=typeForTableName.getTableName();
mapTableColumns.computeIfAbsent(tableName,k -> getFieldByTableName(tableName));
String[] row= getDateByTableName.getDatasbyName(tableName,jsonItem,mapTableColumns.get(tableName));
mapTableValues.computeIfAbsent(tableName,k->new ArrayList<String[]>());
List<String[]> rowList=mapTableValues.get(tableName);
rowList.add(row);
mapTableValues.computeIfPresent(tableName,(k,v) -> rowList);
}
}
}
- 人工智能尚处探索阶段,为何我们对此异常焦虑
- ExtJs学习笔记(22)-XTemplate + WCF 打造无刷新数据分页
- 同步服务器系统时间操作记录
- kvm虚拟化管理平台WebVirtMgr部署-完整记录(安装Windows虚拟机)-(4)
- ExtJs学习笔记(11)_Absolute布局和Accordion布局
- ExtJs学习笔记(9)_Window的基本用法
- DateTime在ExtJs中无法正确序列化的问题
- ELK实时日志分析平台环境部署--完整记录
- 梳理Linux下OSI七层网络与TCP/IP五层网络架构
- 字符编码-使用c#研究
- iframe高度自适应的IE解决方案
- javascript读写本机文本文件
- 崔立鹏:腾讯云为知识竞技游戏提供解决方案
- Mysql之binlog日志说明及利用binlog日志恢复数据操作记录
- java教程
- Java快速入门
- Java 开发环境配置
- Java基本语法
- Java 对象和类
- Java 基本数据类型
- Java 变量类型
- Java 修饰符
- Java 运算符
- Java 循环结构
- Java 分支结构
- Java Number类
- Java Character类
- Java String类
- Java StringBuffer和StringBuilder类
- Java 数组
- Java 日期时间
- Java 正则表达式
- Java 方法
- Java 流(Stream)、文件(File)和IO
- Java 异常处理
- Java 继承
- Java 重写(Override)与重载(Overload)
- Java 多态
- Java 抽象类
- Java 封装
- Java 接口
- Java 包(package)
- Java 数据结构
- Java 集合框架
- Java 泛型
- Java 序列化
- Java 网络编程
- Java 发送邮件
- Java 多线程编程
- Java Applet基础
- Java 文档注释
- php给数组赋值的实例方法
- php实现分页功能的详细实例方法
- 浅谈Laravel中的三种中间件的作用
- laravel 使用auth编写登录的方法
- laravel框架 laravel-admin上传图片到oss的方法
- php实现推荐功能的简单实例
- php实现登录页面的简单实例
- Laravel 自带的Auth验证登录方法
- laravel实现上传图片的两种方式小结
- Laravel开启跨域请求的方法
- Laravel 5.4前后台分离,通过不同的二级域名访问方法
- Yii框架数据库查询、增加、删除操作示例
- 在laravel5.2中实现点击用户头像更改头像的方法
- laravel 多图上传及图片的存储例子
- Laravel 添加多语言提示信息的方法