Java中l(wèi)ambda表達式實現(xiàn)aop切面功能
背景:最近項目中涉及到自定義線程池中子線程獲取父線程的traceId,這個數(shù)據(jù)的傳遞過程可以用lamdba表達式進行封裝實現(xiàn)的。這讓我想到spring容器的三級緩存。其中的一個緩存singletonFactories就是存放的lambda表達式的。
// 緩存的聲明 private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>(16);
// lambda作為參數(shù)調(diào)用addSingletonFactory方法 this.addSingletonFactory(beanName, () -> { return this.getEarlyBeanReference(beanName, mbd, bean); }); // addSingletonFactory方法 protected void addSingletonFactory(String beanName, ObjectFactory<?> singletonFactory) { Assert.notNull(singletonFactory, "Singleton factory must not be null"); synchronized(this.singletonObjects) { if (!this.singletonObjects.containsKey(beanName)) { // 緩存中添加lambda this.singletonFactories.put(beanName, singletonFactory); this.earlySingletonObjects.remove(beanName); this.registeredSingletons.add(beanName); } } }
一些業(yè)務(wù)邏輯可以通過lambda表達式進行封裝,就可以當(dāng)作一個參數(shù)一樣進行傳遞,然后在需要的時候進行執(zhí)行。但是它的強大并不止于此,還可以當(dāng)作aop切面進行使用。通過一個demo進行展示
lambda表達式實現(xiàn)切面功能
定義一個函數(shù)式接口
@FunctionalInterface public interface DemoInterface { void Demo(); }
創(chuàng)建兩個實現(xiàn)類
public class DemoSonOne implements DemoInterface{ public DemoSonOne(Integer age) { this.age = age; } private Integer age; public Integer getAge() { return age; } // 重寫接口 @Override public void Demo() { System.out.println("I'm DemoSonOne, My age is " + age); } }
public class DemoSonTwo implements DemoInterface{ public DemoSonTwo(String name) { this.name = name; } private String name; public String getName() { return name; } // 實現(xiàn)接口 @Override public void Demo() { System.out.println("I'm DemoSonOne, My name is " + name); } }
客戶端
public class DemoMain { // lambda表達式進行封裝 public static DemoInterface wrap(final DemoInterface demoInterface){ return () -> { System.out.println("Demo方法要執(zhí)行了"); demoInterface.Demo(); System.out.println("Demo方法要執(zhí)行完了"); }; } public static void main(String[] args) { DemoSonOne demoSonOne = new DemoSonOne(18); DemoSonTwo demoSonTwo = new DemoSonTwo("haha"); demoSonOne.Demo(); System.out.println("-----------------------"); demoSonTwo.Demo(); System.out.println("-----------------------"); DemoInterface wrapOne = wrap(demoSonOne); DemoInterface wrapTwo = wrap(demoSonTwo); wrapOne.Demo(); System.out.println("-----------------------"); wrapTwo.Demo(); }}public class DemoMain { // lambda表達式進行封裝 public static DemoInterface wrap(final DemoInterface demoInterface){ return () -> { System.out.println("Demo方法要執(zhí)行了"); demoInterface.Demo(); System.out.println("Demo方法要執(zhí)行完了"); }; } public static void main(String[] args) { DemoSonOne demoSonOne = new DemoSonOne(18); DemoSonTwo demoSonTwo = new DemoSonTwo("haha"); demoSonOne.Demo(); System.out.println("-----------------------"); demoSonTwo.Demo(); System.out.println("-----------------------"); DemoInterface wrapOne = wrap(demoSonOne); DemoInterface wrapTwo = wrap(demoSonTwo); wrapOne.Demo(); System.out.println("-----------------------"); wrapTwo.Demo(); } }
執(zhí)行結(jié)果
執(zhí)行結(jié)果如下,可以看到經(jīng)過wrap方法封裝后的DemoInterface接口對象,執(zhí)行過程都會走lamdba中的代碼。給人一種aop的感覺
缺點
經(jīng)過wrap方法返回的對象都是DemoInterface類型的,它是接口類型,如果在某種特定的情況下能夠確定它是由某個子類類型實力化得到的,想要強轉(zhuǎn)回去,然后獲取子類獨有的屬性,這種情況下會報錯。
public static void main(String[] args) { DemoSonOne demoSonOne = new DemoSonOne(18); // 經(jīng)過lambda封裝,得到接口類型 DemoInterface wrapOne = wrap(demoSonOne); wrapOne.Demo(); // 由接口類型轉(zhuǎn)換為現(xiàn)實類類型 DemoSonOne wrapOne1 = (DemoSonOne) wrapOne; Integer age = wrapOne1.getAge(); System.out.println(age); }
錯誤結(jié)果顯示如下:
Exception in thread "main" java.lang.ClassCastException: class functionInterface.DemoMain$$Lambda$14/0x0000000800066840 cannot be cast to class functionInterface.DemoSonOne (functionInterface.DemoMain$$Lambda$14/0x0000000800066840 and functionInterface.DemoSonOne are in unnamed module of loader 'app')
at functionInterface.DemoMain.main(DemoMain.java:26)
由此可見該方法進行封裝有好處,也有壞處,所以要謹(jǐn)慎使用。
到此這篇關(guān)于Java中l(wèi)ambda表達式實現(xiàn)aop切面功能的文章就介紹到這了,更多相關(guān)lambda表達式實現(xiàn)aop切面內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringSecurity實現(xiàn)動態(tài)加載權(quán)限信息的方法
這篇文章主要介紹了SpringSecurity實現(xiàn)動態(tài)加載權(quán)限信息,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定需要的朋友可以參考下2022-01-01詳解Java實現(xiàn)數(shù)據(jù)結(jié)構(gòu)之并查集
并查集這種數(shù)據(jù)結(jié)構(gòu),可能出現(xiàn)的頻率不是那么高,但是還會經(jīng)常性的見到,其理解學(xué)習(xí)起來非常容易,通過本文,一定能夠輕輕松松搞定并查集2021-06-06SpringBoot中使用Redis對接口進行限流的實現(xiàn)
本文將結(jié)合實例代碼,介紹SpringBoot中使用Redis對接口進行限流的實現(xiàn),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-07-07Java使用EasyExcel模版導(dǎo)出詳細(xì)操作教程
業(yè)務(wù)中經(jīng)常需要按照一個特定的模板導(dǎo)出特定內(nèi)容,有些單元格還要求特殊的格式,所以下面這篇文章主要給大家介紹了關(guān)于Java使用EasyExcel模版導(dǎo)出的相關(guān)資料,需要的朋友可以參考下2023-10-10