Spring AOP self invocation存在的問題以及如何解決

Self invocation存在的問題

假設(shè)我們有如下的TestService

@Service
public class TestServiceImpl implements TestService {

    @Override
    public void saveAB() {
        this.saveA();
        this.saveB();
    }

    @Transactional
    @Override
    public void saveA() {
        System.out.println("saveA");
    }

    @Transactional
    @Override
    public void saveB() {
        System.out.println("saveB");
    }
}

還有如下的TestServiceTest

@RunWith(SpringRunner.class)
@SpringBootTest
public class TestServiceTest {

    @Autowired
    private TestService testService;

    @Test
    public void test() {
        testService.saveAB();
        testService.saveA();
        testService.saveB();
    }
}

TestServiceTest.test()中踏志,雖然testService.saveA()testService.saveB()會(huì)在事務(wù)中執(zhí)行秒紧,但是testService.saveAB()中的this.saveA()this.saveB()都不會(huì)在事務(wù)中執(zhí)行,這就是self invocation存在的問題窒朋。

為什么self invocation會(huì)存在問題?

首先我們必須要清楚Spring AOP是基于代理的,但是在介紹代理之前,先來看看沒有代理時(shí)是怎么樣的弊添。

public class SimplePojo implements Pojo {

    @Override
    public void foo() {
        System.out.println("foo");
        this.bar();
    }

    @Override
    public void bar() {
        System.out.println("bar");
    }
}
public class Main {

    public static void main(String[] args) {
        Pojo pojo = new SimplePojo();
        pojo.foo();
    }
}

執(zhí)行Main.main(String[] args),會(huì)輸出如下內(nèi)容:

foo
bar

當(dāng)我們調(diào)用pojo.foo()時(shí)捌木,我們是直接調(diào)用pojofoo()油坝,如下圖所示:


接下來我們就看看調(diào)用代理的方法是怎樣的。

我們修改Main為如下所示:

public class Main {

    public static void main(String[] args) {
        SimplePojo target = new SimplePojo();
        ProxyFactory factory = new ProxyFactory(target);
        factory.addInterface(Pojo.class);
        factory.addAdvice(new MethodBeforeAdvice() {
            @Override
            public void before(Method method, Object[] args, Object target) throws Throwable {
                System.out.println("before invoking method");
            }
        });

        Pojo pojo = (Pojo) factory.getProxy();
        pojo.foo();
    }
}

執(zhí)行Main.main(String[] args)刨裆,會(huì)輸出如下內(nèi)容(注意前面的foo前面的before invoking method):

before invoking method
foo
bar

這里的pojo是一個(gè)代理澈圈,當(dāng)我們調(diào)用pojo.foo()時(shí),其執(zhí)行情況如下圖所示:

當(dāng)調(diào)用pojo.foo()時(shí)帆啃,其實(shí)是調(diào)用代理的foo()瞬女,這個(gè)時(shí)候會(huì)打印出before invoking method,之后調(diào)用會(huì)來到target object努潘,調(diào)用targetfoo()诽偷,這時(shí)會(huì)打印出foo,在targetfoo()中會(huì)調(diào)用this.bar()疯坤,而這個(gè)this不是代理报慕,而是target,這就是為什么沒在打印出bar之前打印before invoking method的原因压怠。

怎么解決眠冈?

Spring proxy self-invocation ? my2cents列出了很多種方法來解決這個(gè)問題,下面我只介紹我個(gè)人認(rèn)為最簡單的方法菌瘫,我們將TestServiceImpl修改為如下所示(注意看注釋)洋闽。

@Service
public class TestServiceImpl implements TestService {

    private final TestService self;

    // 注入自身,一定要加上@Lazy
    public TestServiceImpl(@Lazy TestService self) {
        this.self = self;
    }

    @Override
    public void saveAB() {
        // 將this替換為self
        self.saveA();
        self.saveB();
    }

    @Transactional
    @Override
    public void saveA() {
        System.out.println("saveA");
    }

    @Transactional
    @Override
    public void saveB() {
        System.out.println("saveB");
    }
}

這個(gè)時(shí)候再執(zhí)行TestServiceTest.test()突梦,會(huì)發(fā)現(xiàn)testService.saveAB()中的saveA()saveB()都會(huì)在事務(wù)中執(zhí)行。

參考資料

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末羽利,一起剝皮案震驚了整個(gè)濱河市宫患,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌这弧,老刑警劉巖娃闲,帶你破解...
    沈念sama閱讀 218,451評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件虚汛,死亡現(xiàn)場離奇詭異,居然都是意外死亡皇帮,警方通過查閱死者的電腦和手機(jī)卷哩,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,172評論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來属拾,“玉大人将谊,你說我怎么就攤上這事〗グ祝” “怎么了尊浓?”我有些...
    開封第一講書人閱讀 164,782評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長纯衍。 經(jīng)常有香客問我栋齿,道長,這世上最難降的妖魔是什么襟诸? 我笑而不...
    開封第一講書人閱讀 58,709評論 1 294
  • 正文 為了忘掉前任瓦堵,我火速辦了婚禮,結(jié)果婚禮上歌亲,老公的妹妹穿的比我還像新娘菇用。我一直安慰自己,他們只是感情好应结,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,733評論 6 392
  • 文/花漫 我一把揭開白布刨疼。 她就那樣靜靜地躺著,像睡著了一般鹅龄。 火紅的嫁衣襯著肌膚如雪揩慕。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,578評論 1 305
  • 那天扮休,我揣著相機(jī)與錄音迎卤,去河邊找鬼。 笑死玷坠,一個(gè)胖子當(dāng)著我的面吹牛蜗搔,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播八堡,決...
    沈念sama閱讀 40,320評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼樟凄,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了兄渺?” 一聲冷哼從身側(cè)響起缝龄,我...
    開封第一講書人閱讀 39,241評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后叔壤,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體瞎饲,經(jīng)...
    沈念sama閱讀 45,686評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,878評論 3 336
  • 正文 我和宋清朗相戀三年炼绘,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了嗅战。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,992評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡俺亮,死狀恐怖驮捍,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情铅辞,我是刑警寧澤厌漂,帶...
    沈念sama閱讀 35,715評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站斟珊,受9級特大地震影響苇倡,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜囤踩,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,336評論 3 330
  • 文/蒙蒙 一旨椒、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧堵漱,春花似錦综慎、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,912評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至愉镰,卻和暖如春米罚,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背丈探。 一陣腳步聲響...
    開封第一講書人閱讀 33,040評論 1 270
  • 我被黑心中介騙來泰國打工录择, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人碗降。 一個(gè)月前我還...
    沈念sama閱讀 48,173評論 3 370
  • 正文 我出身青樓隘竭,卻偏偏與公主長得像,于是被迫代替她去往敵國和親讼渊。 傳聞我的和親對象是個(gè)殘疾皇子动看,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,947評論 2 355