一、集合类
集合运算是指集合的并集、交集、差集、补集等一系列运算。
差集运算是业务开发中常见的需求:新增数据保存、已有数据更新、没有数据删除,简单粗暴的解决方式是,保存之前先把原来符合条件的数据删除,删除完成后然后执行新增保存操作。这种比较简单,缺点比较明显:操作数据量较多的场景下,事务回滚容易造成数据库拒绝响应;先删除后再次保存理论情况下存在元数据信息丢失的情况,最明显的是记录的入库时间是变化的;事务回滚异常的情况下,丢失数据的风险极高。
通过交并差集合运算,能够改善上述存在的弊端。
字符串 List 的差集运算,一般情况下,此字符串是主键。
/**
* 差集 A-B
* @param listA 集合List A
* @param listB 集合List B
* @Return 差集集合
*/
public
static
List<String>
diffSet(List<String> listA
,
List<String> listB
)
{
List<String> lists
=
new
ArrayList<>(listA
.size());
// 复制一个新集合
lists
.addAll(listA
);
// 并集
lists
.addAll(listB
);
// 差集
lists
.removeAll(listB
);
return lists
;
}
有关集合运算 JDK 内置解决方式暂缺,考虑到这是一种普遍的需求,理应该有第三方成熟框架的实现。CollectionUtils工具类提供了此能力,详细内容请阅读源码,此处不提供讲解示例。
TreeMap 与 LinkedHashMap 的区别主要通过稳定性和有序性来体现,更准确的说是通过稳定性来区分的。
TreeMap | LinkedHashMap | 解释 | |
稳定性 | 否 | 是 | 稳定性是指当集合发生元素增减时,已存在元素的相对位置不发生改变 |
有序性 | 是 | 是 | key 值按照一定策略排序 |
TreeMap 按照 key 值的字典顺序排序;LinkedHashMap 按照添加的先后顺序排序。
场景推广
当前端按照一定顺序(非字典顺序)请求数据,希望返回结果与提供的顺序一致,用哪种数据结构可以实现?
MapUtils是处理Map
常用的工具类。当 value 为基本类型时,内置非空判断;当 value 为对象类型时,建议使用如下语法:
Optional.ofNullable(shareMap
.get(e
.getShareId())).map(Share::getShareDistinct).get()
上述语法解决了当 value 为对象类型且为null
时,获取属性值空指针异常。
字典顺序正反向排序、整形数据正反向排序比较常见,那么拼音排序
呢,下面讲解拼音排序的实现。
Lambda 表达式的引入,极大的提高了代码书写的优雅性,因此采用此语法提供测试样例。
@Test
public
void
testSort()
{
List<String> list
=
Arrays.asList("天安门",
"中国",
"长城");
// 原集合顺序
list
.forEach(System.out
::println);
// 字典顺序排序
list
.stream().sorted().forEach(System.out
::println);
// 拼音顺序排序
list
.stream().sorted(Collator.getInstance(Locale.CHINA
)).forEach(System.out
::println);
}
其中Collator类表示字符校验方式。
二、通用类
资源自动关闭有利于提高编码效率,实现AutoCloseable
接口的资源类通过try-with-resource
的方式可以实现自动关闭。
// sqlSession不需要显示的关闭,虚拟机通过内部管理自动关闭连接
try
(SqlSession sqlSession
= sqlSessionService
.getSqlSession(config
))
{
LambdaQueryWrapper<MetaTable> wrapper
=
Wrappers.lambdaQuery(MetaTable.class).select(MetaTable::getTableName)
.eq(MetaTable::getTableSchema, config
.getDbName()).eq(MetaTable::getTableType, BASE_TABLE
);
tableNames
.addAll(sqlSession
.getMapper(MetaTableMapper.class).selectList(wrapper
)
.stream().map(MetaTable::getTableName).collect(toList()));
}
流、数据库连接、Socket连接等资源,使用隐式自动关闭既能避免因忘记关闭资源导致的资源泄露问题,也能减少冗余代码。
三、执行第三方脚本
Java 程序有时需要调用外部的脚本命令,比方说用 Python 编写的算法脚本等,因为语言特性不同,应用往往不简单使用一种语言,语言间的调用是常见的需求。
@SneakyThrows
public
void
testScript()
{
// 脚本命令自定义,可以是shell命令也可以是python命令,还可以是java命令
String cmd
=
"python -V > /dev/null";
// 执行命令(确保可执行文件具有执行权限,没有先添加权限)
Process exec
=
Runtime.getRuntime().exec(cmd
);
// 阻塞等待脚本执行完成(默认采用异步执行的方式)
exec
.waitFor(5,
TimeUnit.SECONDS
);
// 返回错误码:0代表正常执行
int exitValue
= exec
.exitValue();
System.out
.println("exitValue = "
+ exitValue
);
}
调用外部脚本,可能会遇到以下问题:执行文件没有执行权限;脚本的执行时间长短;异步或者同步调用脚本;脚本执行返回的状态码。
四、压缩文件
压缩与解压缩文件在项目的需求比较常见,Java 内置了 ZIP 压缩文件处理能力,使用第三方工具(开发)效率更高。
<dependency>
<groupId>cn.hutool
</groupId>
<artifactId>hutool-core
</artifactId>
<version>5.6.2
</version>
</dependency>
hutool 工具包对中文文件名支持较好
// 显示的指明字符编码集
ZipUtil.unzip(zipFilePath
, outFileDir
,
StandardCharsets.UTF_8
);
看完如果有帮助,希望可以给个 三连 ,你的鼓励就是我不断前进的动力。谢谢
关注我:私信获取Java高级架构资料、大厂面试试题、视频