日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

spring aop底層原理及如何實現

瀏覽:30日期:2023-07-17 08:55:36
前言

相信每天工作都要用spring框架的大家一定使用過spring aop,aop的概念是面向切面編程,相對與傳統的面向對象編程oop,aop更關注的是橫向的邏輯,比如說一個大型系統中的日志記錄,異常處理,性能監控等等,都是各個模塊都需要的操作,那樣代表著這些操作會散落在系統的各個地方,不易管理且雜亂無章,而aop就是關注的這些,aop將這些操作與業務代碼分離,統一成一個個的切面,針對這些個切面進行編程處理。spring aop使得我們的aop開發工作變得簡單,這次我就給大家講講spring aop的底層原理和實現

使用

要分析spring aop的底層原理,首先要會使用,先創建一個普通maven webapp項目,引入spring-context依賴,版本為5.1.1RELEASE

<dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>5.1.1.RELEASE</version> </dependency>

然后我使用aspectj作為aop的語法實現,和spring整合起來

<dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.9.4</version> </dependency>

接下來我全稱用注解的形式來模擬spring aop的使用,先創建一個配置類去掃描包,開啟aspectJ的自動代理支持

@Configuration@EnableAspectJAutoProxy@ComponentScan('com.ww')public class Wconfig {}

然后新建一個接口和接口的實現類

public interface Dao { void query();}@Componentpublic class IndexDao implements Dao{ @Override public void query() {System.out.println('query......'); }}

創建切面

//代表是一個切面@Aspect@Componentpublic class WAspect { /** * execution表達式,可以百度寫法 */ @Pointcut('execution(* com.ww.dao.*.*(..))') public void point(){ } /** * 在切點上進行前置通知 */ @Before('point()') public void beforeAd(){System.out.println('before-------------'); }}

創建測試方法

public class TestAspect { public static void main(String[] args) {AnnotationConfigApplicationContext configApplicationContext = new AnnotationConfigApplicationContext(Wconfig.class);Dao dao = configApplicationContext.getBean(Dao.class);dao.query(); }}

執行方法,可以看到在打印query...之前打印了before----------

spring aop底層原理及如何實現

這個時候我們很想知道為什么這句before會打印在query之前呢,稍微對spring aop有所了解的人應該知道,spring是通過動態代理和字節碼技術來實現aop操作的,也就是經常說的jdk動態代理和cglib動態代理兩種模式,那么,spring究竟是怎么創建的代理對象,又是什么時候產生的代理對象呢,下面我們來一起探究一下源碼,來揭開這個謎底

源碼分析

首先我們透過現象看本質,我先把斷點打在測試方法的最后一行,我們來看這個時候的dao對象

spring aop底層原理及如何實現

那么接下來我們就要去找到什么時候這個dao對象變成了動態代理對象的,既然在最后一行的時候對象已經變成了代理對象,那么我門自然就猜想是在上一句代碼的位置spring執行了創建代理對象的操作,我們把斷點移到上一行,debug進去

spring aop底層原理及如何實現

再往下走

spring aop底層原理及如何實現

這行代碼我看方法名覺得應該是有用的代碼,方法意思應該是spring處理好的bean,跟進去看看

@Nullableprivate <T> T resolveBean(ResolvableType requiredType, @Nullable Object[] args, boolean nonUniqueAsNull) {//這行代碼最有用,處理有beanName的bean,debug進入看一下NamedBeanHolder<T> namedBean = resolveNamedBean(requiredType, args, nonUniqueAsNull);if (namedBean != null) {return namedBean.getBeanInstance();}BeanFactory parent = getParentBeanFactory();if (parent instanceof DefaultListableBeanFactory) {return ((DefaultListableBeanFactory) parent).resolveBean(requiredType, args, nonUniqueAsNull);}else if (parent != null) {ObjectProvider<T> parentProvider = parent.getBeanProvider(requiredType);if (args != null) {return parentProvider.getObject(args);}else {return (nonUniqueAsNull ? parentProvider.getIfUnique() : parentProvider.getIfAvailable());}}return null;}

@SuppressWarnings('unchecked')@Nullableprivate <T> NamedBeanHolder<T> resolveNamedBean(ResolvableType requiredType, @Nullable Object[] args, boolean nonUniqueAsNull) throws BeansException {Assert.notNull(requiredType, 'Required type must not be null');Class<?> clazz = requiredType.getRawClass();Assert.notNull(clazz, 'Required type must have a raw Class');//候選name列表,直覺告訴我這行代碼比較重要String[] candidateNames = getBeanNamesForType(requiredType);if (candidateNames.length > 1) {List<String> autowireCandidates = new ArrayList<>(candidateNames.length);for (String beanName : candidateNames) {if (!containsBeanDefinition(beanName) || getBeanDefinition(beanName).isAutowireCandidate()) {autowireCandidates.add(beanName);}}if (!autowireCandidates.isEmpty()) {candidateNames = StringUtils.toStringArray(autowireCandidates);}}//因為我的代碼里只有一個bean,所以我覺得應該會執行這里的代碼if (candidateNames.length == 1) {String beanName = candidateNames[0];return new NamedBeanHolder<>(beanName, (T) getBean(beanName, clazz, args));}else if (candidateNames.length > 1) {Map<String, Object> candidates = new LinkedHashMap<>(candidateNames.length);for (String beanName : candidateNames) {if (containsSingleton(beanName) && args == null) {Object beanInstance = getBean(beanName);candidates.put(beanName, (beanInstance instanceof NullBean ? null : beanInstance));}else {candidates.put(beanName, getType(beanName));}}String candidateName = determinePrimaryCandidate(candidates, clazz);if (candidateName == null) {candidateName = determineHighestPriorityCandidate(candidates, clazz);}if (candidateName != null) {Object beanInstance = candidates.get(candidateName);if (beanInstance == null || beanInstance instanceof Class) {beanInstance = getBean(candidateName, clazz, args);}return new NamedBeanHolder<>(candidateName, (T) beanInstance);}if (!nonUniqueAsNull) {throw new NoUniqueBeanDefinitionException(requiredType, candidates.keySet());}}return null;}

執行完getBeanNamesForType(requiredType)后,我們看idea的變量顯示,果然有一個bean,name是IndexDao

spring aop底層原理及如何實現

那么接下來自然會進到length==1的那個代碼塊,這時候我再debug進入,這里還是一個getBean方法

spring aop底層原理及如何實現

spring aop底層原理及如何實現

在spring容器中還有一些沒有name的其他的bean需要被創建,所以這里我用上了條件斷點,當beanName等于indexDao的時候,才會進入斷點,但是當我F8跑完這行代碼的時候,出乎意料的事情發生了

spring aop底層原理及如何實現

驚不驚喜,意不意外,getSingleton這行代碼執行結束之后,代理對象就已經被創建了,所以需要debug進入這行代碼去看

protected Object getSingleton(String beanName, boolean allowEarlyReference) {//spring所有的bean被放在ioc容器中的地方,就是這個singletonObjects,這是一個concorrentHashMap。Object singletonObject = this.singletonObjects.get(beanName);if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {synchronized (this.singletonObjects) {singletonObject = this.earlySingletonObjects.get(beanName);if (singletonObject == null && allowEarlyReference) {ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);if (singletonFactory != null) {singletonObject = singletonFactory.getObject();this.earlySingletonObjects.put(beanName, singletonObject);this.singletonFactories.remove(beanName);}}}}return singletonObject;}

但是我在這里只看到了get方法,那么這些bean是什么時候放到singletonObjects里的呢,我來找找

protected void addSingleton(String beanName, Object singletonObject) {synchronized (this.singletonObjects) {this.singletonObjects.put(beanName, singletonObject);this.singletonFactories.remove(beanName);this.earlySingletonObjects.remove(beanName);this.registeredSingletons.add(beanName);}}

在DefaultSingletonBeanRegistry注冊器中,我找到了singletonObjects.put方法,代表bean是這個時候被放到這個map里去的,接下來我在這行代碼上進行條件斷點,然后我們來看它的調用鏈,找出是什么時候執行的addSingleton方法,其實從這個時候我已經知道,斷點打在測試方法的倒數第二行是不對的,在getBean之前其實代理對象已經產生了

spring aop底層原理及如何實現

// Create bean instance.if (mbd.isSingleton()) {sharedInstance = getSingleton(beanName, () -> {try { //創建bean,核心代碼return createBean(beanName, mbd, args);}catch (BeansException ex) {// Explicitly remove instance from singleton cache: It might have been put there// eagerly by the creation process, to allow for circular reference resolution.// Also remove any beans that received a temporary reference to the bean.destroySingleton(beanName);throw ex;}});bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd);}

在createBean方法上,我也加上條件斷點,然后debug進入

@Overrideprotected Object createBean(String beanName, RootBeanDefinition mbd, @Nullable Object[] args)throws BeanCreationException {if (logger.isTraceEnabled()) {logger.trace('Creating instance of bean ’' + beanName + '’');}RootBeanDefinition mbdToUse = mbd;// Make sure bean class is actually resolved at this point, and// clone the bean definition in case of a dynamically resolved Class// which cannot be stored in the shared merged bean definition.Class<?> resolvedClass = resolveBeanClass(mbd, beanName);if (resolvedClass != null && !mbd.hasBeanClass() && mbd.getBeanClassName() != null) {mbdToUse = new RootBeanDefinition(mbd);mbdToUse.setBeanClass(resolvedClass);}// Prepare method overrides.try {mbdToUse.prepareMethodOverrides();}catch (BeanDefinitionValidationException ex) {throw new BeanDefinitionStoreException(mbdToUse.getResourceDescription(),beanName, 'Validation of method overrides failed', ex);}try {// Give BeanPostProcessors a chance to return a proxy instead of the target bean instance.Object bean = resolveBeforeInstantiation(beanName, mbdToUse);if (bean != null) {return bean;}}catch (Throwable ex) {throw new BeanCreationException(mbdToUse.getResourceDescription(), beanName,'BeanPostProcessor before instantiation of bean failed', ex);}try { //核心代碼,創建bean實例,需要斷點進入Object beanInstance = doCreateBean(beanName, mbdToUse, args);if (logger.isTraceEnabled()) {logger.trace('Finished creating instance of bean ’' + beanName + '’');}return beanInstance;}catch (BeanCreationException | ImplicitlyAppearedSingletonException ex) {// A previously detected exception with proper bean creation context already,// or illegal singleton state to be communicated up to DefaultSingletonBeanRegistry.throw ex;}catch (Throwable ex) {throw new BeanCreationException(mbdToUse.getResourceDescription(), beanName, 'Unexpected exception during bean creation', ex);}}

接下來我debug進入doCreateBean方法

debug跟進initializeBean方法,條件斷點在兩個初始化處理器上,我隱約覺得代理對象就是從這兩個方法中產生的,我們拭目以待

protected Object initializeBean(final String beanName, final Object bean, @Nullable RootBeanDefinition mbd) {if (System.getSecurityManager() != null) {AccessController.doPrivileged((PrivilegedAction<Object>) () -> {invokeAwareMethods(beanName, bean);return null;}, getAccessControlContext());}else {invokeAwareMethods(beanName, bean);}//包裝的beanObject wrappedBean = bean;if (mbd == null || !mbd.isSynthetic()) { //執行前置初始化beanPostProcessor處理器wrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName);}try {invokeInitMethods(beanName, wrappedBean, mbd);}catch (Throwable ex) {throw new BeanCreationException((mbd != null ? mbd.getResourceDescription() : null),beanName, 'Invocation of init method failed', ex);}if (mbd == null || !mbd.isSynthetic()) { //執行初始化后的beanPostProcessor處理器wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);}return wrappedBean;}

執行完applyBeanPostProcessorsBeforeInitialization方法,這個時候我們看到warppedBean還是indexDao,并沒有產生代理對象

spring aop底層原理及如何實現

我猜想在下一個后置處理器中,代理對象將被創建,我debug進去

public Object applyBeanPostProcessorsAfterInitialization(Object existingBean, String beanName)throws BeansException {Object result = existingBean;for (BeanPostProcessor processor : getBeanPostProcessors()) { //經過處理器處理后的bean,我先看看有多少個處理器Object current = processor.postProcessAfterInitialization(result, beanName);if (current == null) {return result;}result = current;}return result;}

spring aop底層原理及如何實現

可以看到我的猜想被證明是正確的,運行完這個后置處理器,代理對象就被創建出來了。 到了這里我們知道了代理對象是從哪里來的了,但是還是沒搞清楚代理對象是怎么創建出來的,這時候我們就需要debug進入到這個處理器內部去瞧瞧了。

@Overridepublic Object postProcessAfterInitialization(@Nullable Object bean, String beanName) {if (bean != null) { //獲取緩存key,不重要Object cacheKey = getCacheKey(bean.getClass(), beanName);if (!this.earlyProxyReferences.contains(cacheKey)) {//重要方法,需要debug進去看return wrapIfNecessary(bean, beanName, cacheKey);}}return bean;}

于是乎我又進到了wrapIfNecessary這個方法內部

protected Object wrapIfNecessary(Object bean, String beanName, Object cacheKey) {if (StringUtils.hasLength(beanName) && this.targetSourcedBeans.contains(beanName)) {return bean;}if (Boolean.FALSE.equals(this.advisedBeans.get(cacheKey))) {return bean;}if (isInfrastructureClass(bean.getClass()) || shouldSkip(bean.getClass(), beanName)) {this.advisedBeans.put(cacheKey, Boolean.FALSE);return bean;}// Create proxy if we have advice.//看到上面這行注釋,可以確定代理類就是從這里產生的,下面這個方法得到的都是一些包括切點信息,通知類型等等的信息Object[] specificInterceptors = getAdvicesAndAdvisorsForBean(bean.getClass(), beanName, null);if (specificInterceptors != DO_NOT_PROXY) {this.advisedBeans.put(cacheKey, Boolean.TRUE); //顧名思義,創建代理,bebug進入看看Object proxy = createProxy(bean.getClass(), beanName, specificInterceptors, new SingletonTargetSource(bean));this.proxyTypes.put(cacheKey, proxy.getClass());return proxy;}this.advisedBeans.put(cacheKey, Boolean.FALSE);return bean;}

protected Object createProxy(Class<?> beanClass, @Nullable String beanName,@Nullable Object[] specificInterceptors, TargetSource targetSource) {if (this.beanFactory instanceof ConfigurableListableBeanFactory) {AutoProxyUtils.exposeTargetClass((ConfigurableListableBeanFactory) this.beanFactory, beanName, beanClass);}ProxyFactory proxyFactory = new ProxyFactory();proxyFactory.copyFrom(this);if (!proxyFactory.isProxyTargetClass()) {if (shouldProxyTargetClass(beanClass, beanName)) {proxyFactory.setProxyTargetClass(true);}else {evaluateProxyInterfaces(beanClass, proxyFactory);}}Advisor[] advisors = buildAdvisors(beanName, specificInterceptors);proxyFactory.addAdvisors(advisors);proxyFactory.setTargetSource(targetSource);customizeProxyFactory(proxyFactory);proxyFactory.setFrozen(this.freezeProxy);if (advisorsPreFiltered()) {proxyFactory.setPreFiltered(true);}//重要的地方在這里,代理對象是通過這個方法生成的return proxyFactory.getProxy(getProxyClassLoader());} public Object getProxy(@Nullable ClassLoader classLoader) {//debug進去看看return createAopProxy().getProxy(classLoader);}

protected final synchronized AopProxy createAopProxy() {if (!this.active) {activate();}//同樣需要debug進入看看return getAopProxyFactory().createAopProxy(this);}

我們看到這里有一個if語句,當config中的isOptimize和isProxyTargetClass還有hasNoUserSuppliedProxyInterfaces三個判斷條件只要有一個滿足的話,spring就會選擇cglib的方式進行動態代理,而config中的兩個boolean變量的默認值都是false,而我們的被代理對象又是實現接口的,所以spring會選擇jdk動態代理的實現形式來完成動態代理,當然,我們也可以在這種情況下手動的配置config值來讓spring選擇cglib作為動態代理的實現方式,稍后我會演示

@Overridepublic AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {if (config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config)) {Class<?> targetClass = config.getTargetClass();if (targetClass == null) {throw new AopConfigException('TargetSource cannot determine target class: ' +'Either an interface or a target is required for proxy creation.');}if (targetClass.isInterface() || Proxy.isProxyClass(targetClass)) {return new JdkDynamicAopProxy(config);}return new ObjenesisCglibAopProxy(config);}else { //現在知道為什么我們的代理類是動態代理了嗎return new JdkDynamicAopProxy(config);}}

spring aop底層原理及如何實現

總結

我以spring aop實現的調用鏈圖來結束這次的總結

spring aop底層原理及如何實現

以上就是spring aop底層原理及如何實現的詳細內容,更多關于spring aop原理及實現的資料請關注好吧啦網其它相關文章!

標簽: Spring
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
亚洲精品小说| 亚洲精品系列| 精品女同一区二区三区在线观看| 欧美日本二区| 欧美亚洲二区| 国产欧美日本| 欧美aⅴ一区二区三区视频| 国产精品亲子伦av一区二区三区| 国产精品日韩精品中文字幕| 国产欧美88| 免费一区二区三区在线视频| 你懂的网址国产 欧美| 成人台湾亚洲精品一区二区| 国产高清不卡| 久久久久国产| 好看不卡的中文字幕| 99tv成人| 老司机精品久久| 欧美一区自拍| 麻豆久久久久久| 中文字幕高清在线播放| 久久在线免费| 亚洲精选久久| 国产精品99久久久久久董美香| 成人在线超碰| 国产综合精品| 亚洲丝袜啪啪| 精品亚洲成人| 亚洲天堂成人| 无码日韩精品一区二区免费| 久久不卡国产精品一区二区| 国产一区二区三区黄网站| 亚洲香蕉网站| 亚洲字幕久久| 麻豆国产欧美日韩综合精品二区| 在线观看精品| 蜜桃视频一区二区三区| 免费在线成人| 欧美日韩国产欧| 国产日韩亚洲欧美精品| 激情黄产视频在线免费观看| 1024精品久久久久久久久| 亚洲精品乱码| 国产91在线精品| 国产精品老牛| 精品三级av在线导航| 亚洲电影在线| 国产精品一区二区av交换| 久久久人人人| 日韩精品一区二区三区中文在线| 国内自拍视频一区二区三区| 日韩午夜高潮| 国模精品一区| 亚洲毛片视频| 亚洲一级少妇| 日本免费一区二区视频| 国产夫妻在线| 日韩中文字幕一区二区高清99| 国产一区二区三区久久久久久久久| 国产免费成人| 97精品视频在线看| 一区二区国产在线观看| 欧美日韩国产v| 日韩国产精品久久久| 日韩在线观看| 国产亚洲字幕| 性一交一乱一区二区洋洋av| 国产精品不卡| 久久国产欧美日韩精品| 国产日韩专区| 日韩啪啪电影网| 久久的色偷偷| 日韩国产成人精品| 图片区亚洲欧美小说区| 成人午夜在线| 欧美精品福利| 国产一级久久| 99久久亚洲精品| 久久精品日韩欧美| 日韩av在线免费观看不卡| 亚洲激情婷婷| 91精品蜜臀一区二区三区在线| 国产精品欧美大片| 亚洲综合小说| 国产精品毛片一区二区三区| 色一区二区三区| 国产欧美91| 日本一区二区三区视频在线看| 婷婷成人基地| 日韩国产网站| 国产va免费精品观看精品视频| 亚洲3区在线| 免费欧美在线视频| 精品在线播放| 日韩欧美另类一区二区| 黄色欧美在线| 久久久国产精品入口麻豆| 日韩高清一区在线| 男女精品网站| 亚洲一卡久久| 九九综合在线| 欧美搞黄网站| 999国产精品视频| 中文在线免费视频| 97精品国产一区二区三区| 精品资源在线| 国产乱子精品一区二区在线观看| 老鸭窝亚洲一区二区三区| 99日韩精品| 国产视频一区欧美| 黄色成人在线网址| 尤物网精品视频| 一区在线免费| 精品一区在线| 亚洲欧洲一区| 国产免费成人| 在线精品观看| 日韩在线观看中文字幕| 日日夜夜免费精品视频| 日韩一区二区三区精品视频第3页| 亚洲精品影视| 亚洲视频二区| 日本v片在线高清不卡在线观看| 中文字幕亚洲影视| 蜜桃av一区二区| 日韩精品1区2区3区| 国产亚洲一卡2卡3卡4卡新区| 欧美视频久久| 国产精品视频首页| 国产一区二区三区亚洲| 成人在线超碰| 天堂√8在线中文| 久久精品亚洲人成影院| 伊人精品在线| 日韩精品一级| 你懂的亚洲视频| 日本精品黄色| 久久久国产精品一区二区中文| 国产99精品| 日韩中文字幕不卡| 欧美一区网站| 成人影视亚洲图片在线| 激情欧美一区| 免费成人性网站| 久久国产乱子精品免费女| 九九久久国产| 久久激情婷婷| 久久不射网站| 日韩av三区| 日韩成人a**站| 亚洲精品1区| 日本一不卡视频| 精品国产鲁一鲁****| 999精品一区| 日本中文字幕视频一区| 老鸭窝一区二区久久精品| 日韩福利一区| 亚洲综合精品四区| 国产精久久一区二区| 日韩在线欧美| 亚洲精品护士| 欧美韩日一区| 视频一区二区三区在线| 国产精品啊v在线| 成人看片网站| 亚洲三级国产| 激情久久一区二区| 亚洲一区黄色| 欧美国产另类| 亚洲性色视频| 国产欧美日韩一级| 久久蜜桃av| 日韩av中文字幕一区二区| yellow在线观看网址| 老司机久久99久久精品播放免费| 美女尤物国产一区| 国产一区导航| 精品香蕉视频| 一区二区高清| 日韩国产激情| 日本免费一区二区视频| 国产在线看片免费视频在线观看| 免费日韩av| a日韩av网址| 日本免费在线视频不卡一不卡二| 久久久久久网| 日本不卡视频在线| 国产91精品对白在线播放| 国产欧美三级| 99综合视频| 国精品产品一区| 日本不卡一二三区黄网| 婷婷国产精品| 麻豆免费精品视频| 男人操女人的视频在线观看欧美 | 正在播放日韩精品| 日韩av不卡一区二区| 欧美一区二区三区高清视频| 日本成人在线视频网站| 99国产精品免费视频观看|