类spring ioc 泛型保留

什么是泛型擦除

Java并不会传递泛型类,举个直观的栗子:

@Component
public class BaseProvider<T>
{
public void doSomething()
{
Class<?> clazz = getClass();
System.out.println(clazz.getName()+" doSomething");
Type superclass = clazz.getGenericSuperclass();
System.out.println("superclass="+superclass);
}
}

这里doSomething尝试打印泛型类型,

@Component
public class TestService
{
@Autowired
private BaseProvider<User> userProvdier;
public void doSomething()
{
userProvdier.doSomething();
}
}

BaseProvider<User>泛型指定了User类,来个测试看看User是否能被获取到?

 @ComponentScan
public class Main
{
public static void main(String[] args)
{
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(Main.class);
TestService service = context.getBean(TestService.class);
System.out.println("service=" + service);
service.doSomething();
}
}

依赖脚本build.gradle

dependencies {
String springVersion="5.1.9.RELEASE"
implementation "org.springframework:spring-context:$springVersion"
}

运行可以看到结果是,spring ioc并不能注入获取泛型

service=TestService@a68df9
BaseProvider doSomething
superclass=class java.lang.Object

自定义IOC泛型注入

在解决spring泛型问题之前,先做一个简单的IOC泛型注入框架,来了解其原理

    TestService service = Context.getBean(TestService.class);
}

这里将自写一个Context类,理解为上下文也好,BeanFactory也好,其原理不过是创建管理对象的东西

public class Context
{
public static <T> T getBean(Class<T> clazz)
{
return createBean(clazz, null, null, null);
}
private static <T> T createBean(Class<T> clazz, Type type, Object target, Field source)
{
//...
}
//....
}

设计createBean创建bean对象,依赖于bean类型,泛型类型,上级调用对象,来源位置(字段或方法)。再一个缓存bean对象,基本功能:

private static final Map<String, Object> beanCache = new HashMap<>();
@SuppressWarnings("unchecked")
private static <T> T createBean(Class<T> clazz, Type type, Object target, Member source)
{
try
{
String beanName = getBeanName(clazz, type, target, source);
if (beanCache.containsKey(beanName))
{
return (T) beanCache.get(beanName);
}
if (type != null && type instanceof ParameterizedType)
{
//创建泛型对象代理类
}
Constructor<T> cts = clazz.getDeclaredConstructor();
T obj = cts.newInstance();// 创建object
Field[] fields = clazz.getDeclaredFields();
for (Field field : fields)
{
if (field.getAnnotation(Autowried.class) != null)
{
setField(field, obj, createBean(field.getType(), field.getGenericType(), obj, field));
}
}
beanCache.put(beanName, obj);
return obj;
} catch (Exception e)
{
e.printStackTrace();
}
return null;
}
private static String getBeanName(Class<?> clazz, Type type, Object target, Member source)
{
if (type != null && type instanceof ParameterizedType)
{
ParameterizedType pt = (ParameterizedType) type;
StringJoiner sb = new StringJoiner("_");
for (Type p : pt.getActualTypeArguments())
{
sb.add(p.getTypeName().replaceAll("\\.", "_"));
}
return clazz.getName() + "_$_" + sb.toString() + "_Proxy";
}
return clazz.getName();
}

getBeanName规定了beanname的生成规则,createBean中创建泛型代理的部分,这里使用javassist去生成动态代理类

  implementation 'org.javassist:javassist:3.25.0-GA'

生成泛型子类

ClassPool pool = ClassPool.getDefault();
ParameterizedType pt = (ParameterizedType) type;
CtClass subClass = pool.get(clazz.getName());
StringJoiner genericSignature=new StringJoiner(",","L"+getClassPath(subClass.getName())+"<",">;");
Type[] ptts = pt.getActualTypeArguments();
for(int i=0;i<ptts.length;i++) {
genericSignature.add("L"+getClassPath(ptts[i].getTypeName())+";");
}
CtClass proxyClass = pool.makeClass( beanName,subClass);
proxyClass.setGenericSignature( genericSignature.toString());
clazz = (Class<T>) proxyClass.toClass();

getClassPath替换classname为class路径

private static String getClassPath(String name)
{
return name.replaceAll("\\.", "/");
}

再来看看运行效果

service=TestService@5d6f64b1
BaseProvider_$_User_Proxy doSomething
superclass=BaseProvider<User>

泛型<User>能够获取出来

为什么要泛型注入

泛型注入的应用场景,Java获取泛型一直是个很头疼的是,如果能够轻松获取泛型,就能够减少大量的代码。举个栗子,写一个常用的数据库操作工具类,不能获取泛型情况下,就必须传入Class<T> beanClazz参数:

public class DbUtil<T>{
List<T> getAll(Class<T> beanClazz){
//....
}
}

思考

方法的泛型应该如何去获取?

预告

将在下篇文章中讲解如何在spring 中解决泛型问题

类spring ioc 泛型保留的更多相关文章

  1. Spring Ioc容器核心类继承图

    Spring IOC容器其实就是BeanFactory的实例,Spring中BeanFactory的类关系结构如下图: 从上图可以看出Beanfactory作为根接口又细化出三个二级接口,最后又有Co ...

  2. 深入Spring IOC源码之ResourceLoader

    在<深入Spring IOC源码之Resource>中已经详细介绍了Spring中Resource的抽象,Resource接口有很多实现类,我们当然可以使用各自的构造函数创建符合需求的Re ...

  3. Spring IOC之基于JAVA的配置

    基础内容:@Bean 和 @Configuration 在Spring中新的支持java配置的核心组件是 @Configuration注解的类和@Bean注解的方法. @Bean注解被用于表明一个方法 ...

  4. Spring IOC之基于注解的容器配置

    Spring配置中注解比XML更好吗?基于注解的配置的介绍提出的问题是否这种途径比XML更好.简单来说就是视情况而定. 长一点的答案是每一种方法都有自己的长处也不足,而且这个通常取决于开发者决定哪一种 ...

  5. Spring IOC之依赖

    一个标准的企业级应用不只有一个对象组成.即使是最简单的引用也会有个相互作用的对象以使最终呈现 在用户面前的是个连贯一致的引用. 1依赖注入 依赖注入(DI)是一个对象定义他们依赖的过程,也就是说他们一 ...

  6. Spring IOC(四)总结

    目录 1.spring容器中Bean生命周期 2.IOC容器中核心接口 3.IOC容器启动流程 4.IOC依赖注入流程 =============正文分割线================== Spr ...

  7. 框架源码系列六:Spring源码学习之Spring IOC源码学习

    Spring 源码学习过程: 一.搞明白IOC能做什么,是怎么做的  1. 搞明白IOC能做什么? IOC是用为用户创建.管理实例对象的.用户需要实例对象时只需要向IOC容器获取就行了,不用自己去创建 ...

  8. Spring IOC(四)总结升华篇

    本系列目录 Spring IOC(一)概览 Spring IOC(二)容器初始化 Spring IOC(三)依赖注入 Spring IOC(四)总结升华 =============正文分割线===== ...

  9. Spring IOC 低级容器解析

    1.IOC是什么 IOC-Inversion of Control,即"控制反转",不是什么技术,而是一种设计思想.在Java开发中,Ioc意味着将你设计好的对象交给容器控制,而不 ...

随机推荐

  1. 【题解】射击-C++

    Description 不难发现,豆豆能从很多事情中去思考数学,于是豆豆父母决定让他去练习射击,这是项需要集中注意力的运动,相信 能够让豆豆暂时脱离数学.学习射击的第一天就让豆豆产生 了浓厚的兴趣,射 ...

  2. POJ2533&&SP1799 The Bottom of a Graph(tarjan+缩点)

    POJ2553 SP1799 我们知道单独一个强连通分量中的所有点是满足题目要求的 但如果它连出去到了其他点那里,要么成为新的强连通分量,要么失去原有的符合题目要求的性质 所以只需tarjan缩点求出 ...

  3. 请问 imgbtn上怎样添加文字呢

    陈桂城(49868971) 2013/10/14 21:29:57 <imgbtn>文字</imgbtn>

  4. 《ElasticSearch6.x实战教程》之复杂搜索、Java客户端(下)

    第八章-复杂搜索 黑夜给了我黑色的眼睛,我却用它寻找光明. 经过了解简单的API和简单搜索,已经基本上能应付大部分的使用场景.可是非关系型数据库数据的文档数据往往又多又杂,各种各样冗余的字段,组成了一 ...

  5. pycharm编辑器配置(持续更新完善)

    谨记:pycharm仅是一款编辑器,不要太依赖 pycharm的提示,不然后期换了编辑器就不行了 python解释器安装.多版本共存等 去python官网下载安装,配置环境变量.多版本共存等问题请参见 ...

  6. 统计学习方法(李航)朴素贝叶斯python实现

    朴素贝叶斯法 首先训练朴素贝叶斯模型,对应算法4.1(1),分别计算先验概率及条件概率,分别存在字典priorP和condP中(初始化函数中定义).其中,计算一个向量各元素频率的操作反复出现,定义为c ...

  7. Git学习(二):Git的初步使用

    一.Git的最小配置 1.使用如下命令创建Git的用户名和邮箱,如下所示: $git config --global user.name 'your_name' $git config --globa ...

  8. Java Web项目案例之---登录和注册(精华版)

    登录和注册(精华版) (一)实现功能 1.使用cookie记录登录成功的用户名,用户选择记住用户名,则用户再次登录时用户名自动显示 2.实现文件上传功能(上传文件的表单上传于普通的表单上传不同,必须是 ...

  9. Vue 报错 listen EADDRINUSE :::8080

    今天在重启vue项目的时候,发现报了错, listen EADDRINUSE :::8080错误提示 原因:因为另一个项目占用了8080端口,我直接在命令行npm run dev第二个项目,就给出了这 ...

  10. export,export default,module.exports,import,require之间的区别和关联

    module.exports Node 应用由模块组成,采用 CommonJS 模块规范.根据这个规范,每个文件就是一个模块,有自己的作用域.在这些文件里面定义的变量.函数.类,都是私有的,对外不可见 ...