最近在學習ssh框架時,照著網上做了一個商城系統,之前在一些需要用戶存在的操作中,都是在每一個action中寫重復的代碼,這樣做現在想起來并不好,想起了spring的aop,于是想通過aop來給每個需要用戶操作的Action驗證用戶登錄狀態。
想法是這樣的:
1. 用戶登錄時把userId放入session中
2. 通過spring 寫一個advice來獲取session中的userId,判斷用戶登錄狀態,如果userId不符合,則拋出自定義異常
3. 通過struts中配置來捕獲異常,跳轉界面
以下是代碼:
advice代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
|
public class IsUserLoginAdvice{ public void isUserLogin() throws UserNotFoundException{ // TODO Auto-generated method stub int id= 0 ; Map sessionMap=ActionContext.getContext().getSession(); System.out.println(sessionMap); try { //這里在一開始時userId是不存在的可能會拋出NullPointException,catch起來 id=( int ) sessionMap.get( "userId" ); //在用戶注銷時我把session中的userId設為0 if (id== 0 ){ throw new UserNotFoundException(); } } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); throw new UserNotFoundException(); } } } |
struts.xml:
這里通過全局異常映射來處理這個異常:
1
2
3
4
5
6
7
8
9
10
11
|
< package name = "struts-global" namespace = "/" extends = "struts-default" > < global-results > < result name = "userNotFound" >/web_resource/error_jsp/user_not_found.jsp </ result > </ global-results > < global-exception-mappings > < exception-mapping result = "userNotFound" exception = "com.lsj.market.exception.UserNotFoundException" ></ exception-mapping > </ global-exception-mappings > </ package > |
全局異常有個name屬性,給那些想要共享該異常捕獲的package繼承,這樣就可以共享該異常捕獲行為:
1
|
< package name = "com.lsj.market.action.user" extends = "struts-global" > |
applicationContext.xml:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
<!-- aop設置 --> < aop:config proxy-target-class = "true" > < aop:aspect ref = "isUserLoginAdvice" > < aop:pointcut id = "isUserLoginPointcut" expression="execution (* com.lsj.market.action..GetUser*.*(..)) or execution (* com.lsj.market.action..*Update*Action*.*(..)) or execution (* com.lsj.market.action..*Delete*Action*.*(..)) or execution (* com.lsj.market.action..GetMarketCar*.*(..)) or execution (* com.lsj.market.action..MarketCar*.*(..)) or execution (* com.lsj.market.action..ToFlower*.*(..)) or execution (* com.lsj.market.action..Flower*Add*.*(..))"/> < aop:before method = "isUserLogin" pointcut-ref = "isUserLoginPointcut" /> </ aop:aspect > </ aop:config > <!-- 聲明advice Bean --> < bean id = "isUserLoginAdvice" class = "com.lsj.market.aop.IsUserLoginAdvice" ></ bean > |
其中pointcut可以通過or 來連接多個切入點,這里有這么多切入點是因為第一次做,沒想到用aop,各個Action的命名沒有考慮太多,導致現在必須配置這么多個切入點表達式- -!!!
還有一個,如果struts交由spring管理時,即struts.xml中配置了這一句:
1
|
< constant name = "struts.objectFactory" value = "spring" /> |
在生成代理類時會發生錯誤,無法捕捉到拋出的異常,在網上查了后發現需要在struts.xml加入這一句,struts就可以捕捉到該異常了:
1
2
|
<!-- 總是確保使用spring的自動裝備策略 --> < constant name = "struts.objectFactory.spring.autoWire.alwaysRespect" value = "true" /> |
剛剛還想刪除這一句配置后把異常發上來,但是發現刪除后居然還可以運行?!
算了還是寫上來,以后遇到這個問題,還可以看一下博客。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。
原文鏈接:http://blog.csdn.net/caser_hdmi/article/details/75205783