日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

面向切面编程AOP的最佳入门示例

發(fā)布時間:2024/9/30 编程问答 38 豆豆
生活随笔 收集整理的這篇文章主要介紹了 面向切面编程AOP的最佳入门示例 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

1.AOP簡單上手

? ? ? ?AOP(Aspect Oriented Programming),意為:面向切面編程,通過預編譯方式和運行期動態(tài)代理實現程序功能的統(tǒng)一維護的一種技術。它通過對既有程序定義一個切入點,然后在其前后切入不同的執(zhí)行內容,比如常見的有:打開數據庫連接/關閉數據庫連接、打開事務/關閉事務記錄日志等。AOP可以降低耦合,把通用的業(yè)務提出來,提高代碼的可重用性,同時提高開發(fā)的效率,使開發(fā)人員只關心真正的業(yè)務邏輯.

  • 引入AOP的依賴
  • <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId></dependency>

    ? ? ? ?@EnableAspectJAutoProxy已經默認開啟,不需要在啟動類上添加.
    ? ? ? ?而當我們需要使用CGLIB來實現AOP的時候,需要配置spring.aop.proxy-target-class=true.但是注意:高版本spring自動根據運行類選擇JDK或CGLIB代理,也就是當運行類沒有繼承接口,spring也會自動使用CGLIB代理。我們無需設置proxy-target-class屬性,JDK動態(tài)代理是模擬接口實現的方式,cglib是模擬子類繼承的方式,一般采用前者,因為前者效率高。

    2 簡單示例

    @Aspect @Component public class LogAspect {@Around(value = "execution(* com.jun.test.microservice..*(..))")public Object doAroundAdvice(ProceedingJoinPoint proceedingJoinPoint){Object returnVal = null;final Logger log = getLog(proceedingJoinPoint);final String methodName = proceedingJoinPoint.getSignature().getName();try {final Object[] args = proceedingJoinPoint.getArgs();final String arguments;if (args == null || args.length == 0) {arguments = "";} else {arguments = Arrays.deepToString(args);}log.info("Entering method [" + methodName + " with arguments [" + arguments + "]");returnVal = proceedingJoinPoint.proceed();return returnVal;} catch (Throwable throwable) {throwable.printStackTrace();} finally {log.info("Leaving method [" + methodName + "] with return value [" + (returnVal != null ? returnVal.toString() : "null") + "].");}return null;}protected Logger getLog(final JoinPoint joinPoint) {final Object target = joinPoint.getTarget();if (target != null) {return LoggerFactory.getLogger(target.getClass());}return LoggerFactory.getLogger(getClass());}}

    注意

    1.當有多個切面針對同一個切點時,可以使用@order()來指定執(zhí)行順序,可以理解成多個同心圓,要執(zhí)行的方法為圓心,最外層的order最小。從最外層按照AOP1、AOP2的順序依次執(zhí)行doAround方法,doBefore方法。然后執(zhí)行method方法,最后按照AOP2、AOP1的順序依次執(zhí)行doAfter、doAfterReturn方法。也就是說對多個AOP來說,先before的,一定后after。

    2.@Order的作用于可以在類,方法,字段上,但是他指定的是容器中bean的加載順序.

    2.自定義注解+AOP

    可以通過自定義注解,來更針對性的定義切點.
    2.1 首先自定義注解

    @Target({ElementType.PARAMETER, ElementType.METHOD,ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Documented public @interface SystemServiceLog {String description() default ""; }

    2.2 定義切面類
    實現注有@SystemServiceLog的方法拋出異常時,記錄日志。

    @Aspect @Component public class AnnotationAspect {//Service層切點,攔截添加@SystemServiceLog的@Pointcut("@annotation(com.jun.test.microservice.aspect.SystemServiceLog)")public void serviceAspect() {}@AfterThrowing(pointcut = "serviceAspect()", throwing = "e")public void doAfterThrowing(JoinPoint joinPoint, Throwable e) {//獲得request和sessionHttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();HttpSession session = request.getSession();//獲取請求ipString ip = request.getRemoteAddr();//獲取用戶請求方法的參數并序列化為JSON格式字符串String params = "";if (joinPoint.getArgs() != null && joinPoint.getArgs().length > 0) {for ( int i = 0; i < joinPoint.getArgs().length; i++) {params += JsonUtils.object2Json(joinPoint.getArgs()[i]) + ";";}}Logger log = getLog(joinPoint);try {/*==========數據庫日志=========*/log.info("異常處理");log.info(e.getMessage());log.info((joinPoint.getTarget().getClass().getName() + "." + joinPoint.getSignature().getName() + "()"));log.info(params);log.info(ip);//保存數據庫//logService.add(log);System.out.println("=====異常通知結束=====");} catch (Exception ex) {//記錄本地異常日志log.error("==異常通知異常==");log.error("異常信息:{}", ex.getMessage());}/*==========記錄本地異常日志==========*/log.error("異常方法:{}異常信息:{}參數:{}", joinPoint.getTarget().getClass().getName() + joinPoint.getSignature().getName(), e.getMessage(), params);}protected Logger getLog(final JoinPoint joinPoint) {final Object target = joinPoint.getTarget();if (target != null) {return LoggerFactory.getLogger(target.getClass());}return LoggerFactory.getLogger(getClass());}}

    2.3 在需要的方法上添加自定義注解

    @Service public class UsrServiceImpl implements IUserService {@SystemServiceLog(description = "查詢用戶")public UserAccount getUser(String id){int i = 0/0;return new UserAccount(id,"hello");} }

    ? ? ? ?以上即可.結合自定義注解和模板,可以實現很多方便的功能,如記錄系統(tǒng)操作日志(添加,刪除等等),記錄異常等.對一些方法都有的邏輯可以提出來,如參數校驗,限制訪問頻率等.

    與50位技術專家面對面20年技術見證,附贈技術全景圖

    總結

    以上是生活随笔為你收集整理的面向切面编程AOP的最佳入门示例的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    主站蜘蛛池模板: 国产怡红院 | 好好热视频 | av免费不卡| 国产中文| 欧美精品网站 | 成人资源在线观看 | 青青射| 欧美另类在线观看 | 欧美日韩黄色片 | 99成人在线观看 | 中文av一区二区 | 国产又爽又黄的视频 | 高清久久久久久 | av小说天堂网 | 97少妇| 欧美极品在线播放 | 久青草视频在线 | 天堂精品一区二区三区 | 欧美日韩色图片 | 中文字幕在线播放一区二区 | 貂蝉被到爽流白浆在线观看 | 7777精品久久久久久 | 99久久精品日本一区二区免费 | 中文字幕永久在线观看 | 91视频在| 免费看操片 | 黄色片免费在线播放 | 日韩黄色在线视频 | 草草草在线观看 | 色老头在线一区二区三区 | 日韩人妻精品无码一区二区三区 | av导航网| 一区二区三区免费观看 | xxxxxhd亚洲人hd | 久久久久久久久网站 | 噼里啪啦动漫高清在线观看 | 在线看黄色网 | 高清在线一区二区三区 | 蜜桃久久久久久 | 深爱激情丁香 | jzzjzz日本丰满少妇 | 超碰在线免费公开 | 国产毛片毛片毛片 | 日本少妇一区二区 | 久色视频在线播放 | 校园春色欧美 | 日韩在线黄色 | 影音先锋久久久久av综合网成人 | 国产福利专区 | 女人高潮娇喘1分47秒 | 筱田优全部av免费观看 | 国产精品av一区二区 | 香蕉在线播放 | 少妇久久久久久被弄高潮 | 中文字幕高潮 | 免费国产小视频 | 人日人视频| 国产福利一区二区 | 欧美1| 国产成人h| www.精品在线 | 日韩毛片大全 | 懂色一区二区三区免费观看 | 欧美日韩视频无码一区二区三 | 极品人妻一区二区三区 | 国产日韩网站 | 青春草在线视频免费观看 | 黄色国产毛片 | 荒野求生21天去码版网站 | 国产黄站 | 梦梦电影免费高清在线观看 | 国产污视频在线 | 福利片第一页 | xxx视频网站 | 久久久久久久久久99精品 | 中文字幕二区在线观看 | 毛片毛片毛片毛片毛片毛片毛片毛片毛片 | 国产精品入口夜色视频大尺度 | 人妻 丝袜美腿 中文字幕 | 欧美激情欧美激情在线五月 | 三上悠亚在线播放 | 日韩无码精品一区二区三区 | 色姑娘av | 1024福利 | 肉色超薄丝袜脚交一区二区 | 国产亚洲精品久 | 琪琪电影午夜理论片八戒八戒 | xxxx国产精品 | 国产欧美日韩在线 | 免费在线观看a级片 | 国产日韩成人内射视频 | 国产成人综合精品 | 大奶子在线观看 | 免费超爽大片黄 | 福利二区视频 | 精品日韩| 亚洲国产视频一区 | caoporn视频在线 | 久久久欧洲 |