国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|JavaScript|易語(yǔ)言|

服務(wù)器之家 - 編程語(yǔ)言 - Java教程 - 創(chuàng)建動(dòng)態(tài)代理對(duì)象bean,并動(dòng)態(tài)注入到spring容器中的操作

創(chuàng)建動(dòng)態(tài)代理對(duì)象bean,并動(dòng)態(tài)注入到spring容器中的操作

2021-08-10 10:54lichuangcsdn Java教程

這篇文章主要介紹了創(chuàng)建動(dòng)態(tài)代理對(duì)象bean,并動(dòng)態(tài)注入到spring容器中的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧

使用過(guò)Mybatis的同學(xué),應(yīng)該都知道,我們只需要編寫mybatis對(duì)應(yīng)的接口和mapper XML文件即可,并不需要手動(dòng)編寫mapper接口的實(shí)現(xiàn)。這里mybatis就用到了JDK動(dòng)態(tài)代理,并且將生成的接口代理對(duì)象動(dòng)態(tài)注入到Spring容器中。

這里涉及到幾個(gè)問(wèn)題。也許有同學(xué)會(huì)有疑問(wèn),我們直接編寫好類,加入@Component等注解不是可以注入了嗎?或者在配置類(@Configuration)中直接聲明該Bean類型不也可以注入嗎?

但具體到mybatis,這里我們用的是接口。由于spring實(shí)例化對(duì)象時(shí),如果沒有特殊情況,默認(rèn)都是通過(guò)反射形式來(lái)實(shí)例化Bean。而接口是無(wú)法直接通過(guò)Class.newInstance()方式進(jìn)行實(shí)例化的。

第二個(gè)問(wèn)題,如果手動(dòng)聲明Bean,其實(shí)也可以。但是會(huì)比較麻煩。因?yàn)槲覀冞€要手動(dòng)創(chuàng)建代理對(duì)象,可能還需要給該對(duì)象的屬性,比如(sqlSessionFactory,dataSource)設(shè)置對(duì)應(yīng)的Bean實(shí)例。這些都會(huì)比較麻煩。況且Mapper接口可能會(huì)有很多個(gè)。

下面,我也寫一個(gè)簡(jiǎn)單例子。用于說(shuō)明如何將動(dòng)態(tài)代理生成的接口實(shí)例,動(dòng)態(tài)的注入到Spring容器中,并且能正常調(diào)用這2個(gè)接口里面的方法,獲取調(diào)用結(jié)果。

解釋下這里為什么說(shuō)是動(dòng)態(tài)注入?因?yàn)槲覀兪孪炔⒉恢罆?huì)有多少個(gè)這樣的Bean,可以通過(guò)指定包路徑,來(lái)掃描特定路徑下的Bean。

整個(gè)代碼結(jié)構(gòu)如下:

創(chuàng)建動(dòng)態(tài)代理對(duì)象bean,并動(dòng)態(tài)注入到spring容器中的操作

如圖所示,我有2個(gè)接口CalculateService和TestService,這2個(gè)接口并沒有對(duì)應(yīng)的實(shí)現(xiàn)類。現(xiàn)在我們通過(guò)動(dòng)態(tài)代理生成實(shí)例,然后注入到TestController中

首先是創(chuàng)建一個(gè)SpringBoot maven工程

TestController源碼

  1. package com.company.controller;
  2. import com.company.service.CalculateService;
  3. import com.company.service.TestService;
  4. import org.springframework.beans.factory.annotation.Autowired;
  5. import org.springframework.web.bind.annotation.RequestMapping;
  6. import org.springframework.web.bind.annotation.RestController;
  7. @RestController
  8. public class TestController {
  9.  
  10. @Autowired
  11. private TestService testService;
  12.  
  13. @Autowired
  14. private CalculateService calculateService;
  15.  
  16. @RequestMapping("/test")
  17. public String getHello() {
  18. String testList = testService.getList("code123","name456");
  19. String calculateResult = calculateService.getResult("測(cè)試");
  20. return (testList + "," +calculateResult);
  21. }
  22. }

handler包下的ServiceBeanDefinitionRegistry源碼:

  1. package com.company.handler;
  2. import org.springframework.beans.BeansException;
  3. import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
  4. import org.springframework.beans.factory.support.BeanDefinitionBuilder;
  5. import org.springframework.beans.factory.support.BeanDefinitionRegistry;
  6. import org.springframework.beans.factory.support.BeanDefinitionRegistryPostProcessor;
  7. import org.springframework.beans.factory.support.GenericBeanDefinition;
  8. import org.springframework.context.ApplicationContext;
  9. import org.springframework.context.ApplicationContextAware;
  10. import org.springframework.context.ResourceLoaderAware;
  11. import org.springframework.core.env.Environment;
  12. import org.springframework.core.io.Resource;
  13. import org.springframework.core.io.ResourceLoader;
  14. import org.springframework.core.io.support.ResourcePatternResolver;
  15. import org.springframework.core.io.support.ResourcePatternUtils;
  16. import org.springframework.core.type.classreading.CachingMetadataReaderFactory;
  17. import org.springframework.core.type.classreading.MetadataReader;
  18. import org.springframework.core.type.classreading.MetadataReaderFactory;
  19. import org.springframework.stereotype.Component;
  20. import org.springframework.util.ClassUtils;
  21. import java.io.IOException;
  22. import java.util.LinkedHashSet;
  23. import java.util.Set;
  24.  
  25. /**
  26. * 用于Spring動(dòng)態(tài)注入自定義接口
  27. * @author lichuang
  28. */
  29. @Component
  30. public class ServiceBeanDefinitionRegistry implements BeanDefinitionRegistryPostProcessor,ResourceLoaderAware,ApplicationContextAware {
  31. @Override
  32. public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {
  33. //這里一般我們是通過(guò)反射獲取需要代理的接口的clazz列表
  34. //比如判斷包下面的類,或者通過(guò)某注解標(biāo)注的類等等
  35. Set<Class<?>> beanClazzs = scannerPackages("com.company.service");
  36. for (Class beanClazz : beanClazzs) {
  37. BeanDefinitionBuilder builder = BeanDefinitionBuilder.genericBeanDefinition(beanClazz);
  38. GenericBeanDefinition definition = (GenericBeanDefinition) builder.getRawBeanDefinition();
  39.  
  40. //在這里,我們可以給該對(duì)象的屬性注入對(duì)應(yīng)的實(shí)例。
  41. //比如mybatis,就在這里注入了dataSource和sqlSessionFactory,
  42. // 注意,如果采用definition.getPropertyValues()方式的話,
  43. // 類似definition.getPropertyValues().add("interfaceType", beanClazz);
  44. // 則要求在FactoryBean(本應(yīng)用中即ServiceFactory)提供setter方法,否則會(huì)注入失敗
  45. // 如果采用definition.getConstructorArgumentValues(),
  46. // 則FactoryBean中需要提供包含該屬性的構(gòu)造方法,否則會(huì)注入失敗
  47. definition.getConstructorArgumentValues().addGenericArgumentValue(beanClazz);
  48.  
  49. //注意,這里的BeanClass是生成Bean實(shí)例的工廠,不是Bean本身。
  50. // FactoryBean是一種特殊的Bean,其返回的對(duì)象不是指定類的一個(gè)實(shí)例,
  51. // 其返回的是該工廠Bean的getObject方法所返回的對(duì)象。
  52. definition.setBeanClass(ServiceFactory.class);
  53.  
  54. //這里采用的是byType方式注入,類似的還有byName等
  55. definition.setAutowireMode(GenericBeanDefinition.AUTOWIRE_BY_TYPE);
  56. registry.registerBeanDefinition(beanClazz.getSimpleName(), definition);
  57. }
  58. }
  59.  
  60. private static final String DEFAULT_RESOURCE_PATTERN = "**/*.class";
  61. private MetadataReaderFactory metadataReaderFactory;
  62.  
  63. /**
  64. * 根據(jù)包路徑獲取包及子包下的所有類
  65. * @param basePackage basePackage
  66. * @return Set<Class<?>> Set<Class<?>>
  67. */
  68. private Set<Class<?>> scannerPackages(String basePackage) {
  69. Set<Class<?>> set = new LinkedHashSet<>();
  70. String packageSearchPath = ResourcePatternResolver.CLASSPATH_ALL_URL_PREFIX +
  71. resolveBasePackage(basePackage) + '/' + DEFAULT_RESOURCE_PATTERN;
  72. try {
  73. Resource[] resources = this.resourcePatternResolver.getResources(packageSearchPath);
  74. for (Resource resource : resources) {
  75. if (resource.isReadable()) {
  76. MetadataReader metadataReader = this.metadataReaderFactory.getMetadataReader(resource);
  77. String className = metadataReader.getClassMetadata().getClassName();
  78. Class<?> clazz;
  79. try {
  80. clazz = Class.forName(className);
  81. set.add(clazz);
  82. } catch (ClassNotFoundException e) {
  83. e.printStackTrace();
  84. }
  85. }
  86. }
  87. } catch (IOException e) {
  88. e.printStackTrace();
  89. }
  90. return set;
  91. }
  92.  
  93. protected String resolveBasePackage(String basePackage) {
  94. return ClassUtils.convertClassNameToResourcePath(this.getEnvironment().resolveRequiredPlaceholders(basePackage));
  95. }
  96.  
  97. @Override
  98. public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
  99.  
  100. }
  101.  
  102. private ResourcePatternResolver resourcePatternResolver;
  103. private ApplicationContext applicationContext;
  104.  
  105. @Override
  106. public void setResourceLoader(ResourceLoader resourceLoader) {
  107. this.resourcePatternResolver = ResourcePatternUtils.getResourcePatternResolver(resourceLoader);
  108. this.metadataReaderFactory = new CachingMetadataReaderFactory(resourceLoader);
  109. }
  110.  
  111. @Override
  112. public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
  113. this.applicationContext = applicationContext;
  114. }
  115.  
  116. private Environment getEnvironment() {
  117. return applicationContext.getEnvironment();
  118. }
  119. }

ServiceFactory源碼:

  1. package com.company.handler;
  2. import org.springframework.beans.factory.FactoryBean;
  3. import java.lang.reflect.InvocationHandler;
  4. import java.lang.reflect.Proxy;
  5.  
  6. /**
  7. * 接口實(shí)例工廠,這里主要是用于提供接口的實(shí)例對(duì)象
  8. * @author lichuang
  9. * @param <T>
  10. */
  11. public class ServiceFactory<T> implements FactoryBean<T> {
  12. private Class<T> interfaceType;
  13. public ServiceFactory(Class<T> interfaceType) {
  14. this.interfaceType = interfaceType;
  15. }
  16.  
  17. @Override
  18. public T getObject() throws Exception {
  19. //這里主要是創(chuàng)建接口對(duì)應(yīng)的實(shí)例,便于注入到spring容器中
  20. InvocationHandler handler = new ServiceProxy<>(interfaceType);
  21. return (T) Proxy.newProxyInstance(interfaceType.getClassLoader(),
  22. new Class[] {interfaceType},handler);
  23. }
  24.  
  25. @Override
  26. public Class<T> getObjectType() {
  27. return interfaceType;
  28. }
  29.  
  30. @Override
  31. public boolean isSingleton() {
  32. return true;
  33. }
  34. }

ServiceProxy源碼

  1. package com.company.handler;
  2. import com.alibaba.fastjson.JSON;
  3. import java.lang.reflect.InvocationHandler;
  4. import java.lang.reflect.Method;
  5. import java.util.Arrays;
  6.  
  7. /**
  8. * 動(dòng)態(tài)代理,需要注意的是,這里用到的是JDK自帶的動(dòng)態(tài)代理,代理對(duì)象只能是接口,不能是類
  9. * @author lichuang
  10. */
  11.  
  12. public class ServiceProxy<T> implements InvocationHandler {
  13. private Class<T> interfaceType;
  14. public ServiceProxy(Class<T> intefaceType) {
  15. this.interfaceType = interfaceType;
  16. }
  17.  
  18. @Override
  19. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  20. if (Object.class.equals(method.getDeclaringClass())) {
  21. return method.invoke(this,args);
  22. }
  23. System.out.println("調(diào)用前,參數(shù):{}" + args);
  24. //這里可以得到參數(shù)數(shù)組和方法等,可以通過(guò)反射,注解等,進(jìn)行結(jié)果集的處理
  25. //mybatis就是在這里獲取參數(shù)和相關(guān)注解,然后根據(jù)返回值類型,進(jìn)行結(jié)果集的轉(zhuǎn)換
  26. Object result = JSON.toJSONString(args);
  27. System.out.println("調(diào)用后,結(jié)果:{}" + result);
  28. return result;
  29. }
  30. }

另外2個(gè)接口源碼:

  1. package com.company.service;
  2. public interface CalculateService {
  3. String getResult(String name);
  4. }

TestService接口

  1. package com.company.service;
  2. public interface TestService {
  3. String getList(String code, String name);
  4. }

我們DEBUG運(yùn)行,可以看到程序正常運(yùn)行,兩個(gè)Service接口都已正常的注入到控制器中了,程序也能正常返回接口。

創(chuàng)建動(dòng)態(tài)代理對(duì)象bean,并動(dòng)態(tài)注入到spring容器中的操作

創(chuàng)建動(dòng)態(tài)代理對(duì)象bean,并動(dòng)態(tài)注入到spring容器中的操作

補(bǔ)充:Spring動(dòng)態(tài) 注入/刪除 Bean

我們通過(guò)getBean來(lái)獲得對(duì)象,但這些對(duì)象都是事先定義好的,我們有時(shí)候要在程序中動(dòng)態(tài)的加入對(duì)象.因?yàn)槿绻捎门渲梦募蛘咦⒔猓覀円尤雽?duì)象的話,還要重啟服務(wù),如果我們想要避免這一情況就得采用動(dòng)態(tài)處理bean,包括:動(dòng)態(tài)注入,動(dòng)態(tài)刪除。

1 動(dòng)態(tài)注入bean思路

在具體進(jìn)行代碼實(shí)現(xiàn)的時(shí)候,我們要知道,Spring管理bean的對(duì)象是BeanFactory,具體的是DefaultListableBeanFactory,在這個(gè)類當(dāng)中有一個(gè)注入bean的方法:registerBeanDefinition,在調(diào)用registerBeanDefinition方法時(shí),需要BeanDefinition參數(shù),那么這個(gè)參數(shù)怎么獲取呢?

Spring提供了BeanDefinitionBuilder可以構(gòu)建一個(gè)BeanDefinition,那么我們的問(wèn)題就是如何獲取BeanFactory了,這個(gè)就很簡(jiǎn)單了,只要獲取到ApplicationContext對(duì)象即可獲取到BeanFacory了。

2. 動(dòng)態(tài)注入實(shí)現(xiàn)代碼

綜上所述,如果我們要編寫一個(gè)簡(jiǎn)單里的例子的話,那么分以個(gè)幾個(gè)步驟進(jìn)行編碼即可進(jìn)行動(dòng)態(tài)注入了:

1、獲取ApplicationContext;

2、通過(guò)ApplicationContext獲取到BeanFacotory;

3、通過(guò)BeanDefinitionBuilder構(gòu)建BeanDefiniton;

4、調(diào)用beanFactory的registerBeanDefinition注入beanDefinition;

5、使用ApplicationContext.getBean獲取bean進(jìn)行測(cè)試;

我們需要先定義個(gè)類進(jìn)行測(cè)試,比如TestService代碼如下:

  1. package com.kfit.demo.service;
  2. public class TestService {
  3. private String name;
  4. public String getName() {
  5. return name;
  6. }
  7. public void setName(String name) {
  8. this.name = name;
  9. }
  10. public void print(){
  11. System.out.println("動(dòng)態(tài)載入bean,name="+name);
  12. }
  13. }

那么下面我們的目標(biāo)就是動(dòng)態(tài)注入TestService了,根據(jù)以上的分析,我們進(jìn)行編碼,具體代碼如下:

  1. //獲取context.
  2. ApplicationContext ctx = (ApplicationContext) SpringApplication.run(App.class, args);
  3.  
  4. //獲取BeanFactory
  5. DefaultListableBeanFactory defaultListableBeanFactory = (DefaultListableBeanFactory)ctx.getAutowireCapableBeanFactory();
  6.  
  7. //創(chuàng)建bean信息
  8. BeanDefinitionBuilderbeanDefinitionBuilder =BeanDefinitionBuilder.genericBeanDefinition(TestService.class);
  9. beanDefinitionBuilder.addPropertyValue("name","張三");
  10.  
  11. //動(dòng)態(tài)注冊(cè)bean
  12. defaultListableBeanFactory.registerBeanDefinition("testService",beanDefinitionBuilder.getBeanDefinition());
  13.  
  14. //獲取動(dòng)態(tài)注冊(cè)的bean
  15. TestService testService =ctx.getBean(TestService.class);
  16. testService.print();

執(zhí)行代碼

  1. 動(dòng)態(tài)載入bean,name=張三

到這里,就證明我們的代碼很成功了。

3 多次注入同一個(gè)bean的情況

多次注入同一個(gè)bean的,如果beanName不一樣的話,那么會(huì)產(chǎn)生兩個(gè)Bean;如果beanName一樣的話,后面注入的會(huì)覆蓋前面的。

第一種情況:beanName一樣的代碼:

  1. beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(TestService.class);
  2. beanDefinitionBuilder.addPropertyValue("name","李四");
  3. defaultListableBeanFactory.registerBeanDefinition("testService", beanDefinitionBuilder.getBeanDefinition());

運(yùn)行看控制臺(tái):

  1. 動(dòng)態(tài)載入bean,name=李四

第二種情況:beanName不一樣的代碼:

  1. beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(TestService.class);
  2. beanDefinitionBuilder.addPropertyValue("name","李四");
  3. defaultListableBeanFactory.registerBeanDefinition("testService1",beanDefinitionBuilder.getBeanDefinition());
  4. TestService testService =ctx.getBean(TestService.class);
  5. testService.print();

此時(shí)如果沒有更改別的代碼直接運(yùn)行的話,是會(huì)報(bào)如下錯(cuò)誤的:

Caused by: org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type [com.kfit.demo.service.TestService] is defined: expected single matching bean but found 2: testService1,testService

大體意思就是在按照 byType getBean的時(shí)候,找到了兩個(gè)bean,這時(shí)候就不知道要獲取哪個(gè)了,所以在獲取的時(shí)候,我們就要使用byName指定我們是要獲取的testService還是testService1,只需要修改一句代碼:

  1. TestService testService =ctx.getBean("testService");

一般重復(fù)注入一個(gè)新Bean的情況較少,多數(shù)情況都是講已有的Bean注入到容器中,

  1. applicationContext.getAutowireCapableBeanFactory().applyBeanPostProcessorsAfterInitialization(obj, obj.getClass().getName());
  2. beanFactory.registerSingleton(obj.getClass().getName(), obj);

第一行:讓obj完成Spring初始化過(guò)程中所有增強(qiáng)器檢驗(yàn),只是不重新創(chuàng)建obj,

第二行:將obj以單例的形式入駐到容器中,此時(shí)通過(guò)obj.getClass().getName()或obj.getClass()都可以拿到放入Spring容器的Bean。

4 動(dòng)態(tài)刪除

相對(duì)于動(dòng)態(tài)注入,動(dòng)態(tài)刪除就很簡(jiǎn)單了,直接奉上代碼:

  1. //刪除bean.
  2. defaultListableBeanFactory.removeBeanDefinition("testService");

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持我們。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。

原文鏈接:https://blog.csdn.net/lichuangcsdn/article/details/89694363

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 欧美男人的天堂 | 久久思久久 | 欧美日韩一区二区电影 | 三级网站在线播放 | 欧美精品一区在线 | 欧美怡红院视频一区二区三区 | 黄色毛片看看 | 欧美 日韩 国产 成人 在线 | 成人免费在线观看 | 久久久在线 | 亚洲男人第一天堂 | 日本电影中文字幕 | 欧美黄色网视频 | 北条麻妃一区二区三区中文字幕 | 一区二区三区中文字幕 | 色狠狠网 | 久久综合一 | 成人国产精品久久 | 亚洲成人自拍 | 乱视频在线观看 | 欧美三区 | 激情小视频| 午夜影院网站 | 国产精品久久久久久福利一牛影视 | 日本黄色a视频 | 国产黄色av | 永久黄网站色视频免费观看w | 看一级片 | 亚洲亚色 | 中文字幕在线免费 | 国产精品第十页 | 亚洲欧美另类久久久精品2019 | 韩日一区 | 国产成人免费在线 | 欧美一级片在线 | 久久精品99 | 欧美日韩成人在线观看 | 视频一区二区三区中文字幕 | 国产精品久久久久久久浪潮网站 | 日韩视频免费在线播放 | 任你躁久久久久久妇女av |