大家好,感谢邀请,今天来为大家分享一下google guava的问题,以及和谷歌工具类java的一些困惑,大家要是还不太明白的话,也没有关系,因为接下来将为大家分享,希望可以帮助到大家,解决大家的问题,下面就开始吧!
如何使初学java的人更加对其感兴趣
学习Java语言有以下入门级的书(注意:下面一些书在入门篇中有所提及,但为了完整性,还是要在这里提一下,因为可能有朋友是跳着看的)。
《Java核心技术:卷1基础知识》,这本书本来是Sun公司的官方用书,是一本Java的入门参考书。对于Java初学者来说,是一本非常不错的值得时常翻阅的技术手册。书中有较多地方进行Java与C++的比较,因为当时Java面世的时候,又被叫作"C++Killer"。C->C++->Java整条线融汇贯通,这对我未来的技术成长有非常大的帮助。
有了上述的入门后,Java的Spring框架是你玩Java所无法回避的东西,所以接下来是两本Spring相关的书,《Spring实战》和《SpringBoot实战》。前者是传统的Spring,后者是新式的微服务的Spring。如果你只想看一本的话,那么就看后者吧。
认真学习前面的书可以让你成功入门Java,但想要进一步成长,就要看下面我推荐的几本提升级的书。
接下来,你需要了解了一下如何编写高效的代码,于是必需看一下《EffectiveJava》(注意,这里我给的引用是第三版的,也是2017年末出版的书),这本书是模仿ScottMeyers的经典图书《EffectiveC++》的。Effective这种书基本上都是各种经验之谈,所以,这是一本非常不错的书,你一定要读。这里需要推荐一下GoogleGuava库,这个库不但是JDK的升级库,其中有如:集合(collections)、缓存(caching)、原生类型支持(primitivessupport)、并发库(concurrencylibraries)、通用注解(commonannotations)、字符串处理(stringprocessing)、I/O等库,其还是EffectiveJava这本书中的那些经验的实践代表。
《Java并发编程实战》,是一本完美的Java并发参考手册。书中从并发性和线程安全性的基本概念出发,介绍了如何使用类库提供的基本并发构建块,用于避免并发危险、构造线程安全的类及验证线程安全的规则,如何将小的线程安全类组合成更大的线程安全类,如何利用线程来提高并发应用程序的吞吐量,如何识别可并行执行的任务,如何提高单线程子系统的响应性,如何确保并发程序执行预期任务,如何提高并发代码的性能和可伸缩性等内容。最后介绍了一些高级主题,如显式锁、原子变量、非阻塞算法以及如何开发自定义的同步工具类。
了解如何编写出并发的程序,你还需要了解一下如何优化Java的性能。我推荐《Java性能权威指南》。通过学习这本书,你可以比较大程度地提升性能测试的效果。其中包括:使用JDK中自带的工具收集Java应用的性能数据,理解JIT编译器的优缺点,调优JVM垃圾收集器以减少对程序的影响,学习管理堆内存和JVM原生内存的方法,了解如何最大程度地优化Java线程及同步的性能,等等。看完这本书后,如果你还有余力,想了解更多的底层细节,那么,你有必要去读一下《深入理解Java虚拟机》。
《Java编程思想》,真是一本透着编程思想的书。上面的书让你从微观角度了解Java,而这本书则可以让你从一个宏观角度了解Java。这本书和Java核心技术的厚度差不多,但这本书的信息密度比较大。所以,读起来是非常耗大脑的,因为它会让你不断地思考。对于想学好Java的程序员来说,这是一本必读的书。
《精通Spring4.x》,也是一本很不错的书,就是有点厚,一共有800多页,都是干货。我认为其中最不错的是在分析原理,尤其是针对前面提到的Spring技术,应用与原理都讲得很透彻,IOC和AOP也分析得很棒,娓娓道来。其对任何一个技术都分析得很细致和全面,不足之处就是内容太多了,所以导致很厚,但这并不影响它是一本不错的工具书。
当然,学Java你一定要学面向对象的设计模式,这里就只有一本经典的书《设计模式》。如果你觉得有点儿难度了,那么可以看一下《HeadFirst设计模式》。
hutool为什么不推荐使用
Hutool是一个小而全的Java工具类库,通过静态方法封装,降低相关API的学习成本,提高工作效率,使Java拥有函数式语言般的优雅,让Java语言也可以“甜甜的”。
这个工具是国人写的,个人感觉不比谷歌的Guava工具差。
Hutool中的工具方法来自于每个用户的精雕细琢,它涵盖了Java开发底层代码中的方方面面,它既是大型项目开发中解决小问题的利器,也是小型项目中的效率担当;
Hutool是项目中“util”包友好的替代,它节省了开发人员对项目中公用类和公用工具方法的封装时间,使开发专注于业务,同时可以最大限度的避免封装不完善带来的bug。
使用Java如何更优雅的处理空值
在笔者几年的开发经验中,经常看到项目中存在到处空值判断的情况,这些判断,会让人觉得摸不着头绪,它的出现很有可能和当前的业务逻辑并没有关系。但它会让你很头疼。
有时候,更可怕的是系统因为这些空值的情况,会抛出空指针异常,导致业务系统发生问题。
此篇文章,我总结了几种关于空值的处理手法,希望对读者有帮助。
业务中的空值
场景
存在一个UserSearchService用来提供用户查询的功能:
publicinterfaceUserSearchService{
List<User>listUser();
Userget(Integerid);
}
问题现场
对于面向对象语言来讲,抽象层级特别的重要。尤其是对接口的抽象,它在设计和开发中占很大的比重,我们在开发时希望尽量面向接口编程。
对于以上描述的接口方法来看,大概可以推断出可能它包含了以下两个含义:
listUser():查询用户列表
get(Integerid):查询单个用户
在所有的开发中,XP推崇的TDD模式可以很好的引导我们对接口的定义,所以我们将TDD作为开发代码的”推动者”。
对于以上的接口,当我们使用TDD进行测试用例先行时,发现了潜在的问题:
listUser()如果没有数据,那它是返回空集合还是null呢?
get(Integerid)如果没有这个对象,是抛异常还是返回null呢?
深入listUser研究
我们先来讨论
listUser()
这个接口,我经常看到如下实现:
publicList<User>listUser(){
List<User>userList=userListRepostity.selectByExample(newUserExample());
if(CollectionUtils.isEmpty(userList)){//springutil工具类
returnnull;
}
returnuserList;
}
这段代码返回是null,从我多年的开发经验来讲,对于集合这样返回值,最好不要返回null,因为如果返回了null,会给调用者带来很多麻烦。你将会把这种调用风险交给调用者来控制。
如果调用者是一个谨慎的人,他会进行是否为null的条件判断。如果他并非谨慎,或者他是一个面向接口编程的狂热分子(当然,面向接口编程是正确的方向),他会按照自己的理解去调用接口,而不进行是否为null的条件判断,如果这样的话,是非常危险的,它很有可能出现空指针异常!
根据墨菲定律来判断:“很有可能出现的问题,在将来一定会出现!”
基于此,我们将它进行优化:
publicList<User>listUser(){
List<User>userList=userListRepostity.selectByExample(newUserExample());
if(CollectionUtils.isEmpty(userList)){
returnLists.newArrayList();//guava类库提供的方式
}
returnuserList;
}
对于接口(ListlistUser()),它一定会返回List,即使没有数据,它仍然会返回List(集合中没有任何元素);
通过以上的修改,我们成功的避免了有可能发生的空指针异常,这样的写法更安全!
深入研究get方法
对于接口
Userget(Integerid)
你能看到的现象是,我给出id,它一定会给我返回User.但事实真的很有可能不是这样的。
我看到过的实现:
publicUserget(Integerid){
returnuserRepository.selectByPrimaryKey(id);//从数据库中通过id直接获取实体对象
}
相信很多人也都会这样写。
通过代码的时候得知它的返回值很有可能是null!但我们通过的接口是分辨不出来的!
这个是个非常危险的事情。尤其对于调用者来说!
我给出的建议是,需要在接口明明时补充文档,比如对于异常的说明,使用注解@exception:
publicinterfaceUserSearchService{
/**
*根据用户id获取用户信息
*@paramid用户id
*@return用户实体
*@exceptionUserNotFoundException
*/
Userget(Integerid);
}
我们把接口定义加上了说明之后,调用者会看到,如果调用此接口,很有可能抛出“UserNotFoundException(找不到用户)”这样的异常。
这种方式可以在调用者调用接口的时候看到接口的定义,但是,这种方式是”弱提示”的!
如果调用者忽略了注释,有可能就对业务系统产生了风险,这个风险有可能导致一个亿!
除了以上这种”弱提示”的方式,还有一种方式是,返回值是有可能为空的。那要怎么办呢?
我认为我们需要增加一个接口,用来描述这种场景.
引入jdk8的Optional,或者使用guava的Optional.看如下定义:
publicinterfaceUserSearchService{
/**
*根据用户id获取用户信息
*@paramid用户id
*@return用户实体,此实体有可能是缺省值
*/
Optional<User>getOptional(Integerid);
}
Optional有两个含义:存在or缺省。
那么通过阅读接口getOptional(),我们可以很快的了解返回值的意图,这个其实是我们想看到的,它去除了二义性。
它的实现可以写成:
publicOptional<User>getOptional(Integerid){
returnOptional.ofNullable(userRepository.selectByPrimaryKey(id));
}
深入入参
通过上述的所有接口的描述,你能确定入参id一定是必传的吗?我觉得答案应该是:不能确定。除非接口的文档注释上加以说明。
那如何约束入参呢?
我给大家推荐两种方式:
强制约束
文档性约束(弱提示)
1.强制约束,我们可以通过jsr303进行严格的约束声明:
publicinterfaceUserSearchService{
/**
*根据用户id获取用户信息
*@paramid用户id
*@return用户实体
*@exceptionUserNotFoundException
*/
Userget(@NotNullIntegerid);
/**
*根据用户id获取用户信息
*@paramid用户id
*@return用户实体,此实体有可能是缺省值
*/
Optional<User>getOptional(@NotNullIntegerid);
}
当然,这样写,要配合AOP的操作进行验证,但让spring已经提供了很好的集成方案,在此我就不在赘述了。
2.文档性约束
在很多时候,我们会遇到遗留代码,对于遗留代码,整体性改造的可能性很小。
我们更希望通过阅读接口的实现,来进行接口的说明。
jsr305规范,给了我们一个描述接口入参的一个方式(需要引入库com.google.code.findbugs:jsr305):
可以使用注解:@Nullable@Nonnull@CheckForNull进行接口说明。比如:
publicinterfaceUserSearchService{
/**
*根据用户id获取用户信息
*@paramid用户id
*@return用户实体
*@exceptionUserNotFoundException
*/
@CheckForNull
Userget(@NonNullIntegerid);
/**
*根据用户id获取用户信息
*@paramid用户id
*@return用户实体,此实体有可能是缺省值
*/
Optional<User>getOptional(@NonNullIntegerid);
}
小结
通过空集合返回值,Optional,jsr303,jsr305这几种方式,可以让我们的代码可读性更强,出错率更低!
空集合返回值:如果有集合这样返回值时,除非真的有说服自己的理由,否则,一定要返回空集合,而不是null
Optional:如果你的代码是jdk8,就引入它!如果不是,则使用Guava的Optional,或者升级jdk版本!它很大程度的能增加了接口的可读性!
jsr303:如果新的项目正在开发,不防加上这个试试!一定有一种特别爽的感觉!
jsr305:如果老的项目在你的手上,你可以尝试的加上这种文档型注解,有助于你后期的重构,或者新功能增加了,对于老接口的理解!
空对象模式
场景
我们来看一个DTO转化的场景,对象:
@Data
staticclassPersonDTO{
privateStringdtoName;
privateStringdtoAge;
}
@Data
staticclassPerson{
privateStringname;
privateStringage;
}
需求是将Person对象转化成PersonDTO,然后进行返回。
当然对于实际操作来讲,返回如果Person为空,将返回null,但是PersonDTO是不能返回null的(尤其Rest接口返回的这种DTO)。
在这里,我们只关注转化操作,看如下代码:
@Test
publicvoidshouldConvertDTO(){
PersonDTOpersonDTO=newPersonDTO();
Personperson=newPerson();
if(!Objects.isNull(person)){
personDTO.setDtoAge(person.getAge());
personDTO.setDtoName(person.getName());
}else{
personDTO.setDtoAge("");
personDTO.setDtoName("");
}
}
优化修改
这样的数据转化,我们认识可读性非常差,每个字段的判断,如果是空就设置为空字符串(“”)
换一种思维方式进行思考,我们是拿到Person这个类的数据,然后进行赋值操作(setXXX),其实是不关系Person的具体实现是谁的。
那我们可以创建一个Person子类:
staticclassNullPersonextendsPerson{
@Override
publicStringgetAge(){
return"";
}
@Override
publicStringgetName(){
return"";
}
}
它作为Person的一种特例而存在,如果当Person为空的时候,则返回一些get*的默认行为.
所以代码可以修改为:
@Test
publicvoidshouldConvertDTO(){
PersonDTOpersonDTO=newPersonDTO();
Personperson=getPerson();
personDTO.setDtoAge(person.getAge());
personDTO.setDtoName(person.getName());
}
privatePersongetPerson(){
returnnewNullPerson();//如果Person是null,则返回空对象
}
其中getPerson()方法,可以用来根据业务逻辑获取Person有可能的对象(对当前例子来讲,如果Person不存在,返回Person的的特例NUllPerson),如果修改成这样,代码的可读性就会变的很强了。
使用Optional可以进行优化
空对象模式,它的弊端在于需要创建一个特例对象,但是如果特例的情况比较多,我们是不是需要创建多个特例对象呢,虽然我们也使用了面向对象的多态特性,但是,业务的复杂性如果真的让我们创建多个特例对象,我们还是要再三考虑一下这种模式,它可能会带来代码的复杂性。
对于上述代码,还可以使用Optional进行优化。
@Test
publicvoidshouldConvertDTO(){
PersonDTOpersonDTO=newPersonDTO();
Optional.ofNullable(getPerson()).ifPresent(person->{
personDTO.setDtoAge(person.getAge());
personDTO.setDtoName(person.getName());
});
}
privatePersongetPerson(){
returnnull;
}
Optional对空值的使用,我觉得更为贴切,它只适用于”是否存在”的场景。
如果只对控制的存在判断,我建议使用Optional.
Optioanl的正确使用
Optional如此强大,它表达了计算机最原始的特性(0or1),那它如何正确的被使用呢!
Optional不要作为参数
如果你写了一个public方法,这个方法规定了一些输入参数,这些参数中有一些是可以传入null的,那这时候是否可以使用Optional呢?
我给的建议是:一定不要这样使用!
举个例子:
publicinterfaceUserService{
List<User>listUser(Optional<String>username);
}
这个例子的方法listUser,可能在告诉我们需要根据username查询所有数据集合,如果username是空,也要返回所有的用户集合.
当我们看到这个方法的时候,会觉得有一些歧义:
“如果username是absent,是返回空集合吗?还是返回全部的用户数据集合?”
Optioanl是一种分支的判断,那我们究竟是关注Optional还是Optional.get()呢?
我给大家的建议是,如果不想要这样的歧义,就不要使用它!
如果你真的想表达两个含义,就給它拆分出两个接口:
publicinterfaceUserService{
List<User>listUser(Stringusername);
List<User>listUser();
}
我觉得这样的语义更强,并且更能满足软件设计原则中的“单一职责”。
如果你觉得你的入参真的有必要可能传null,那请使用jsr303或者jsr305进行说明和验证!
请记住!Optional不能作为入参的参数!
Optional作为返回值
当个实体的返回
那Optioanl可以做为返回值吗?
其实它是非常满足是否存在这个语义的。
你如说,你要根据id获取用户信息,这个用户有可能存在或者不存在。
你可以这样使用:
publicinterfaceUserService{
Optional<User>get(Integerid);
}
当调用这个方法的时候,调用者很清楚get方法返回的数据,有可能不存在,这样可以做一些更合理的判断,更好的防止空指针的错误!
当然,如果业务方真的需要根据id必须查询出User的话,就不要这样使用了,请说明,你要抛出的异常.
只有当考虑它返回null是合理的情况下,才进行Optional的返回
集合实体的返回
不是所有的返回值都可以这样用的!如果你返回的是集合:
publicinterfaceUserService{
Optional<List<User>>listUser();
}
这样的返回结果,会让调用者不知所措,是否我判断Optional之后,还用进行isEmpty的判断呢?
这样带来的返回值歧义!我认为是没有必要的。
我们要约定,对于List这种集合返回值,如果集合真的是null的,请返回空集合(Lists.newArrayList);
使用Optional变量
Optional<User>userOpt=...
如果有这样的变量userOpt,请记住:
一定不能直接使用get,如果这样用,就丧失了Optional本身的含义(比如userOp.get())
不要直接使用getOrThrow,如果你有这样的需求:获取不到就抛异常。那就要考虑,是否是调用的接口设计的是否合理
getter中的使用
对于一个javabean,所有的属性都有可能返回null,那是否需要改写所有的getter成为Optional类型呢?
我给大家的建议是,不要这样滥用Optional.
即便我javabean中的getter是符合Optional的,但是因为javabean太多了,这样会导致你的代码有50%以上进行Optinal的判断,这样便污染了代码。(我想说,其实你的实体中的字段应该都是由业务含义的,会认真的思考过它存在的价值的,不能因为Optional的存在而滥用)
我们应该更关注于业务,而不只是空值的判断。
请不要在getter中滥用Optional.
小结
可以这样总结Optional的使用:
当使用值为空的情况,并非源于错误时,可以使用Optional!
Optional不要用于集合操作!
不要滥用Optional,比如在javabean的getter中!
Java程序员最常用的技术有哪些
下面是我本人总结的工作中常用的!不仅仅是技术,还有工作中遇到的问题,使用的工具等!
技术:
多线程:synchonize加锁,forkjoin并行处理框架,java8的parelleStream并行流,线程池Executor,threadlocal本地线程!
设计模式:策略模式,适配器模式,修饰器模式,单例模式,工厂方法,建造者模式等等常用的!
微服务:有springcloud和dubbo两种最常用微服务框架,配合eureka,zuul,consul,hystrix等等对服务注册与发现,网关,服务熔断等!
消息队列:最常使用的有redis,kafka,rabbitMQ,activeMQ等,各种模型侧重点不同,性能也不尽相同!
缓存:redis(单线程排除并发干扰),memcache(轻量级,多线程),ehcache(常用做本地缓存)!
文件处理:File常用类,使用OSS工具上传,下载!POI生成pdf等类型文件!
spring:用做对象管理,springaop用做拦截器(日志,前置处理等),springmvc成熟的web开发框架,springboot微服务!
mybatis:数据库处理框架,轻量级,可编写动态SQL!
Druid:有着监控,统计等网页显示的数据库连接池框架!
json:轻量级数据传输!
restful:一种服务暴露约定!
slf4j:成熟的日志框架!
netty:编写高性能,高并发服务器模型!
shiro:安全认证框架!
工作流引擎:activiti等!
规则引擎:drools等!
工作处理常遇到的问题:
幂等处理:防止数据重复!
分布式锁:使用场景很多,比如消息重复消费等!
全局唯一id:使用redis,uuid,snowflake等!
死锁:参见我另一个回答!
工具&插件:
maven/gradle:项目构建工具!
svn/git:版本控制工具,git分布式!
jenkins:可视化持续集成工具,可方便自动部署服务!
docker:应用容器引擎,打包应用到环境中提供服务!
eclipse:最常用IDE,有svn,git,maven,findfugs,sonar等插件,方便开发!
postman:模拟各种形式的请求!
axsureRP:原型图查看工具!
navicate-mysql:mysql可视化操作工具!
还有很多东西,一时想不起,希望朋友们添加在评论里,做一个汇总,方便你我他!很多技术分享,敬请关注。。。
google guava和谷歌工具类java的问题分享结束啦,以上的文章解决了您的问题吗?欢迎您下次再来哦!