人生靠反省,Java靠泛型
昨天有同事问 UserService、XxxService 都会调用 Dao 的 insert、update ... ...,这些重复的代码,有没有办法变得灵活一些?
巧了,和咱们分享的主题刚好碰上,卖个关子,先不谈解决方案,就当啥事没有发生,重新引入今天的话题(捂嘴笑)。
想蜕变的研发人员,偶尔会品味一下 Java 的源码;久经职场的码农,时不时也会搭建一下项目架构。其实无论你是刚入猿门,还是骨灰级战神,今天的分享你多多少少都听过、关注过、迷茫过甚至用过。
好了,准备好小板凳,让我们一起聊聊,在你看源码、搭架构过程中都躲避不开的 Java 中那些 E、T、?等字母都是啥意思?
先科普一下知识,什么是泛型?聊啥概念,直接上代码,直奔主题,先从 JDK 1.8 摘点源码出来,一起与泛型打个照面,混个脸熟。
1.
啥是 E?
E 可以说在 JDK 源码中无处不在,咱们就从 ArrayList 的源码进行聊起。
public class ArrayList<E> extends AbstractList<E>
implements List<E>, RandomAccess, Cloneable, java.io.Serializable { /**
* Appends the specified element to the end of this list.
*
* @param e element to be appended to this list
* @return <tt>true</tt> (as specified by {@link Collection#add})
*/
public boolean add(E e) {
ensureCapacityInternal(size + 1); // Increments modCount!!
elementData[size++] = e;
return true;
}
// 更多代码 ... ...
}
这个 E 在此处用来创建与初始 ArrayList 的类型,其实我们可以把它换成实际的类型,举个栗子:
发现编译器会把 E 被真正的类型所取代,其实也就是当我们创建出 ArrayList<String>,那么对应的 add() 会变成 add(String e);当我们创建出 ArrayList<Dog>,那么对应的 add() 会变成 add(Dog e)。
但是往往一不留神你会这么写,其实你也不想这么写,程序猿心里的苦,其实说也说不清楚(捂嘴笑)。
这就说明了为什么写代码的时候,老是经常编译不过去,老是有警告,谁让咱定义的是 String 类型,而咱们又非要往集合中放入一条狗呢。
来自灵魂的拷问:只能用“E”来表示吗?(身边同事还真问过我这个问题,在这我还是再认真的回答一次)
其实我们可以使用任何合法的 Java 标识字符串,但是大家用单一的字母来表示,已经成为一种习惯,而 E 又代表 Element 元素的意思,所以在集合中经常用 E 来表示,但是源码中大概率会碰到字母 T。
2.
啥是T?
public static <T extends Comparable<? super T>> void sort(List<T> list) {
list.sort(null);
}
上面代码摘自 JDK 1.8 Collections 的源码,我们发现了“T”,而且还有“?”,问号咱们暂且放一边,着重聊一聊“T”。
分两部分去看,其中第一部分 <T extends Comparable> 指的是 T 必须是 Comparable 的实现(这是重点、这是重点、这是重点);第二部分方法入参 sort(List<T> list),指的是仅能传入继承 Comparable 的参数化类型的 list。
其实也很难理解,不妨再摘个栗子配合理解。
从 JDK 1.8 源码中摘出 String 的源码,把 String 代入 Collections 的 sort 方法,替换为 T 尝试悟一下,看看是否 ok?!变量替换数学中大家都学过,就不深入了。
但是你实际开发中,有没有遇到过上图的情形,在进行狗狗列表排序时,就死活报错!报错!!原因就是因为要排序的狗狗,必须要实现 Comparable,方能进行排序。
好了,T 就不再聊了,咱们还是说说这个问号吧。
3.
“?”问号是啥?
问号,看到这个估计会一脸懵逼,其实就是未知,代表一万种可能性,在 Java 中就是万用字符。
那我们再看看上面摘自 JDK 1.8 Collections 的源码,那么 Comparable<? super T> 则代表 Comparable 的类型参数必须是 T 或 T 的父型,你可能有迷糊了,还是再抛点代码吧。
看到效果了没,因为要针对狗狗排序,排序的类型必须是 Dog 或者是 Dog 的父类型,咱们传入 String 类型,当然是编译不通过啦,不妨改成 Dog 或者 Object 自己试一下,看看效果,在此不做演示。不过结论还是要说一下:Comparable<? super T> 则代表 Comparable 的类型参数必须是 T 或 T 的父类型。
提到 <? super T>,那不得不再提一提 <? extends T>,其实只要上面的搞明白了,这个也就非常清晰了,问号代表继承和实现 T,栗子就不抛了,在框架源码中遇到知道是啥意思就行了。
4.
任性的总结。
其实泛型是编译期的一种检查,能够有效防止狗入人海,其中主要分为使用泛型的类以及使用泛型的方法;其中 E 主要用于集合的元素,除了 E 之外绝大部分是 T,然后 Java 还引入了一种万用字符是问号,不过可以用任意 Java 有效标识符进行表示,不要再纠结、不要再纠结、不要再纠结。
说了这么多,咱们开篇的问题还没有解决啊?话不多说,直接抛代码,不懂也没关系,注意理解上面几个字母就行了,下面这段代码分享给需要的朋友(哎呦我去,又出来个字母 D)。
@Transactional(readOnly = true)
public abstract class CrudService<D extends CrudDao<T>, T extends DataEntity<T>> extends BaseService { /**
* 持久层对象
*/
@Autowired
protected D dao; /**
* 获取单条数据
* @param entity
* @return
*/
public T get(T entity) {
return dao.get(entity);
} /**
* 查询列表数据
* @param entity
* @return
*/
public List<T> findList(T entity) {
return dao.findList(entity);
} /**
* 查询分页数据
* @param page 分页对象
* @param entity
* @return
*/
public Page<T> findPage(Page<T> page, T entity) {
entity.setPage(page);
page.setList(dao.findList(entity));
return page;
} /**
* 保存数据(插入或更新)
* @param entity
*/
@Transactional(readOnly = false)
public int save(T entity) {
if (entity.getIsNewRecord()){
entity.preInsert();
return dao.insert(entity);
}else{
entity.preUpdate();
return dao.update(entity);
}
} /**
* 删除数据
* @param entity
*/
@Transactional(readOnly = false)
public int delete(T entity) {
return dao.delete(entity);
}
}
5.
好了,今天分享就到这儿吧,希望能够解你所惑;希望能在你前进的道路上,帮你披荆斩棘。如果感觉有点帮助,欢迎在看、秒赞,疯狂分享转发,因为你的每一次分享,我都认真当成了鼓励与喜欢。
人生靠反省,Java靠泛型的更多相关文章
- [改善Java代码]Java的泛型是类型擦除的
泛型可以减少强制类型的转换,可规范集合的元素类型,还可以提高代码的安全性和可读性,正是因为有了这些优点,自从Java引入泛型之后,项目的编码规则上便多了一条,优先使用泛型. Java泛型(Generi ...
- Java 中泛型的全面解析(转)
Java泛型(generics) 是JDK 5中引入的一个新特性,允许在定义类和接口的时候使用类型参数(type parameter).声明的类型参数在使用时用具体的类型来替换.泛型最主要的应用是在J ...
- Java中泛型 类型擦除
转自:Java中泛型是类型擦除的 Java 泛型(Generic)的引入加强了参数类型的安全性,减少了类型的转换,但有一点需要注意:Java 的泛型在编译器有效,在运行期被删除,也就是说所有泛型参数类 ...
- Java 泛型 Java使用泛型的意义
Java 泛型 Java使用泛型的意义 @author ixenos 直接意义 在编译时保证类型安全 根本意义 a) 类型安全问题源自可复用性代码的设计,泛型保证了类型安全的复用模板 b) 使用复用性 ...
- 跟着刚哥梳理java知识点——泛型(十三)
一. 泛型概念的提出(为什么需要泛型)? 首先,我们看下下面这段简短的代码: public class GenericTest { public static void main(String[] a ...
- 【Java】泛型学习笔记
参考书籍 <Java核心技术:卷1> 泛型, 先睹为快 先通过一个简单的例子说明下Java中泛型的用法: 泛型的基本形式类似于模板, 通过一个类型参数T, 你可以"私人定制&qu ...
- [转] Java 的泛型擦除和运行时泛型信息获取
原文链接 https://my.oschina.net/lifany/blog/875769 前言 现在很多程序员都会在简历中写上精通 Java.但究竟怎样才算是精通 Java 呢?我觉得不仅要熟练掌 ...
- Java 容器 & 泛型:五、HashMap 和 TreeMap的自白
Writer:BYSocket(泥沙砖瓦浆木匠) 微博:BYSocket 豆瓣:BYSocket Java 容器的文章这次应该是最后一篇了:Java 容器 系列. 今天泥瓦匠聊下 Maps. 一.Ma ...
- 【译】9. Java反射——泛型
原文地址:http://tutorials.jenkov.com/java-reflection/generics.html ===================================== ...
- Java“禁止”泛型数组
Java“禁止”泛型数组 原文:https://blog.csdn.net/yi_Afly/article/details/52058708 1. 泛型定义泛型编程是一种通过参数化的方式将数据处理与数 ...
随机推荐
- 读《Java并发编程的艺术》学习笔记(二)
第2章 Java并发机制的底层实现原理 Java代码在编译后变成字节码,字节码被类加载器加载到JVM中,JVM执行字节码,最终转换为汇编指令在CPU上执行,Java中所使用的并发机制依赖于JVM的实 ...
- Redis系列三 - 缓存雪崩、击穿、穿透
前言 从学校出来,做开发工作也有一定时间了,最近有想系统地进一步深入学习,但发现基础知识不够扎实,故此来回顾基础知识,进一步巩固.加深印象. 最初开始接触编程时,总是自己跌跌撞撞.不断摸索地去学习,再 ...
- 我的学习笔记之node----node.js+socket.io实时聊天(1)
本想着从hello word开篇,也确实写了相关学习笔记.各种原因吧,现在又着急写出作品,便作罢. 这里将记录一个node.js+socket.io的实时聊天程序.(当然我也是跟着网上各种教程资料学习 ...
- JavaScript的自调用函数
函数表达式可以 "自调用". 自调用表达式会自动调用. 如果表达式后面紧跟 () ,则会自动调用. 不能自调用声明的函数. 通过添加括号,来说明它是一个函数表达式: <scr ...
- 在Java中使用Collections.sort 依据多个字段排序
一.如何使用Collections工具类进行排序 使用Collections工具类进行排序主要有两种方式: 1.对象实现Comparable接口,重写compareTo方法 /** * @author ...
- vue中的$props、$attrs和$listeners研究 [包装iview组件]
$props:当前组件接收到的 props 对象.Vue 实例代理了对其 props 对象属性的访问. $attrs:包含了父作用域中不作为 prop 被识别 (且获取) 的特性绑定 (class 和 ...
- 如何使用域名访问自己的Windows服务器(Java web 项目)
如何使用域名访问自己的Windows服务器(Java web 项目) 写在前面 前段时间在阿里云弄了个学生服务器,就想着自己搭建一个网站试一试,在网上查阅相关资料时发现大部分都是基于服务器是Linux ...
- C语言-转义字符
%d 十进制有符号整数 %u 十进制无符号整数 %f 浮点数 %s 字符串 %c 单个字符 %p 指针的值 %e 指数形式的浮点数 %X 无符号以十六进制表示的整数 ...
- angular vue通过node启动项目局域网内关闭防火墙无法访问的解决办法
先试 ng serve --host 0.0.0.0 不行再试 ng serve --host 0.0.0.0 --disable-host-check
- 「 Offer收割机之JVM」:生存还是毁灭
这两天,广州的天气又开始热了起来,又到了小动物交配的季节,啊呸,又到了一个收割 offer 的季节.年底将至,又到了面试的高峰期,JVM 作为Java 程序员面试绕不过的一道坎儿,它又来了,你准备好了 ...