当前位置: 首页 > news >正文

网站建设 服务器 预算报价清单江苏建设工程建设网

网站建设 服务器 预算报价清单,江苏建设工程建设网,店名logo设计在线生成,亚洲最新永久在线观看一、前言 本文章所讲并没有基于Aspectj#xff0c;而是直接通过Cglib以及ProxyFactoryBean去创建代理Bean。通过下面的例子#xff0c;可以看出Cglib方式创建的代理Bean和ProxyFactoryBean创建的代理Bean的区别。 二、基本测试代码 测试实体类#xff0c;在BPP中创建BppTest… 一、前言   本文章所讲并没有基于Aspectj而是直接通过Cglib以及ProxyFactoryBean去创建代理Bean。通过下面的例子可以看出Cglib方式创建的代理Bean和ProxyFactoryBean创建的代理Bean的区别。 二、基本测试代码   测试实体类在BPP中创建BppTestDepBean类型的代理Bean。 Component public static class BppTestBean {Autowiredprivate BppTestDepBean depBean;public void test1() {depBean.testDep();}public void test2() {depBean.testDep();}TestMethodpublic void test3() {depBean.testDep();} }Component public static class BppTestDepBean {public void testDep() {System.out.println(HEHE);} }Target(ElementType.METHOD) Retention(RetentionPolicy.RUNTIME) Documented public interface TestMethod { }   测试类 RunWith(SpringRunner.class) SpringBootTest public class BppTest {Autowiredprivate BppTestBean bppTestBean;Testpublic void test() {bppTestBean.test1();bppTestBean.test2();bppTestBean.test3();} } 三、使用Cglib创建代理Bean public class ProxyBpp1 implements BeanPostProcessor {private static final Logger LOGGER LoggerFactory.getLogger(ProxyBpp1.class);Overridepublic Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {if (bean instanceof BppTestBean) {Enhancer enhancer new Enhancer();enhancer.setSuperclass(bean.getClass());//标识Spring-generated proxiesenhancer.setInterfaces(new Class[]{SpringProxy.class});//设置增强enhancer.setCallback((MethodInterceptor) (target, method, args, methodProxy) - {if (test1.equals(method.getName())) {LOGGER.info(ProxyBpp1 开始执行...);Object result methodProxy.invokeSuper(target, args);LOGGER.info(ProxyBpp1 结束执行...);return result;}return method.invoke(target, args);});return enhancer.create();}return bean;} }   主要是代理 BppTestBean的test1方法。其实这种方式创建的代理Bean使用问题的Autowired字段没有注入进来所以会有出现NPE。methodProxy.invokeSuper(target, args)这一行代码是有问题的targe是代理类对象而真实的对象是postProcessBeforeInitialization(Object bean, String beanName) 中的bean对象此时bean对象Autowired字段已经注入了。所以可以将methodProxy.invokeSuper(target, args) 修改为method.invoke(bean, args)解决无法注入Autowired字段的问题。 四、使用ProxyFactoryBean创建代理Bean public class ProxyBpp2 implements BeanPostProcessor {private static final Logger LOGGER LoggerFactory.getLogger(ProxyBpp2.class);Overridepublic Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {if (bean instanceof BppTestBean) {ProxyFactoryBean pfb new ProxyFactoryBean();pfb.setTarget(bean);pfb.setAutodetectInterfaces(false);NameMatchMethodPointcutAdvisor advisor new NameMatchMethodPointcutAdvisor();advisor.addMethodName(test1);advisor.setAdvice((MethodInterceptor) invocation - {LOGGER.info(ProxyBpp2 开始执行...);Object result invocation.getMethod().invoke(invocation.getThis(), invocation.getArguments());LOGGER.info(ProxyBpp2 结束执行...);return result;});pfb.addAdvisor(advisor);return pfb.getObject();}return bean;} }    使用ProxyFactoryBean创建代理Bean的时候一定要一个targe对象的。Advisor在切入的时候会逐个执行Advice。invocation.getThis()就是在通过ProxyFactoryBean创建代理Bean的时候传入的target对象。由于target对象就是postProcessBeforeInitialization(Object bean, String beanName) 中的bean对象所以Autowired字段也已经注入进来了。 五、Autowired注解何时被处理   想必大家都知道Autowired字段的处理也是通过一个BPP不过这个BPP比我们平常使用的要高级一些它就是InstantiationAwareBeanPostProcessor。这个BPP可以实现Bean的创建、属性的注入和解析比如Autowired、Value、Resource等等大家可以参考一下CommonAnnotationBeanPostProcessor处理JSR-250相关注解AutowiredAnnotationBeanPostProcessor处理Autowired、Value、Inject相关注解。   InstantiationAwareBeanPostProcessor中有一个如下的方法AutowiredAnnotationBeanPostProcessor就是覆盖这个方法实现了带有相关注解属性的自动注入。 Nullable default PropertyValues postProcessProperties(PropertyValues pvs, Object bean, String beanName)throws BeansException {return null; } Override public PropertyValues postProcessProperties(PropertyValues pvs, Object bean, String beanName) {InjectionMetadata metadata findAutowiringMetadata(beanName, bean.getClass(), pvs);try {metadata.inject(bean, beanName, pvs);}catch (BeanCreationException ex) {throw ex;}catch (Throwable ex) {throw new BeanCreationException(beanName, Injection of autowired dependencies failed, ex);}return pvs; }   InstantiationAwareBeanPostProcessor的postProcessProperties方法实在Spring AbstractAutowireCapableBeanFactory的populateBean方法中被调用。在AbstractAutowireCapableBeanFactory的doCreateBan中有如下代码。 // Initialize the bean instance. Object exposedObject bean;# try {populateBean(beanName, mbd, instanceWrapper);exposedObject initializeBean(beanName, exposedObject, mbd); }   也就是先进行了Bean的属性填充然后进行Bean的初始化工作。initializeBean方法中主要做了四件事。   1、invokeAwareMethods  2、applyBeanPostProcessorsBeforeInitialization  3、invokeInitMethods  4、applyBeanPostProcessorsAfterInitialization   其中2和4就是分别调用的普通的BPP中的postProcessBeforeInitialization方法和postProcessAfterInitialization方法。   这就是为什么在BPP中创建代理Bean的时候对应的目标Bean相关的Autowired字段已经注入的原因了。 六、InstantiationAwareBeanPostProcessor方式创建动态代理Bean   InstantiationAwareBeanPostProcessor接口中有个postProcessBeforeInstantiation方法可以让我们自己去实例化Bean。通过查看AbstractAutowireCapableBeanFactory方法调用createBean方法 - resolveBeforeInstantiation方法 - applyBeanPostProcessorsBeforeInstantiation方法 -InstantiationAwareBeanPostProcessor#postProcessBeforeInstantiation方法如果最终返回一个非null的实例那么就不会再执行doCreateBean方法。这就意味着不会有Bean属性的填充和初始化的流程了但是可以借助AbstractAutowireCapableBeanFactory帮助我们实现。 public T T postProcess(T object) {if (object null) {return null;}T result;try {// 使用容器autowireBeanFactory标准依赖注入方法autowireBean()处理 object对象的依赖注入this.autowireBeanFactory.autowireBean(object);// 使用容器autowireBeanFactory标准初始化方法initializeBean()初始化对象 objectresult (T) this.autowireBeanFactory.initializeBean(object,object.toString());} catch (RuntimeException e) {Class? type object.getClass();throw new RuntimeException(Could not postProcess object of type type, e);}return result; }   上图代码可以帮组我们实现非Spring容器Bean自动注入和初始化的功能。使用过Spring security同学都知道内部也是用了这个方式解决对象中的属性注入问题。如果你阅读了Spring security的源码你会发现很多对象比如WebSecurity、ProviderManager、各个安全Filter等这些对象的创建并不是通过bean定义的形式被容器发现和注册进入spring容器的而是直接new出来的。Spring security提供的AutowireBeanFactoryObjectPostProcessor这个工具类可以使这些对象具有容器bean同样的生命周期也能注入相应的依赖从而进入准备好被使用的状态。   使用Cglib在InstantiationAwareBeanPostProcessor 中创建动态代理Bean。 public class ProxyBpp3 implements InstantiationAwareBeanPostProcessor {private static final Logger LOGGER LoggerFactory.getLogger(ProxyBpp3.class);private final AutowireCapableBeanFactory autowireBeanFactory;ProxyBpp3(AutowireCapableBeanFactory autowireBeanFactory) {this.autowireBeanFactory autowireBeanFactory;}Overridepublic Object postProcessBeforeInstantiation(Class? beanClass, String beanName) throws BeansException {if (beanClass.equals(BppConfig.BppTestBean.class)) {Enhancer enhancer new Enhancer();enhancer.setSuperclass(beanClass);//标识Spring-generated proxiesenhancer.setInterfaces(new Class[]{SpringProxy.class});//设置增强enhancer.setCallback((MethodInterceptor) (target, method, args, methodProxy) - {if (test1.equals(method.getName())) {LOGGER.info(ProxyBpp3 开始执行...);Object result methodProxy.invokeSuper(target, args);LOGGER.info(ProxyBpp3 结束执行...);return result;}return methodProxy.invokeSuper(target, args);});return this.postProcess(enhancer.create());}return null;}... }   使用ProxyFactoryBean在InstantiationAwareBeanPostProcessor 中创建动态代理Bean。 public class ProxyBpp4 implements InstantiationAwareBeanPostProcessor {private static final Logger LOGGER LoggerFactory.getLogger(ProxyBpp4.class);private final AutowireCapableBeanFactory autowireBeanFactory;ProxyBpp4(AutowireCapableBeanFactory autowireBeanFactory) {this.autowireBeanFactory autowireBeanFactory;}Overridepublic Object postProcessBeforeInstantiation(Class? beanClass, String beanName) throws BeansException {if (beanClass.equals(BppConfig.BppTestBean.class)) {ProxyFactoryBean pfb new ProxyFactoryBean();pfb.setTarget(this.postProcess(BeanUtils.instantiateClass(beanClass)));pfb.setAutodetectInterfaces(false);NameMatchMethodPointcutAdvisor advisor new NameMatchMethodPointcutAdvisor();advisor.addMethodName(test1);advisor.setAdvice((MethodInterceptor) invocation - {LOGGER.info(ProxyBpp4 开始执行...);Object result invocation.getMethod().invoke(invocation.getThis(), invocation.getArguments());LOGGER.info(ProxyBpp4 结束执行...);return result;});pfb.addAdvisor(advisor);return pfb.getObject();}return null;}... }   上述向两种方式注意实例化bean后主动通过postProcess方法借助AbstractAutowireCapableBeanFactory完成对象相关属性的注入以及对象的初始化流程。 七、源码分享   点我查看源码如果有任何疑问请关注公众号后进行咨询。 转载于:https://www.cnblogs.com/hujunzheng/p/10463798.html
http://www.pierceye.com/news/124478/

相关文章:

  • 长安网站建设详细教程鸿科经纬教网店运营推广
  • 微信营销模式有seo短视频网页入口引流推广
  • 做商城网站简单吗长春网站建设服务
  • 工厂弄个网站做外贸如何app开发报价公司
  • 网销网站建设流程如何创建网站挣钱
  • 韶关网站制作手机推广app
  • Linux做视频网站网速均衡网页编辑实践报告
  • 做ppt好的模板下载网站如何查看网站空间商
  • 武义公司网站建设公司如何建设网站首页
  • hdwiki做网站罗湖网站建设联系电话
  • 深圳网站建设 利科技wordpress插件 手机版
  • 南通优普网站建设团队课程设计模板
  • 网站建设与维护的选择题浦东新区做网站
  • 做视频网站视频放在哪里网站备案目的
  • 建设部安全事故通报网站怎么更改网站的备案号
  • 重庆网站建设维护网络推广引流方法
  • 精品网站开发分销网站建站
  • 建设一个教程视频网站需要什么资质策划书案例范文
  • 郑州汉狮做网站的大公司海尔网站建设
  • 成都网站制作成都重庆网红景点排名
  • 广西南宁市网站制作公司制作图片的软件加字体
  • 新手搭建网站教程品牌推广费用预算
  • 广州网站设计网站制作竞价托管多少钱
  • 创建企业营销网站包括哪些内容软考高项彻底没用了
  • 企业品牌网站建设方案无锡网站设计多少钱
  • 轻量级网站开发在线旅游网站平台有哪些
  • 怎么用vs做网站推广优化网站排名
  • 免费推广网站软件常宁网站建设常宁网站建设
  • 冀州市网站建设html编辑器安卓版手机版软件
  • 广州专业网站改版方案网站建设要做ui和什么