TA的每日心情 | 衰 2021-2-2 11:21 |
---|
签到天数: 36 天 [LV.5]常住居民I
|
9#
发表于 2015-06-02 12:45:13
|只看该作者
三、 Struts21 }' {* f2 c1 X' t
1. Action名称的搜索顺序
7 K( W8 F; L; D6 N6 l# N) @4 v(1).获取请求路径的URL,例如URL是:http://server/struts2/path1/path2/path3/test.action
; |8 ?3 ]& @- Z# X. ^5 n+ Q
3 k7 U, j/ m3 E2 S9 Q7 [4 J(2).首先寻找namespace为/path1/path2/path3的package,如果不存在这个package则执行步棸3,如果存在这个package,则在这个package中寻找名字为test的action,当在该package下寻找不到action时就会直接跑到默认namespace的package里面中寻找action(默认的命名空间为空字符串),如果在默认namespace的package里面还寻找不到该action,页面提示找不到action1 h. w) u" I5 K; ^
# J: }9 d; c) ^
(3).寻找namespace为/path1/path2的package,如果不存在这个package,则转至步棸4,如果存在这个package,则在这个package中寻找名字为test的action,当在该package中寻找不到action时就会直接跑到默认namespace的package里面去找名字为test的action,在默认的namespace的package里面还寻找不到该action,页面提示找不到action) Y8 ?0 N9 c, W1 D
) }9 w- `1 O/ P6 Y) c(4).寻找namespace为/path1的package,如果不存在这个package则执行步棸5,如果存在这个package,则在这个package中寻找名字为test的action,当在该package中寻找不到action时就会直接跑到默认namespace的package里面去找名字为test的action,在默认namespace的package里面还寻找不到该action,页面提示找不到action
5 C# o0 V' E3 A5 j $ \: n" K( G" z
(5).寻找namespace为"/"的package,如果存在这个package,则在这个package中寻找名字为test的action,当在package中寻找不到action或者不存在这个package时,都会去默认namespace的package里面寻找action,如果还是找不到,页面提示找不到action
C7 Y0 P9 C4 h* U3 b z/ r% a- }! u4 }
(6).在struts2中默认的处理后缀是.action,不加后缀也可以2 `; J i8 n! L- M
. e( T0 q! m2 |6 f2 q& S
2. Action配置的各项默认值
, U, Q, i$ E! b- ?& X: [3 z( B(1).struts1中:<actionpath="/control/employee/addUI"forward="/WEB-INF/page/employeeAdd.jsp"/>实现请求转发,action将请求转发给视图jsp K* E" }! a3 z7 @2 g
- l) @7 X O. F; X8 m" I! g7 W
(2).在struts2中,<actionname="addUI"><result>/WEB-INF/page/employeeAdd.jsp</result></action>,不需要设置addUI的类路径class属性了
5 h B# s9 J' _4 a
2 ?( Q( i5 L+ c# y(3).Action配置中的各项默认值:
$ q6 T, L3 T# x3 x' x<package name="itcastnamespace="/test" extends="struts-default">7 p2 P( K7 F- o% W( f! A
<action name="hellowrold"class="cn.itcast.action.HelloWorldAction"method="execute">
* _ A% _: Y2 U) p<result name="success">/WEB-INF/page/hello.jsp</result>
& p6 Z3 T9 K2 o' h R- k</action>4 Q+ X. U: _7 R
</package>
0 Z# V) D6 h6 N, @9 Z如果没有为action指定class,默认是ActionSupport,可以查看ActionSupport的源代码,首先交给ActionSupport类处理.* Z- l3 ~& x# K) N3 m6 x
如果没有为action指定method,默认执行action中的execute()方法,这个方法的返回值为"success";" }9 ?+ r7 j; t( l' \1 S6 D" }! E
7 H" f2 y+ k: c. I. X' B4 k
(4).如果没有指定result的name属性,默认值为success,正好和execute方法的返回值相同,所以可以实现视图的转发; e+ s7 X' v3 L Y) O1 I8 S% H
; |! j' L$ D( U7 N' _# i
3.OGNL表达式
, O+ l, H# j7 ?/ {(1).OGNL表达式:OGNL是Object Graphic Navigation Language(对象图导航语言)的缩写,它是一个开源项目,struts2框架使用OGNL作为默认的表达式语言5 }7 J" z7 M# f& T: V. B0 ?) a
第一:相对EL表达式,它提供了平时我们需要的一些功能,如:- t5 F( f* C9 t. {- p: x
支持对象方法的调用,如:xxx.sayHello();
$ h4 w2 n. o3 q9 S- \4 O第二:支持类静态方法调用和值访问,表达式的格式为@[类全名 (包括包路径) ]@[方法名|值名],例如:@java.lang.String@format('foo %s,'bar')或@cn.itcast.Constant@APP_NAME E+ |& X5 m. u4 L6 _
第三:操作集合对象3 Q+ P% A9 l, n
# Q& m1 a4 t8 [" |2 t& p) {; c" K, O(2).OGNL有一个上下文(Context)概念,说白了上下文就是一个MAP结构,他实现了java.utils.Map接口,在struts2中上下文(context)的实现为ActionContext
3 X- m4 \4 U, l% D, Z
% V* s& T! q% M. T( l(3).struts2中的OGNLContext是实现者为ActionContext,它的结构为:OGNL Context:ValueStack(值栈,他是根对象),parameters,request,session,application,attr,当struts2接受一个请求时,会迅速创建ActionContext,ValueStack,action,然后把action存放进ValueStack,所以action的实例变量可以被OGNL访问
N. N# o8 o, Y3 h# I, F, D ' B9 Z/ O3 F: r1 |7 e" G# _. t
(4).当要访问某个对象只需在其前面加上一个'#',例如:#request,当然有一个特殊的例子,就是根对象,会省略'#',OGNL会设定一个根对象(root对象),在struts2中根对象就是ValueStack(值栈),如果要访问根对象(即ValueStack)中的对象的属性,则可以省略#命名空间,直接访问该对象的属性即可.; g6 \! B9 Z8 L0 B8 {+ y/ c
; R& U$ V5 O* T
(5).在struts2中,根对象ValueStack的实现类为OgnlValueStack,该对象不是我们想象的只存放单个值,而是存放一组对象,在OgnlValueStack类里有一个List类型的root变量,就是使用它存在一组对象,在root变量中处于第一位的对象叫栈定对象(存放action),通常我们在OGNL表达式里直接写上属性的名称即可访问root变量对象的属性,搜索顺序是从栈定对象开始寻找,如果栈定对象不存在该属性,就会从第二个对象寻找,如果没有找到就从第三个对象寻找,依次往下访问,直到找到为止。大家注意:struts2中,OGNL表达式需要配合struts标签才可以使用,如:<s:property value="name"/>,value属性接受的是OGNL表达式,搜索是否含有name属性
) P2 g; J: z, N7 h, Y3 Q1 z* H 7 ^$ y5 x' D) e! w0 p8 X
(6).由于ValueStack(值栈)是struts2中OGNL的根对象,如果用户需要访问值栈中的对象,在JSP页面可以直接通过下面的EL表达式访问ValueStack(值栈)中对象的属性:
' S# B7 q: L G* i& F: R${foo}获得值栈中某个对象的foo属性: Q$ d2 B Y- _9 k# t4 V
如果访问其他Context中的对象,由于他们不是根对象,所以在访问时,需要添加#前缀2 J# M6 g# J* m0 H4 o) Z# d3 z$ h
第一:application对象:用于访问ServletContext,例如#application.userName或者#application['userName'],相当于调用ServletContext的getAttribute("username");8 T( ?* I: d( R; f; Z7 G8 L" a
第二:session对象:用来访问HttpSession,例如#session.userName或者#session['userName'],相当于调用session.getAttribute("userName");
. m: s9 v7 l/ f第三:request对象:用来访问HttpServletRequest属性(attribute)的Map,例如#request.userName或者#request['userName'],相当于调用request.getAttribut("userName");
1 h8 o. _& W7 s! N第四:parameters对象:用于访问HTTP的请求参数,例如#parameters.userName或者#parameters['userName'],相当于调用request.getParameter("username");
8 `3 j0 P- ~8 a3 f0 K' l第五:attr对象:用于按page->request->session->application顺序访问其属性.
% `' X5 g: T1 G) t9 i
+ u/ S- m4 n3 _+ z* h(7).为何使用EL表达式能够访问valueStack中对象的属性:原因是struts2对HttpServletRequest做了进一步的封装,简单代码如下! x# X g1 W( U# t8 h9 _) i
public class StrutsRequestWrapper extends HttpServletRequestWrapper{
+ l: n0 m) L% C3 M: L) o& S, H5 spublicStrutsRequestWrapper(HttpServletRequest req){
! W/ z% ^: i# [$ u2 A6 ~8 [2 msuper(req);0 p. n- H/ G5 z% |3 _- E' H- @0 z
}. X$ @5 S2 J0 `; @6 O' i- r# G
public Object getAttribut(String s){
9 P) w8 `! B9 C ActionContext ctx = ActionContext.getContext();
% p( S- U5 n( Z' aObject attribute = super.getAttribute(s)//先从request范围获取属性值
1 ]/ i( k' i( h: B; y' cif(ctx !=null){
5 l: h' B: h) C4 a' p if(attribute==null){
; Q; j9 I8 ^; g9 z1 |7 `$ P3 M....
1 z2 U9 }( l4 eValueStack stack =ctx.getValueStack();. @( h; B/ T& {* Z, L0 z
attribute=stack.findValue(s);//寻找规则就是前面的寻找规则
% p- q7 s' }& n7 A....2 a; I3 x0 v& a# x! c5 N; d
}
8 ?8 F0 H d1 Q2 `) J}5 B6 I0 Y; o6 V8 J0 s7 V
return attribute' E/ h, T8 a' [* X0 o# v4 s
}! f8 Y D, I2 @ L G' X' }" P
}
% H1 C+ I4 w# k4 B& r; NEL表达式只能访问ValueStack对象中的内容
5 S, R0 X7 i' m- H
) j: ]7 M2 y. D7 _0 x2 `(8).采用OGNL表达式创建List/Map集合对象,如果需要一个集合元素的时候(例如List对象或者Map对象),可以使用OGNL中同集合相关的表达式,使用如下代码直接生成一个List对象:% {3 R% B: {; m; N3 ^& y
<s:set name="list"value="{'zhangming','xiaoi','liming'}"/>把当前迭代的对象放在值栈的栈顶- K3 R$ f# h8 Z. I4 N5 s, y% W5 W
<s:iterator value="#list">, b0 N& Z+ d4 x5 c! |
<s:property/><br>( E; a; i8 A2 v, }
</s:iterator>3 B9 V; ^3 [, {( P W1 h* J, i
Set标签用于将某个值放入指定范围5 O$ l; e) l( D6 |; d. L" n: y0 b
scope:指定变量被放置的范围,该属性可以接受application,session,request,page或action.如果没有设置该属性,则默认放置在OGNL Context中.
g; t: \ T2 I$ P/ ?5 O( a6 uvalue:赋给变量的值,如果没有设置该属性,则将ValueStack栈顶的值赋给变量
$ }: e/ _. x8 y m D T生成一个Map对象:8 q1 ~6 x6 R! `" e
<s:set name="foobar"value="#{'foo1':'bar1','foo2':'bar2'}"/>
" T z, d8 K7 h! p<s:iteratorvalue="#foobar">//迭代标签,把当前迭代的对象放在值栈的栈顶(entry对象),foobar是Map对象和request等对象是同等地位,访问时需要使用'#'8 A" `) [$ b" T" a
<s:property value="key"/>=<s:propertyvalue="value"/><br>
/ D- q$ l6 t2 F, J' L: \" C, W7 s</s:iterator>; V; p: z- v0 L9 z9 J
数字不用任何符号,字符串使用单引号('),对于Map采用的是maps.entrySet()这个方式进行迭代的.
7 m: m( s- y9 A" K# U! e 4 X/ j( ^6 j8 O% J
(9).property标签用于输出指定值:' b7 _$ C6 e( f
<s:set name="name"value="kk"/>
: V* d; _) k: n. A<s:property value="#name"/>
/ B- x' l* w R4 g, v8 d4 Hdefault:可选属性,如果需要输出的属性值为null,则显示该属性指定的值6 R% K8 k `0 q1 y
escape:可选属性,指定是否格式化HTML代码9 \5 d% @) @6 R# V* n* \- u- J
value:可选属性,指定需要输出的属性值,如果没有指定该属性,则默认输出ValueStack栈顶的值
! b. k* s& n7 R+ K5 gid:可选属性,指定该元素的标识
' j1 W# X8 t6 z% o9 y: C( l + ?- D$ {& A+ f9 Q' n3 L( J& t
(1)0.对于集合类型,OGNL表达式可以使用in和not in两个元素符号,其中in表达式用来判断某个元素是否在指定的集合对象中,not in判断某个元素是否不在指定集合对象中,如下所示:/ B' c {; z' Z _( B$ [5 N M
in表达式:
1 _' w7 J& Y5 z4 J<s:if test="'foo' in {'foo','bar'}">" Q+ X! ~( n/ i3 g( _2 j$ Y
在1 P$ }$ y4 ^7 B* Q, ]) O
</s:if>0 T# m: G# r7 u y5 e/ @1 q
<s:else>* d/ A# A) m" M [% F i$ I
不在
3 e) W8 T$ q+ w+ d3 m k* F</s:else>7 F0 }1 R% h! L
no in 表达式, c# i% x5 y7 a# d" M
<s:if test="'foo' not in {'foo','bar'}">: X% Q; r8 M; w. a7 _
不在
" }8 o: J9 T9 X% k5 `0 U" A</s:if>
; x+ K2 a! I- m" N4 R' P4 o<s:else>
* G9 z$ v6 ~0 \3 O在
U# p5 C% Q& F; M: T0 v7 P</s:else>: k H* L2 |0 B8 n
$ u& q) B0 \( {8 z- s) `0 M(11).OGNL表达式的投影功能,除了in和not in之外,OGNL还允许使用某个规则获得集合对象的子集,常用的有以下3个相关操作符:
1 j% k5 [1 J4 ?! ?第一:?:获得所有符号逻辑的元素
( y g6 Q& P6 B! m& _第二:^:获得符合逻辑的第一个元素& ?" r/ y& p3 W3 e) T' A* `( L7 i/ G
第三:$:获得符合逻辑的最后一个元素) G- s) [9 ^9 c/ K+ V( r8 `
例如代码:
: e; n6 g2 A9 a, c/ ?<s:iterator value="books.{?#this.price>35}"># |- l. |- s2 C+ O
<s:propertyvalue="title"/>-$<s:propertyvalue="price"/><br>
! ~+ D- [; z) B" ], I5 A</s:iterator>
3 |/ C) V! R: i+ h; { i4 b在上面代码中,直接在集合后紧跟{}运算符表明用于取出该集合的子集,{}内的表达式用于获取符合条件的元素,this指的是为了从大集合books筛选数据到小集合,需要对大集合books进行迭代,this代表当前迭代的元素,本例的表达式用于获取集合中价格大于35的书集合.
0 W; K& e$ J- @+ B) }: vpublic class BookAction extends ActionSupport{
* Q* L3 K. I1 Z3 w8 N2 @0 n5 zprivate List<Book> books;* E, ^9 ]6 j/ j z% l
1 Q, i; M0 T! n1 I: H# y
@Override% G- Y6 `* m5 j3 L' L
public String execute(){; m! _: A8 |1 U8 m
books = new LinkedList<Book>();& p% ?! p, ~: \( q) g
books.add(new Book("aadfsd","spring",23));: p6 L' T9 ^$ E" L4 A+ c1 e# L
books.add(newBook("basdfd","ejb3.0",15));" V8 J1 ~0 x! z* U4 @% h
}
, c) b R& ]& K3 ~8 _+ [/ H+ m}5 K2 Q W" d; R# {% I
' a5 d2 ]3 H# {" e" v+ c$ w0 i4.result配置的各种视图转发类型
/ s! J! y$ j2 k0 i(1).在struts1中有两种视图转发类型:容器内转发,容器外转发(重定向);; T: g) ^7 b& U ?! K b1 h# J. n
<actionpath="/control/employee/manage".../>
0 k7 j- J9 T3 C" D* A, a! l<forwardname="add">/index.jsp</forward>3 n; J8 c) e4 s; [$ g/ W( \
<forward name="add"redirect=""/index.jsp</forward>
H2 O7 m0 D! i& A6 c5 G</action>
6 [! J* s/ }- m. S/ P+ L
( m; _4 @) k7 l(2).struts2中的视图转发类型:result配置类似于struts1中的froward,但是struts2中提供了多种结果类型,常用的类型有:dispatcher(默认值)、rdierect、redirectAction、plainText;dispatcher对应于struts1中的容器内部请求转发,redirect对应于struts1中的容器外部转发(重定向)1 ~- M$ ]. O& E9 D$ L
+ \( z: g% j4 ?) S/ ~
(3).<actionname="helloworld"class="cn.itcast.action.HelloWorldAction">
, M5 r% f; e9 c8 ^) Y<resultname="success">/WEB-INF/page/hello.jsp</result>
2 L1 p- Z7 X6 L6 h</action>' B& u6 |/ p& y9 z, b+ q( R# |
在result中还可以使用${属性名}表达式访问action中的属性,表达式里的属性名对应的action中的属性,如下:<resulttype="redirect">/View.jsp?id=${id}</result>,使用重定向可能需要将Action中的数据属性代入视图页面,这种方式太重要了,很实用.这种表达式叫做ognl表达式,struts1中是没有的,只能将属性值在代码中写死了,不像struts2中的这个表达式,很灵活.当传递的属性是中文时,需要进行URLEncoder.encode("传智播客","UTF-8")编码.
8 f8 n$ H9 T# A" \& a5 U
! C1 G1 e, d, c(4).下面是redirectAction结果类型的例子,如果重定向的action中同一个包下:) M" g! S; G* @, l7 n J
<resulttype="redircetAction">helloworld</result>
" q5 M6 T# N6 t( M如果重定向的action在别的命名空间下:* j1 {* `' q4 H- B! J
<resulttype="redirectAction">; c2 t4 p& [* y0 m0 N& |
<paramname="actionName">helloworld</param>2 i( @' J% j% j3 I8 u* T& a8 B
<paramname="namespace">/test</param>- e3 C, I( y1 P% K5 z
</result>
% o3 y h/ K; ~7 B/ c当添加用户完后,可以回到一个用户列表,此时可以重定向到action' l9 c) |4 u/ c% J& r/ o
5 d- s/ I, L5 `# f8 e(5).plaintext显示原始文件内容,例如:当我们需要原样显示JSP文件源代码的时候,我们可以使用此类型2 k4 e4 m; o& E2 A2 `5 d0 E2 ~
<result name="source"type="plainText">
, h6 o7 y, j7 ^ u. s$ p<paramname="location">/xxx.jsp</param>
0 q: @" }4 J# t( A3 `7 S<paramname="charSet">UTF-8</param><!--指定读取文件的编码-->
" Q9 v7 M/ T8 [: W) L& `2 Y5 T</result># S2 }0 @2 R; d: B9 q3 d$ y
在Eclipse中jsp是用UTF-8编码存放的,当读取jsp的内容时,是用本地字符编码的,可能出现乱码,所以要设置读取文件的编码集.8 `9 K2 j* |% y+ s8 r3 [
0 L" i; @3 X2 e* R" q6 S
(6).浏览器重定向的JSP不能放在web-inf目录中,而请求转发的JSP可以放在web-inf目录中
, Y* T0 t8 J/ O; J1 c H+ J' ~& X* O7 I2 b Z
(7).struts2中的全局视图:
- m. k+ Z3 a8 t0 H4 J8 Z% i! S<global-results>
7 ?2 a) v+ _6 ]1 K<resultname="index.jsp"></result>
( L6 ?& J5 B7 m<global-results>
. h9 W$ m( i7 w2 w; _; D6 l, k1 ]! i和struts1中的全局视图是很相似的
* A. P. ^% X P# N& m我们可以定义一个包,然后将全局视图的配置放到这个包中
3 W; W" E' s' `9 C A" d* y & m S4 \! d, y* b; B
5.struts2常用标签
' ?& R8 J& }; e5 s(1).property标签用于输出指定值:
! K) J% O E0 r5 t8 n<s:setname="name" value="kk"/>
& j [7 ]. n) E8 ]; |<s:propertyvalue ="#name"/>
) G4 ^! {9 M8 f+ j& k, t* G9 Hdefault:可选属性,如果需要输出的属性值为null,则显示该属性指定的值' p0 m8 l/ ~6 _" ^2 s. \0 w ]
escape:可选属性,指定是否格式化HTML代码
) m2 b; F7 J1 J1 F0 h' Qvalue:可选属性,指定需要输出的属性值,如果没有指定该属性,则默认输出ValueStack栈顶的值
3 n0 v f9 J( n2 }, e" n3 @; Hid:可选属性,指定该元素的标识2 N& P3 F* q. K# S" [7 P: S
! e4 S6 @ c& |1 b5 H4 N. k(2).iterator标签用于对集合进行迭代,这里的集合包含List、Set和数组
' |0 V* l" T0 Y4 x7 w# A<s:set name="list"value="{'aa','bb','cc'}"/>. Y& t3 [$ n. j. M/ `- ?2 v
<s:iterator value="#list"status="st">3 n9 C8 e4 r7 |# L4 H% ?1 `# ?* z' x7 R
<font color=<s:iftest="#st.odd">red</s:if><s:else>blue</s:else>>
4 ^# Y, i8 R3 b& O4 A7 e<s:property/></font><br>% c& ~" h" I. I! |+ k; e
</s:iterator>
# ?9 i4 l& h; H: M3 p* r$ X) wvalue:可选属性,指定被迭代的集合,如果没有设置该属性,则使用ValueStack栈顶的集合,! Q# w" v! g5 H$ x
id:可选属性,指定集合里元素的id(已被标注为过时)
. e" _( _; m; L" xstatus:可选属性,该属性指定迭代时的iteratorStatus实例,该实例包含如下几个方法:' \ c: ]) l3 a; @
int getCount(),返回当前迭代了几个元素
& h- H8 w6 }: Q9 _1 l, Tint getIndex(),返回当前迭代元素的索引
- n$ X3 t0 G& G/ i4 n1 Vboolean isEven(),返回当前被迭代元素的索引是否为偶数
8 n' U. V' D* y- j/ l! vboolean isOdd(),返回当前被迭代元素的索引是否是奇数
* g P0 S3 i7 r% Lboolean isFirst(),返回当前被迭代元素是否是第一个元素5 v5 R" O9 _2 z# S# J5 A" W2 m
boolean isLast(),返回当前被迭代元素是否是最后一个元素, A1 ^; O z7 r! {
& _# ^1 D( F/ ?, J; d7 b(3).<s:setname="age" value="21" scope="request"/>
: n4 E- N5 ^2 j5 |, L' j<s:iftest="#request.age==23">8 p+ z3 F$ r4 _% n6 Z5 p
235 t, B% x2 }1 B' i& b( Z
</s:if>/ F `% E: M" o$ X
<s:elseif test ="#age==21">3 q- `, i D+ ? I6 G
21
* q* c m0 Y4 E2 b</s:elseif>
+ a5 Z" ^' u( S: X/ K9 f- T8 ?<s:else>: l& o! j$ T, j2 I/ ^) W( C: }2 J
都不等
% j+ q% [& p3 L4 U2 |9 _</s:else>
/ \1 d8 T: z$ R2 ? ; w# H! d8 B- V' Y H6 Z& p3 Z
(4).url标签:% @3 g$ q4 \. ~( @" F+ J& s1 J
<s:url action="helloworld_add"namespace="/test">: ^" k0 M0 E1 D+ y" ^6 T; L
<s:param name="personid"value="3"/>
`. X( q9 D* I/ V! o- t</s:url>( X; J5 |- r) l+ k/ w) H5 e1 V
生成类似如下路径:
) e! M9 F. }2 t3 H6 c/struts/test/helloworld_add.action?persionid=3
7 I' x9 N" H0 Y0 S6 Y
; e7 D/ Z& z: J& h当标签的属性值作为字符串类型处理时,"%"符号的用途是计算OGNL表达式的值
7 f) f+ ^: M m o% n( v- Z; W<s:set name="myurl"value='"http://www.foshanshop.net"'/>
) O& z! F4 G" P) [ j<s:url value="#myurl"/>! E) y4 S+ ?2 Z" G4 s% _
<s:url value="%{#myurl}"/>0 E% m+ z0 B; _! N0 X
输出结果:
* p% c" `/ k' R0 N; b#myurl
; a# z) P/ o' s9 O; [& _http://www.foshanshop.net
* x2 ~" j/ S4 d6 m4 ]
* J: t& h$ K# j6 Z* ]. q(5).表单标签checkboxlist复选框
1 T$ R) Q7 v6 T4 ^: v' A* K6 T5 N4 n如果集合为list+ E _. K; Z4 r* E {
<s:checkboxlist name="list"list={'Java','Net','RoR','PHP'}" value="{'Java','Net'}"/>8 J& p% o6 u# ~: M4 k' m4 t8 o
<input type="checkbox"name="list" value="Java"checked="checked"/><label>Java</label>. K- }) W7 M$ D
<input type="checkbox"name="list" value="Net" checked="checked"/><label>Net</label>
& L" V* s6 b0 d% e# n; z* N<input type="checkbox"name="list" value="RoR" ><label>Java</label>
. ~: a/ s2 D+ d5 a: p/ k<input type="checkbox"name="list" value="Java"/><label>Java</label>2 | n3 X2 m' }
如果集合为MAP7 B% I& h' p* [ J& @% i4 V; w' v
<s:checkboxlist name="map"list="#{1:'aa',2:'bb'}" listKey="key"listValue="value" value="{1}"/>2 w' T ~; k7 [- |4 y: Y$ ^( ~* b
生成如下html代码:4 {( B' f- A* q! e( @( l( O
<input type="checkbox"name="map" value="1"checked="checked"/><label>aa</label>0 |& K) G6 M' Q( W
<input type="checkbox"name="map" value="2" /><label>bb</label>
p7 E0 z! b8 x+ f* `3 W当然集合里面存放的也可以是对象类型& d. X9 _& t- p5 |
/ Q/ \% I C6 r(6).单选框
) X: x" y" w W4 ^7 ]<s:radio name="beans"list="#request_person" listKey="personid"listValue="name"/>. f: l+ R' t O+ n4 ?
! U+ O. u# K& r6.struts2的处理流程与Action的管理方式1 H2 m/ ^: X6 w& U
(1).用户请求->(查看web.xml文件)StrutsPrepareAndExecuteFilter->Interceptor(struts2内置的一些拦截器或用户自定义拦截器)->Action(用户编写的Action类,类似Struts1中的action,针对每一次请求,都创建一个Action)->Result(类似struts1中的forward)->Jsp/html(响应)# @ Y: e+ j: s7 f# ]
/ P' D0 R" n' D2 {! S
(2).StrutsPrepareAndExecuteFilter是struts2框架的核心控制器,它负责拦截由<url-pattern>/"</url-pattern>指定的所有用户请求,当用户请求到达时,该fileter会过滤用户的请求,默认情况下,如果用户请求的路径不带后缀或者后缀以.action结尾,这时请求将被转入到struts2框架处理,否则struts2框架将略过该请求的处理,当请求转入struts2框架处理时会先经过一系列的拦截器,然后再到Action,与struts1不同,struts2对用户的每一次请求都会创建一个Action,所以struts2中的action是线程安全的.
! ]6 m4 T0 Q4 A7 c7 Q! I' D
# \# u" r" E& \$ S8 B; Q# H( U7.XML配置方式实现对action的所有方法进行校验, E7 h0 i+ Y& ?' I" ?
(1).基于XML配置方式实现对action的所有方法进行输入校验:) `) y3 ?( ~) r9 |
使用基于XML配置方式实现输入校验时,Action也需要继承ActionSupport,并且提供校验文件,校验文件和action类放在同一个包下,文件的取名格式为:ActionClassName-validation.xml,其中,ActionClassName为action的简单类名,-validation为固定写法,如果Action类为cn.itcast.UserAction,那么该文件的取名应为:UserAction-validation.xml,下面是校验文件的模板:
' u9 p- |+ J y$ J( m, t7 D0 Q$ p9 @<validators>
+ ~2 s" F C& r8 K" d% W<field name="username">
8 ^( o* K. D1 k! f* Q<field-validatortype="requiredstring">) f. t6 d* S- V
<paramname="trim">true</param>
* J% y7 O& K8 _5 ]; _1 o<message>用户名不能为空</message>9 y1 A7 S& `8 |3 S! x. r
</field-validator>
$ Q6 i9 i' {; b7 Y- Z4 O% H</field>$ A1 W* m! a+ F
</validators>
3 c/ U+ X9 y% [1 Y' s<field>指定action中要校验的属性,<field-validator>指定校验器,上面指定的校验器requirestring是由系统提供的,系统提供了能满足大部分验证需求的校验器,这些校验器的定义可以在xwork-2.x.jar中的com.opensymphony.xwork2.validator.validators下的default.xml中找到,<message>为校验失败后的提示信息,如果需要国际化,可以为message指定key属性,key的值为资源文件中的key,在这个校验文件中,对action中字符串类型的username属性进行验证,首先要求调用trim()方法去掉空格,然后判断用户名是否为空.
+ u3 ~7 I1 V2 _ 2 y o2 T. o9 r& W I. ~; |
(2).struts2提供的校验器列表:! [0 w) U5 W/ ]2 R" E* e3 m& y3 g
required(必填校验器,要求field的值不能为Null)2 c( c4 m5 T6 ` w9 E7 J) P2 d
requiredstring(必填字符串校验器,要求field的值不能为null,并且长度大于0,默认情况下会对字符串取钱后空格)
' @6 @1 I/ Q) r7 L6 L# mstringlength(字符串长度校验器,要求field的值必须在指定的范围内,否则校验失败,minLength参数指定最小长度,maxLength参数指定最大长度,trim参数指定校验field之前是否去除字符串前后的空格)9 u2 P7 j& V* p! V5 {
regex(正则表达式校验器,检查被校验的field是否匹配一个正则表达式,expression参数指定正则表达式,caseSensitive参数指定进行正则表达式匹配时,是否区分大小写,默认值为true)
6 D/ q& v! C* \int(整数校验器,要求field的整数值必须在指定范围内,min指定最小值,max指定最大值), \ \7 W* L0 v
double(双精度浮点数校验器,要求field的双精度浮点数必须在指定范围内,min指定最小值,max指定最大值)$ w# |! f o$ U; Q, K
fieldexpression(字段OGNL表达式校验器,要求field满足一个OGNL表达式,expression参数指定OGNL表达式,该逻辑表达式基于ValueStack进行求值,返回true时校验通过,否则不通过)8 G% T3 W2 K" ]
email(邮件地址校验器,要求如果field的值非空,则必须是合法的邮件地址)% L, G( Z6 n0 Z, c
URL(网址校验器,要求如果field的值非空,则必须是合法的URL地址)" h3 {. w% K, E, a
date(日期校验器,要求field的日期值必须在指定范围内,min指定最小值,max指定最大值)
2 z8 F' z' C! ~conversion(转换校验器,指定在类型转换失败时,提示的错误信息)
6 u0 v+ w# M/ L! ^ Ivisitor(用于校验action中的符合属性,它指定一个校验文件用于校验符合属性中的属性)
2 X( ^0 ]0 h* Y" ~2 sexpression(OGNL表达式校验器,expression参数指定ognl表达式,该逻辑表达式基于ValueStack进行求值,返回true时校验通过,否则不通过,该校验器不可用在字段校验器风格的配置中)
9 K S" K, A3 E9 D3 m" X4 ^+ Y
) o) c! g8 V% d" v& P(3).![CDATA[文本内容]]:文本内容不会被解析,只会原封不动的当做文本处理$ Q2 M+ r- m( ]/ S0 D E6 {
& j5 Q8 \! d) C# B0 ~6 R(4).编写校验文件时,不能出现帮助信息:
/ s/ h. ]8 {, x8 ~; f! R' M( M' q在编写ActionClassName-validation.xml校验文件时,如果出现不了帮助信息,可以按照下面方式解决:$ k- `3 `3 f: q, r7 P: ]+ H$ v
windows->preferences->myeclipse->filesand editors->xml->xmlcatalog:点击add,在出现的窗口中的location中选"file system"然后再xwork-2.1.2戒烟目录的src\java目录中选择xwork-validator-1.0.3.dtd,回到设置窗口的时候,不要急着关闭窗口,应把窗口中的Key Type改为URI,Key改为http://www.opensymphoney.com/xwork/xwork-validaor-1.0.3.dtd
4 X1 q. w! ?& _# \9 G% x. B 7 ]! ]$ u! a% v7 }3 Y
8.XML配置方式实现对action的指定方法校验
) J: r" e$ S/ D* _ B u, i7 x* p, F(1).基于XML配置方式对指定action方法实现输入校验:
8 S* V6 } e2 {8 \4 W当校验文件的取名为ActionClassName-validation.xml时,会对action中的所有处理方法实施输入校验,如果你只需要对action中的某个action方法实施校验,那么校验文件的取名应为:ActionClassName-ActionName-validation.xml,其中ActionName为struts.xml中的action的名称,例如:在实际应用中,常有以下配置:8 [6 L; y* ]! s3 Z ~
<action name="user_*"class="cn.itcast.action.UserAction" method="{1}">1 N* V2 Q0 W) h, W7 k+ P3 T' d5 [
<resultname="success">/WEB-INF/page/message.jsp</result>
+ U# z$ W: Q& b6 _% Y( v; P<resultname="input">/WEB-INF/page/addUser.jsp</result>! x4 z: l ]& B6 j0 k4 j- r
</action> y0 _/ Z1 l4 s5 J& T5 b- Q
UserAction中有以下两个处理方法:
2 i, q! |4 v, w+ opublic String add() throws Exception{
: h$ Z' A9 D9 k7 n
. s" p5 `! S+ B K X}3 h& I, q$ T. K4 l+ o5 @ P
public String update() throws Exception{: s. O/ g: g6 o7 M
1 |2 g6 L6 h( d g" C}' d1 ]" d V, u& e$ }- K! {
要对add()方法实施验证,校验文件的取名为:UserAction-user_add-validation.xml
( ?4 ~: D( {9 l* n2 w, Y要对update()方法实施验证,校验文件的取名为:UserAction-user_update-validation.xml4 T& G$ I$ j9 [) V) _( B
3 V5 W- ]6 s4 O(2).基于XML校验的一些特点:
& A% i1 s" N, ~9 R" D3 [, E当为某个action提供了ActionClassName-validation.xml和ActionClassName-ActionName-validation.xml两种规则的校验文件时,系统按下面顺序寻找校验文件:
8 m6 v v; W' ]5 r, QActionClassName-validation.xml m0 s* i, ~- [9 [3 S
ActionClassName-ActionName-validation.xml
: Q9 x+ O, R# ?( C- J2 p系统寻找到第一个校验文件时还会继续搜索后面的校验文件,当搜索到所有校验文件时,会把校验文件里的所有校验规则汇总,然后全部应用于action方法的校验,如果两个校验文件中指定的校验规则冲突,则只使用后面文件中的规则。
: _& L q- P3 W( l2 E$ }当action继承了另一个action,父类action的校验文件会先被搜索到' l# P6 F, f8 a. c( [6 I
假设UserAction继承BaseAction
5 x# F) x& N4 y% b5 z/ [2 A<actionname="user" class="cn.itcast.action.UserAction"method="{1}">
* e% s9 G7 ]9 c</action>4 Z2 v+ Z. W" d0 f( X# |
访问上面的action,系统先搜索父类的校验文件:BaseAction-validation.xml,BaseAction-user-validation.xml,接着搜索子类的校验文件:UserAction-validation.xml,UserAction-user-validation.xml,应用于上面action的校验规则为这四个文件的总和4 a+ f, o+ g) c, a0 f
& m5 r) L6 v0 F5 e3 } r
9.动态方法调用和使用通配符定义action5 K% _: W! X1 [4 ]
(1).在struts1中实现方法的动态调用:- \! c/ z" g% ^8 n6 ]
<actionpath="/control/employee/manage" type="....DispatchAction"parameter="method"/>9 `7 R0 H1 w0 n, ]. r* S2 ~
</action>
. A( `6 \& I7 S7 x4 G+ y/control/employee/manage?method=addUI' i& o! m0 h1 N) R0 |$ P
但是Action必须继承DispatchAction$ n6 {" T% F, d, q. E6 Q% J
4 l" o3 i7 B1 ?$ A
(2).在struts2中有两种方式:2 \8 W0 @' T- P5 Q5 j }3 Z
第一种:(struts2.1版本后就不建议使用了)是动态方法调用:如果Action中存在多个方法时,我们可以使用!+方法名调用指定方法,如下:
) m. k7 g* R4 u, K7 tpublic class HelloWorldAction{, |8 ]2 G' c) c8 Z. L# d& y
private String message;6 B% f0 A! Q, J) \2 ~ ?
..../ S4 S' Q: {: s% R4 l
public String execute()throws Exception{% u/ d' W- l4 N9 K$ W. e9 F
this.message="我的第一个struts2应用";
" S4 R# U) K, K}' b9 O) d8 x" i6 ^
public String other() throws Exception{- d: v+ J5 W1 |/ _" H" D' J
this.message="第二个方法";& q; M! A! G1 [- Y
return "success";
6 D) }0 T- Q4 |+ t}
; Y: ?! U; r/ j- R" ~0 m} M; h. \8 V9 l3 ?( j
假设访问上面的action的URL路径为:/struts/test/helloworld.action,要访问action的other方法,我们就可以这样调用:/struts/test/helloworld!other.action,如果不想使用动态方法调用,我们可以通过常量struts.enable.DynamicMethodInvocation关闭动态方法调用:' _2 o+ K8 Y+ e' j
<constantname="struts.enable.DynamicMethodInvocation"value="false"/>
3 P$ K; i! k* r& {1 c7 `第二种:使用通配符定义action(推荐使用的)
0 M, ?# F, n4 l3 m- K<package name="itcast"namespace="/test" extends="struts-default">
4 S; \' c. L- u% c9 E<action name="helloworld_*"class="cn.itcast.action.HelloWorldAction" method="{1}>
4 a9 q2 h5 j" u; X' ]( h p+ s<resultname="success">/WEB-INF/page/hello.jsp</result>5 m. y* ^. c4 @3 k0 m3 k1 W
</action>
1 j' x& e, V s. }6 H2 W3 E</package>- l+ }, @, ~' R
public class HelloWorldAction{: u8 k7 F0 b2 a: A
private String message;
2 A( f3 q h' e....5 m5 w$ k# q3 [& t7 ]3 d
public String execute()throws Exception{9 M1 D4 p( }: I, T1 J9 P) x3 j
this.message="我的第一个struts2应用";5 L7 _$ z# v9 ~8 A$ h
}: `0 I* W" E7 r# e0 _
public String other() throws Exception{7 Q7 ]/ C1 I) W( d4 U
this.message="第二个方法";
( B- T0 F, @8 R4 S, o; g" T: K1 wreturn "success";
; E" G5 H% Y; L* @6 g; R5 ]}% U8 S. j) W7 N; N! H
}
9 q2 f6 v: @# N' `+ X3 P要访问other()方法,可以通过这样的URL访问:/test/helloworld_other.action
/ f4 g: q% k. s! \1 i) o, Uname="helloworld_*"后可根据多个*,method={1},'1'表示匹配*的位置9 V) C- s8 ~# E, M7 K
name="helloworld_*_*",method={2}:要访问other()方法,可以通过这样的URL访问:/test/helloworld_xxx_other.action
* {3 X; k6 t4 D) R% Z A5 m/ A
! d! x5 A% B& k6 {* ^. X10.对action指定的方法进行校验
9 G" r" V \$ i9 z手工编写代码实现对action指定方法输入校验:0 ?( H: R6 P/ Z5 w6 n( J/ j: S. j! d
通过validateXxx()方法实现,validateXxx()只会校验action中方法名为Xxx的方法,其中Xxx的第一个字母要大写,当某个数据校验失败时,我们应该调用addFieldError()方法往系统的fieldErrors添加校验失败信息,(为了使用addFieldError()方法,action可以继承ActionSupport(),如果系统的fieldErrors包含失败信息,struts2会将请求转发到名为input的result,在input视图中可以通过<s:fielderror/>显示失败信息.
' N( S2 E& n: S4 RvalidateXxx()方法使用例子:1 Y% e- V3 _) Z! @ q
public Stringadd() throws Exception{return "success";}
' y7 n% M+ D& f7 Y* Rpublic voidvalidateAdd() {, [2 @2 j$ _) Y* {" i F
if(username==null&&"".equals(username.trim()))this.addFieldError("username","用户名不能为空");
+ s& _6 _) z& W' B7 G}! Z5 f+ U+ N3 L6 M' G
验证失败后,请求转发至input视图:<resultname="input">/WEB-INF/page/addUser.jsp</result>
5 a8 X6 m0 f$ p7 }1 b# G在addUser.jsp页面中使用<s:fielderror/>显示失败信息
! j! X$ ?0 ^- y9 q" G- B; g# F" d 8 i5 r* `- x0 J" b) A' M+ s# p
11.对Action中所有方法进行输入校验7 g. c0 A2 W/ V# A2 n% i9 R
(1).在struts2中,我们可以实现对action的所有方法进行校验或者对action的指定方法进行校验& S% M1 R1 M2 m& e" A
( U& z( m1 c5 s# g- w(2).对于输入校验struts2提供了两种实现方法:一种是采用手工编写代码实现,另一种是基于XML配置方式实现
+ w& b H" L0 g3 J
: T' O) e8 c; C8 t0 A(3).手工编写代码实现对action中所有方法输入校验:通过重写validate()方法实现,validate()方法会校验action中所有与execute方法签名相同的方法,当某个数据校验失败时,我们应该调用addFieldError()方法往系统的fieldErrors添加校验失败信息(为了使用addFieldError()方法,action可以继承ActionSupport(),如果系统的fieldErrors包含失败信息,struts2会将请求转发到名为input的result,在input视图中可以通过<s:fielderror/>显示失败信息.4 I( Z2 `8 m2 G+ C. }; h# n7 q
validate()使用例子:+ g# r" U3 h' e& s7 }4 ^$ P" s8 A+ J) T
public voidvalidate(){ o- f' f/ K7 r- ?. k8 G: h
if(this.mobile==null||"".equals(this.mobile.trim())){this.addFieldError("username","手机号不能为空")}else{if(Pattern.compile("^1[358]\\d{9}").matcher(this.mobile.trim()).matchers()){this.addFieldError("mobile","手机号的格式不正确");}}
( K5 q2 \3 | D3 j7 ^7 M7 p! R}/ E6 E# @3 Y+ i, e; R @* F* n
验证失败后,请求转发至input视图:+ Q1 c4 |% y: n. ]* F7 @0 U. n
<resultname="input">/WEB-INF/page/addUser.jsp</result>
( z2 Z; H6 l: p7 `% f在addUser.jsp页面中使用<s:fielderror/>显示失败信息" l* [3 a" B8 H2 M
4 d. V1 I( A" B: L$ e; T* h
12.多文件上传- g+ B5 ?- Q+ A- x* C+ S+ ^& n
(1).多文件上传,就是在一个文件上传的基础上,将属性File变成数组类型File[]类型即可,同时该字段的名称必须要和上传页面中属性name的名称一样.
M/ E- @$ Q$ y4 N1 \7 y然后进行一次迭代,就可以得到所有的文件! V t; v+ r; s
% V% R& M. S: n13.防止表单重复提交
3 o2 ]) N) l5 ~% V( M$ R- b( n. ^<s:token>标签防止表单重复提交
1 }2 [, m2 _; Z: q# C1 S1 z+ s第一步:在表单中加入<s:token/>
- y5 R5 M' a: s+ k! w<s:form action="helloworld_other"method="post" namespace="/test">
, V) B; j% S& V6 [7 r<s:textfieldname="person.name"/><s:token/><s:submit/>3 G2 q2 C5 v% A: v9 H' L, r8 K1 W
</s:form>
+ i: h4 G+ {: j, z$ G1 \第二步:<action name="helloworld_*"class="cn.itcast.action.HelloWorldAction" method="{1}">
: ^, P5 \1 g$ o) F" A<interceptor-refname="defaultStack"/> l$ p# M6 n$ X& ?5 o* ~
<interceptor-refname="token"/>
1 {: ~' a2 D m, e<resultname="invalid.token">/WEB-INF/page/message.jsp</result>
* Z/ s/ q+ o1 @ g; q; g: y5 U( e<result>/WEB-INF/page/result.jsp</result># X0 ^3 W' m# N5 f) S
</action>! T& R/ p* M# `1 U; k% j
以上配置加入了"token"拦截器和"invalid.token"结果,因为"token"拦截器在会话的tlent与请求的token不一致时,将会直接返回"invalid.token"结果. Q0 k& E" a- y9 ~" d0 ^3 M
在debug状态,控制台出现下面信息,是因为Action中并没有struts.token和struts.token.name属性,我们不用关心这个错误
( Q. k+ o- V' c( ^" v/ g使用了<s:form/>标签可以不指定action的上下文标签路径,可以通过命名空间实现.和前面的原理是一样的,在路径后面添加上sessionid号,只是这步操作不需要我们自己得到sessionid号,struts帮我们操作.
' n* _0 M9 }3 x% U' l在值栈中的对象,访问无需添加'#'
+ {0 k* P4 ?" v
2 J5 |, O* w5 G6 t& @$ n4 i14.访问或添加几个属性
B8 g5 a" }- W6 n1 o$ q) `: K, M(1).访问或添加request/session/application属性,在struts2中的Action中的execute方法中没有Servlet api(没有响应的参数);9 O3 y* r- ]1 d2 y; I9 g. V
public String scope() throws Exception{* x6 ?0 O3 Q" z
ActionContextctx=ActionContext.getContext();
; k/ B: f, h3 A0 Rctx.getApplication().put("app","应用范围");
4 o2 G3 G* x6 u3 h. d- b2 M. I& pctx.getSession().put("ses","session范围");
e( [5 E! g6 D# nctx.put("request","request范围");
! t7 W/ ]% x# W8 |6 greturn "scope";
$ v$ R7 H% P6 X* A- w8 o$ \}4 |5 b. X, x2 S% C
1 K8 I: v4 D2 i# u7 S
<body>
& n/ Y3 C$ A/ E7 ~${applicationScope.app}3 F7 c: r+ K3 h9 a8 K
${sessionScope.ses}6 \% R! _* `; Z( z0 l& Z. L
${requestScope.req}3 v2 n; V; w0 b& J3 p
</body>
4 c5 C( }4 H# m* W
- s4 y% a1 d5 M: i& P4 O/ ~(2).获取HttpServletRequest/HttpSession/ServletContext/HttpServletResponse对象:
" @5 h( D) J4 D方法一:通过ServletActionContext类直接获取
* E1 a. s! T: g8 a6 ^5 Bpublic String rsa() throws Exception{
& W8 D4 @, R. O* mHttpServletRequest request =ServletActionContext.getRequest();
: I! C* s, e# W( O) k% ?( [ServletContextservletContext=ServletContext.getServletContext();
# U/ u$ P6 N5 I prequest.getSession();
$ j o; v6 {( y" G1 e- \3 V eHttpServletResponseresponse=ServletActionContext.getResponse();
7 j) c/ m5 f F- _- { m" ^$ Kreturn "scope";( G' b2 W9 g, D4 s+ Y2 ~4 g1 _
}. G* s1 {" H2 r* q
方法二:实现指定接口,由struts框架运行时注入: i# T7 J0 {1 u, @5 p, ?+ r3 N j! k3 U- G
public class HelloWorldAction implementsServletRequestAware,ServletResponseAware,ServletContextAware{6 U% {; ?$ C# {# Q# E* l
private HttpServletRequest request;8 F2 j- ^; q8 V# p' [7 _1 }$ P
private ServletContext servletContext;
/ \2 ^+ e3 i# I1 ]6 j4 @1 N1 Lprivate HttpServletResponse response;1 }6 p$ H; d8 J8 R; _7 s) f+ l
public voidsetServletRequest(HttpServletRequest req){
" Y# y' I0 P3 @this.request.req;5 D! X0 \( [" G
}" `: ]- V2 v* M" a$ m0 s" }7 T/ g6 C2 w
public voidsetServletResponse(HttpServletResponse res){ W+ V; m5 z/ |4 ~& P& o) c6 }9 E4 X
this.response=res;
4 Z/ Q0 H" O& B+ Q" f}7 k; _, Q2 _( P0 c; f3 g- q
public voidsetServletContext(ServletContext ser){
0 m+ p7 U7 d l+ E& g7 }. ?this.servletContext=ser;
& n2 F" ?( L1 U8 G$ D, V+ o0 i: v9 w$ b}
2 K/ i0 C& D" M8 C% E) ?注意1和2的不同,一个不需要得到对象,一个需要得到对象,所以要区分两个的应用场景5 L( _( s- k: S- J$ N) n
}/ U# B# x1 ~5 p+ s( d
7 e" L6 i" ?) [1 x" L: P8 w- ]* u
15.解决struts配置文件无提示问题3 j3 ?' N1 D- t& W. }5 m: {( N: \8 L
找到struts2.0.dtd文件即可,windows->preferences->MyEclipse->XML->XMLCatalog,点击添加strut2.dtd3 F5 k2 V4 R) Z
1 o/ G( q) \0 u- F9 { o( [
16.介绍struts2及struts2开发环境的搭建+ b/ `/ w( c) Y+ y( o- N
(1).struts2是在webwork2基础发展而来的,和struts一样,struts2也属于MVC框架,不过有一点大家需要注意的是:尽管struts2和struts1在名字上的差别不是很大,但是struts2和struts1在代码编写分割上几乎是不一样的,那么既然有了struts1,为何还要推出struts2,主要是因为有一下有点:- P2 ?0 I1 D9 w& `: x. p
第一:在软件设计上struts2没有像struts1那样跟Servlet api和struts api有着紧密的耦合,struts2的应用可以不依赖于servlet api 和struts api,struts2的这种设计属于无侵入式的设计,而struts1却属于侵入式设计,因为其的
; S% y0 x) _3 Bexecute()方法中的参数为ActionMapping,ActionForm,HttpServletRequest,HttpServletResponse
7 {" r& a' M! ]. p( d0 k' m) v第二:struts2提供了拦截器,利用拦截器可以进行AOP编程,实现如权限拦截等功能
2 W& \9 s+ L* j* ]$ W6 g: M第三:struts2提供了类型转换器,我们可以把特殊的请求参数转换成需要的类型,在struts1中,如果我们要实现同样的功能,就必须向struts1的底层实现BeanUtils注册类型转换器才行# u5 M& r: T2 k. p6 A) P2 E
第四:struts2提供支持多种表现层技术,如:JSP,freeMarker,Velocity等
$ G4 {% J$ l7 q X2 b5 W第五:struts2的输入校验可以对指定方法进行校验,解决了struts1长久之痛,struts1中的validate方法对所有的方法进行校验/ I9 o4 U# J$ ]; Y. T, q4 W9 Q* f
第六:提供了全局范围、包范围、和Action范围的国际化资源文件管理实现.9 h" L+ X e/ Q! [7 M6 a
& A; U) g, J! g/ P+ O- D4 Q
(2).搭建struts2的环境和struts1是相同的,第一步导入相关包,第二步建立struts2的配置文件,第三步在web.xml中注册struts2框架的配置
( m8 t8 B$ W2 D3 N
$ V1 C; x, i1 _) A(3).所需的包:struts2-core-2.x.x.jar,xwork-2.x.x.jar(webwork的核心架包),ognl-2.6.x.jar& Q; J: W0 f% Z1 o2 e
9 @' b1 k0 O7 [1 K8 l1 D* i# |(4).struts2默认的配置文件为struts.xml,该文件需要放在/web-inf/classes目录下8 E0 A; S* ]; c
: s2 ^1 \0 J+ e6 n1 Q8 F: k
(5).在struts1中,struts框架是通过servlet启动的,在struts2中,struts框架是通过Filter启动的,它在web.xml中的配置如下所示:可以参照struts文件夹下的例子中拷贝,在strutsperpareExecuteFilter的init()方法中将会读取类路径下默认的配置文件struts.xml完成初始化操作,注意:struts2读取到struts.xml的内容后,以javabean形式存放在内存中,以后struts2对用户的每次请求处理将使用内存中的数据,而不是每次都读取struts.xml文件8 c# J5 k( o; m$ {
# n$ x( L! ]# c; U- I
(6).自从struts2.1.3以后,下面的FilterDispatcher已经标注为过时了,struts2.1.3后期版本为StrutsPrepareAndExecuteFilter类2 w5 P1 q! l3 K
* O9 K7 l, L& `) Q
17.开发第一个应用
' P5 P) ]) Y6 u+ q7 j7 E(1).在struts.xml中的配置:
" f4 f) q/ w7 H<package name="itcast"namespace="/test" extends="struts-default">
0 Z1 r5 c& f V6 Q <actionname="helloworld" class="cn.itcast.action.HelloWorldAction"method="execute">
1 g) k3 M! P! V# @1 f( x V <resultname="success">/WEB-INF/page/hello.jsp</result>
8 q) L& p3 b( r</action>" v; q+ M5 y4 s8 e G( r1 u
</package>
# Q2 V1 V o- [; ]! q" u0 q在struts2框架中使用包来管理Action,包的作用和Java中的类包是非常类似的,它主要用于管理一组业务功能相关的action,在实际应用中,我们应该吧一组业务功能相关的Action放在同一个包下
' F5 s5 Y6 y* y配置包时必须指定name属性,该name属性可以任意取名,但必须唯一,它不对应java的类包,如果其他包要继承该包,必须通过该属性进行引用,包的namespace属性用于定义该报的命名空间,命名空间作为访问该包下Action的路径的一部分,如访问上面例子的Action,访问路径为:/test/helloworld.action,namespace属性可以不配置,对本例而言,如果不指定该属性,默认的命名空间为" "(空字符串).当然配置可以减少重复的代码,struts1中的重复代码就可以使用命名空间来解决' v/ \# R4 n1 I) V
通常每个包都应该继承struts-default包,因为struts2很多核心的功能都是拦截器来实现的,如:从请求中把请求参数封装转到action、文件上传和数据验证等都是通过拦截器实现的,struts-defaul定义了这些拦截器和Result类型,可以这么说:当包继承了struts-default才能使用struts2提供的核心功能,struts-default包是在struts2-core-2.x.x.jar文件中的struts-default.xml中定义,struts-default.xml也是struts2默认配置文件,struts2每次都会自动加载struts-default.xml文件,包还可以通过abstract="true"定义为抽象包,抽象包中不能包含action,可以查看struts-default.xml文件中,就可以看到定义了很多拦截器
9 M2 j2 D6 q* p/ ^$ T k<result></result>和struts1中的forward很相似,定义视图
4 K6 K- \6 y6 ]8 a" A+ Y U
- @* `$ E+ |( }6 ^(2).public Stringexecute(){return 视图的名称;}注意到这个方法和struts1不同,没有参数,返回类型也不同,这就降低了耦合性,非侵入式的编程了.
4 m1 S/ A9 m2 W& N4 @ ) K4 T1 Y* g2 z: E0 o8 I; ?
(3).在jsp中使用el表达式即可${message},message是Action中的一个方法getMessage()方法,而不是根据Action中的成员变量message4 O. n* ]( E Y, K r4 t6 ~' N
. ?2 J r4 h0 k: m9 |18.配置Action范围国际化资源文件4 E- ^: o! i# I
(1).我们也可以为某个action单独制定资源文件,方法如下:在Action类所在的路径,放置ActionClassName_language_country.properties资源文件,ActionClassName为Action类的简单名称当查找指定key的消息时,系统会先从ActionClassName_language_country.properties资源文件查找,如果没有找到对应的key,然后沿着当前包往上查找基本名为package的资源文件,一直找到最顶层包,乳沟还没有找到对应的key,最后会从常量struts.custom.i18n.resources指定的资源文件中查找
% o a% ^, }1 p' A" u+ T( W
4 t2 W. G& t: `8 Z(2).JSP中直接访问某个资源文件1 G# A- a, p7 h
struts2为我们提供了<s:i18n>标签,使用<s:i18n>标签我们可以在类路径下直接从某个资源文件中获取国际化数据,而无需任何配置:
0 A% `6 |2 Q9 o. u3 @<s:i18n name="itcast">
8 N; T- m# t4 }<s:text name="welcome"/>
/ E9 u1 Z8 z6 v7 ^7 v- X# A( {) O</s:i18n>
/ k7 u+ x6 l# L0 ?3 S# R7 ?7 ritcast为类路径下资源文件的基本名
, @9 D" {7 K; ?! H' U& n2 m如果要访问的资源文件在类路径的某个包下,可以这样访问:+ v0 q* E/ f9 O% r4 y+ g, s- k0 u
<s:i18nname="cn/itcast/action/package">
+ l! Q% V Q. i' _; R1 H" [/ M<s:text name="welcome">, g, a4 Q& ?- E
<s:param>小张</s:param>
- i5 n2 h# U' m* \5 T& E5 ~9 F" F; }</s:text>
* K' f0 k5 G: J4 |$ o! y! p</s:i18n>
* t: g. d* m8 ]6 j4 N上面访问cn.itcast.action包下基本名为package的资源文件
2 b$ P# \+ H6 @# \, S$ W+ Z
: E$ T' h) E1 H, F h6 M& a1 F19.配置包范围的国际化资源文件2 \% K" K8 j# b* k6 Y2 P/ x, H
(1).在一个大型应用中,整个应用有大量的内容需要实现国际化,如果我们把国际化的内容都放置在全局资源属性文件中,显然会导致资源文件变得过于庞大、臃肿,不便于维护,这个时候我们可以针对不同模块,使用包范围来组织国际化文件
# G# f' k$ p) ^" ] K# q' P2 h方法如下:在java的包下放置package_language_country.properties资源文件,package为固定写法,处于该包及子包下的action都可以访问该资源,当查找指定key的消息时,系统会先从package资源文件中查找,当找不到对应的key时,才会从常量struts.custom.i18n.resources指定的资源文件中寻找.& L* K0 Q6 B; \! T: [) a, b3 R# v9 N: c
) Y' H2 a, s( s, j! F20.配置国际化全局资源文件、输出国际化信息
1 f$ a- T2 G5 i% | `1 G* _(1).准备资源文件,资源文件的命名格式如下:5 D* `2 ^" U6 @% \
baseName_language_country.properties, ]4 R7 _1 R! r4 k- ?0 f/ F% T
baseName_language.properties
$ ]$ }0 ?3 b" h& z1 X3 [baseName.properties; y @! G4 j5 m6 n
其中baseName是资源文件的基本名,我们可以自定义,但是language和country必须是java支持的语言和国家。如:2 X8 _, Y- J+ P( \
中国大陆:baseName_zh_CN.properties
. c% Z% X/ c3 [. p; e美国:baseName_en_US.properties
2 T# v: B5 X3 _. } ^/ C% T % s( t. L2 x" n9 @
(2).现在为应用添加两个资源文件:
4 k5 y1 L1 L; m' ^. v第一个存放中文:itcast_zh_CN.properties
4 _9 B V4 p+ S% a3 w; H2 ~6 G& c内容为:welcom=欢迎来到传智播客
* }8 q5 m' W4 T2 o第二个存放英语(美国):itcast_en_US.properties
* p% A0 N; F' c% _) U内容为:welcome=welcom to itcast
0 Q+ L9 E2 R0 P2 L( [ ; u( `; j! F8 a+ z# P8 K# x4 N N" J
(3).对于中文的属性文件,我们编写好后,应该使用JDK提供的native2ascii命令把文件转换为unicode编码的文件,命令的使用方式如下:! A6 |% F" l3 W! R( H* Q
native2ascii 源文件.properties 目标文件.properties,在MyEclipse6.6版本以及后面的版本会自动转换.
; x: X( p4 O+ C" u0 ?5 ~8 A
2 b- Z' w6 _! i8 _# b# o(4).struts2有:全局范围,包范围,action范围的资源文件5 c9 r' y ~9 q4 G
6 f' q, x, D( j) P1 G
(5).配置全局资源与输出国际化信息:
# j7 C S' r8 W. x8 }% m. }当准备号资源文件之后,我们可以在struts.xml中通过:struts.custom.i18n.resources常量把资源文件定义为全局资源文件,如下:
B$ r; w. R: L H! h& }' j) _, W+ e<constantname="struts.custom.i18n.resources" vlaue="itcast"/>* B( f) ]' s+ }$ [
itcast为资源文件的基本名
- r4 d$ n' y+ e( e7 g# y后面我们就可以在页面或在action中访问国际化信息:
* r' X8 {3 U! z5 {* v6 O在JSP页面中使用<s:text name=""/>标签输出国际化信息:
2 {3 s3 v( E( E2 N8 y" n<s:textname="user"/>,name为资源文件中的key
; n4 S$ V, ]8 \5 u; m8 Z9 Q) o# t在Action类中,可以继承ActionSupport,使用getText()方法得到国际化信息,该该方法的第一个参数用于指定资源文件中的key,
% ^& P/ n) b) N" m在表单标签中,通过key属性指定资源文件中的key,如:' K$ L- V3 l o( `- x1 t2 t3 S7 |6 a
<s:textfieldname="realname" key="user"/>2 T' m7 E6 }* y; B0 N; t
0 K' I+ S) y5 j- n( E& v
21.请求参数接受
1 o8 d( E+ d" E/ z(1).struts1中是使用ActionForm接受用户的请求参数" O3 P2 r. j. _
5 e0 ?: P4 Z. v' [9 y
(2).采用基本类型接受请求参数(get/post):& R; z; Q- ?6 z* o8 Y
在Action类中定义与请求参数同名的属性,struts2便能自动接受请求参数并赋予给同名属性:请求路径:http://localhost:8080/test/view.action?id=78
' V6 Q6 m+ t3 R: Q* |public classProductAction{+ r8 Q9 @. i1 B5 Q5 }6 E& Y1 G
private Integerid;# z/ V" t) f$ B4 v: q8 ^
public voidsetId(Integer id){//struts2通过反射技术调用与请求参数同名的属性的setter方法获取请求参数值
5 X; Z( [# X$ A) m/ L, n+ Ethis.id=id;: d% o% g' [ p3 i
}
/ |0 v& y7 L U1 }4 Z! a" O2 C$ ppublic IntegergetId(){return id;}
j2 w$ C$ Y8 V( n/ d}
( j8 _4 B" r* ]9 F5 }; q # A! Q5 a0 O/ Q& Z
(3).采用复合类型接受请求参数
+ [0 Z- e: a6 |; k请求路径:http://localhost:8080/test/view.action?product_id=78
4 V+ ?8 {6 J" g! rpublic class ProductAction{: n8 J- Q8 H" K5 T% m
private Product product;6 V$ `+ Q7 b& s, n+ F9 I
public void setProduct(Product product){htis.product=product;}. ~5 t* `4 _" G1 D. g
public Product getProduct(){returnproduct;}8 S& v7 Q* R, n; D$ [
}+ K( d5 C% N5 E( q# `2 N7 _
struts2首先通过反射技术调用Product的默认构造器创建product对象,然后再通过反射技术调用product中与请求参数同名的属性的setter方法来获取请求参数值* A' n4 ~! b: \( a
4 l) {) q& o0 m( L% J/ _+ d(4).关于struts2.1.6版本中存在一个Bug,及接受到的中文请求参数为乱码(以post方式提交),原因是struts2.1.6在获取并使用了请求参数后才调用HttpServletRequest的setCharacterEncoding()方法进行编码设置,导致应用使用的就是乱码请求参数,这个Bug在struts2.1.8中已经解决,如果你使用的是struts2.1.6,要解决这个问题,你可以这样做:新建一个Filter,把这个Filter放置在Struts2的Filter之前,然后再doFilter()方法中添加以下代码:0 {2 [# e# G4 }' f- Y( G
public void doFilter(..){1 _6 j7 z6 l3 J: U2 Q: V7 ]% ?( t
HttpServletRequest req=(HttpServletRequest)request;8 f! e, Q4 N( V( Y! [" U7 P6 @$ p2 F1 l& `
req.setCharacterEncoding("UTF-8");3 y& k7 u$ Q+ G+ m% G3 o
filterchain.doFilter(request,response);9 e% z4 U3 t+ q* T% `. H% G1 G
}1 f9 h2 b# L- T
% p& }2 U' J+ N4 B22.全局类型转换器
- _6 b( m5 s O- Y自定义全局类型转换器:将上面的类型转换器注册为全局类型转换器:在WEB-INF/classes下放置xword-conversion.properties文件,在properties文件中的内容为:待转换的类型=类型转换器的全类名
+ r3 a! V0 v' @5 v/ k. ~% g对于本例而言,xwork-conversion.properties文件中的内容为:
- ], \. {: d. y& [3 N5 \java.util.Date=cn.itcast.conversion.DateConverter
' O0 Z! y5 {+ M9 I. o $ a5 V( E4 d* o& u
23.输出带有占位符的国际化信息
# d, p8 y0 b+ t$ y(1).资源文件中的内容如下:
: o' u6 _+ P- gwelcom={0}欢迎来到传智播客{1}
. j* d+ E' W: I2 N在jsp页面中输出带占位符的国际化信息, B0 o) X, y2 W) t
<s:text name="welcom">' \/ n J8 W2 }3 Z* \5 b _
<s:param><s:propertyvalue="realname"/></s:param>
7 _, J8 Y; `' l' ~: T, U<s:param>学习</s:param>
7 g$ t4 P3 s, A) ~4 E# |</s:text>9 \0 Q4 p, x2 Z6 \' l5 X
在Action类中获取带占位符的国际化信息,可以使用getText(String key,String[] args)或getText(StringaTextName,List args)方法.
* \, a o- d9 ?, I: [0 g; P , q# [ C6 L7 B4 B
(2).占位符就当是一个变量参数,可以传递给定的参数值.: D: L, u T+ P9 f, s/ w2 u
4 ?5 Z4 }* s/ ~0 h: E! ]
24.为Action属性注入值
! v4 a& m$ F8 Q! ~struts2为Action中的属性提供了依赖注入功能,在struts2的配置文件中,我们可以很方便的为Action中的属性注入值,注意:属性必须提供setter方法,: f* M F; Z% E+ w5 H
public class HelloWorldAction{2 k6 M6 p7 Z; C% k7 ]6 u
private String savePah;
' ]5 p/ L L) b4 `# p6 a# }1 ? public String getSavePath(){* w( p( O5 C1 Z! y5 i6 Q _* Q
return savePath;
% i" M; N; a, O2 ?0 d" \ }
- b6 @4 O G F# ` public void setSavePath(String savePath){
9 K; b' A7 S9 V0 @& _ B) z4 Z% k this.savePath=savePath;
5 z+ v) s* v; `! k5 g }
# W" z" ?8 E, r) i$ e}
; r) q( z1 u4 } H( y$ t7 a<package name="itcast"namespace="/test" extends="struts-default">( {. I9 ` x$ P7 d! _
<action name="helloworld"class="cn.itcast.action.helloWorldAction">
& ]' ]3 P; j$ W1 D<paramname="savePath">/images</param>
j. R$ w' `: H% l T# D+ ^3 [<resultname="success">/WEB-INF/page/hello.jsp</result>' G7 a: v; q0 _8 `, j. r6 P+ Y! m
</action>$ a4 o' t) Y3 m L, {; d
</package>
: W! x- X! {. f6 h. ]; G$ i5 Q上面通过<param>节点为action的savePath属性注入"/images";Action的变量的值,不能写死,经常变换,需要通过配置来设置参数4 K3 V: ]8 @- V# x1 d' z7 T; e0 U
! ]' e8 ?( E2 O5 [& \
25.为应用指定多个配置文件% |" m5 n( n5 e6 S, M
(1).在大部分应用中,随着应用规模的增加,系统中的Action的数量也会大量增加,导致struts.xml配置文件变得非常臃肿,为了避免struts.xml文件过于庞大、臃肿,提高struts.xml文件的可读性,我们可以讲一个struts.xml配置文件分解成多个配置文件,然后再struts.xml文件中包含其他配置文件,下面的struts.xml通过<include>元素指定多个配置文件:6 w! r. w4 X K* `! Z
<struts>4 S1 P3 f4 }4 Z! D, T' X9 S! m
<includefile="struts-user.xml"/>2 Q$ X8 V% C2 z- ?
<includefile="struts-order.xml"/>$ [5 _7 h* D# V! A# e5 o7 @
</struts>
d2 N" S9 _ A7 a, J通过这种方式,我们就可以将struts2的Action按模块添加在多个配置文件中
" }9 e6 q* K4 ?/ Z 7 ?+ Y8 `8 V2 S! u
26.文件上传 e4 b6 j' F2 l3 X/ J9 h0 l" P4 H
第一步:在WEB-INF/lib下加入commons-fileupload-1.2.1.jar、commons-io-1.3.2.jar,这两个文件可以从http://commons.apache.org下载,在struts2.1以前的版本需要添加,以后的版本就不需要添加+ M5 k v M/ U9 G7 u6 s; @8 v
" `* |- e3 l9 ^4 P+ c1 x. n; }第二步:把form表的enctype设置为:"multipart/form.data",如下:
8 R- H0 j& m6 N" u3 |6 Z<form+ I% P/ F* e$ a$ h" X/ M
enctype="multipart/form-data"action="${pageContext.request.contextPath}/xxx.action"method="post"># x3 d% X ?4 M3 v+ z
<inputtype="file" name="uploadImage">这个属性的name必须要和类中File名称一样
$ |" _0 U6 U0 Q: p6 d U</form>
& m' e. p6 Q7 E" ~6 O第三步:在Action类中添加以下属性9 M2 Y" B2 v' |3 m1 D( R: W( I4 n
public class HelloWorldAction{0 s+ p* Y; p3 m7 I* E2 L
private File uploadImage;//得到上传文件;
4 q: c0 L$ Q- g: g# N# f! mprivate String uploadImageContentType;//得到文件的类型
\6 g7 m. L& j7 h6 P" X) n9 E; n6 d3 D; vprivate String uploadImageFileName;//得到文件的名称
8 O& p6 b/ j4 y& @3 _3 i- F//这里省略了属性的get/set方法(但是要注意get/set方法是必须的)! |$ Z0 t. z3 @3 C
public String upload() throws Exception{- h* g, f% y- ^
String realpath =ServletActionContext.getServletContext().getRealPath("/images");
$ O* U9 d8 G& }/ D- p0 YFile file=new File(realpath);" o2 O0 |2 t8 C8 D5 n* c6 H
if(file.getParentFile().exists())file.getParentFile().mkdirs();//目录是否存在,不存在就创建% Y& ~; s' K9 e; W/ [; G
FileUtils.copyFile(uploadImage,newFile(file,uploadImageFileName));0 c" h; X% V# }( s) l# R
return "success";
; C |$ R8 S% @6 a* T}
2 x7 ?! `1 _7 u' W}# f3 B; ] P9 H& }
4 [4 R: `& t/ ^8 ]+ I8 W
(1).如果文件不保存,struts2会把文件保存到自己的目录中,但是当这个Action执行完后,该文件就会被删除,所以我们要将上传的文件保存到硬盘上
6 j5 r* b" n5 Y: @! E
2 q* }5 X4 c" ^! W, O" t(2).最好还要判断以下,文件uploadImage是否为空- J' v2 M# a5 t. j
4 z' c" b# T5 T, J' F, a2 }(3).如果上传大的文件,web都会失败,像一些门户网站上传视频,都是通过一个插件,可以把这个插件看成一个程序,只是这个程序是通过Socket变成的,针对一个端口进行传输数据 N* L( J4 o7 E4 r% C+ `% h- b
! J& S* y: t, r6 ?- R2 `$ `" T! V
27.指定struts2处理的请求后缀3 D( | O0 n. `0 U8 f M0 m
(1).前面我们都是默认使用.action后缀访问Action,其实默认后缀是可以通过常量"struts.action.extension"进行修改的,例如:我们可以配置struts2只处理以.do为后缀的请求路径
7 D1 B% O! m: u4 B- \( G% ]7 ?1 o<struts>) s, T( b. a; ]3 U4 v
<constantname="struts.action.extendsion" value="do"/>& g/ u& Q+ c7 U% ]
</struts>
4 I- c$ E4 _0 p. }7 r( i+ O如果用户需要制定多个请求后缀,则多个后缀之间以英文逗号","隔开,如:
; j' s b5 N8 C9 x0 j<constantname="struts.action.extendsion" value="do,go"/>& A" f) j0 C0 P# @ @' I
& h" s& L' w" @4 T0 Q
(2).常量可以在struts.xml或struts.properties中配置,建议在struts.xml中配置,两种配置方式如下:
- e+ L: Q5 Q0 A d/ h% w在struts.xml文件中配置常量:
6 G6 E! A) W. p7 Y<struts>5 s8 m5 I* P: V0 Y- C9 p7 m
<constantname="struts.action.extendsion" value="do"/>
7 v0 [3 q2 i1 Q1 q3 B. e/ ]- C7 _</struts>; @- G0 T1 u- Y! g
在struts.properties中配置常量:
# X6 |- z. Y) ?0 Mstruts.action.extension=do; V' g9 |% v, |% m/ D
因为常量可以在下面多个配置文件中进行定义,所以我们需要了解struts2加载常量的搜索顺序:
( f& V5 ~; I& b* Wstruts-default.xml8 m* Q# J$ s$ B. A0 ^
struts-plugin.xml
4 o/ T) ?! B7 @. g! w# d" vstruts.xml
( a* x9 ^3 O& P f2 o+ fstruts.properties
1 L/ h% j5 h# Z3 wweb.xml% U+ B7 s# q) g; x$ @& N! X
如果在多个文件中配置了同一个常量,则后一个文件中配置的常量值会覆盖前面文件中配置的常量值
2 Q) R8 v3 o! `; A7 h& W" p
2 y2 E: e3 _& y/ L0 d(3).* t, L k% i. n
第一:默认编码集,作用于HttpServletRequest的setCharacterEncoding方法和freemarker、velocity的输出:
! h, P0 Q3 |: j. i+ w% c3 F2 o<constantname="struts.i18n.encoding" value="UTF-8"/>
( i) Z8 i, X1 ~8 W9 b% ]# s第二:该属性指定需要struts2处理的请求后缀,该属性的默认值是action,即所有匹配*.action的请求都由struts2处理,如果用户需要指定多个请求后缀,则多个后缀之间以英文逗号(,)隔开
- Y8 Y; I$ v P9 |1 S$ Y$ g<constant name="strtus.action.extension" value=do"/>3 T6 h4 ~; t) N: \8 G. ]
第三:设置浏览器是否缓存静态内容默认值为true(生产环境下使用)开发阶段最好关闭,不然看不到修改后的数据
9 L- {! M/ ]/ f R5 t; i3 z ]! h! N<constantname="struts.serve.static.browserCache" value="false"/>
4 z8 Q( J$ c# r3 k2 }4 b) o第四:当struts的配置文件修改后系统是否自动重新加载该文件默认值为false(生产环境下使用),开发阶段最好打开
! w) L( S8 s% X+ Q3 p. s; r b8 o& a<constantname="struts.configuration.xml.reload" value="true"/>* G% w! m* S, f+ z3 E, U
第五:开发模式下使用,这样可以打印出更详细的错误信息
: g$ r( @: s. J% R& J- B- H: _<constant name="struts.devMode"value="true"/>
E# M+ Z3 } P. e8 _第六:默认的视图主题4 R" [8 {# S& R# y
<constantname="struts.ui.theme" value="simple"/>
3 b8 f2 E4 Y0 y; N! ?第七:与spring集成时,指定由spring负责action对象的创建/ [( a) V+ U* u! s) h- k, _
<constantname="struts.objectFactory" value="spring"/>/ S$ ^$ j5 a. h9 I @, T0 N
第七:该属性设置struts2是否支持动态方法调用,该属性的默认值是true,如果需要关闭动态方法调用,则可设置该属性为false
/ c" C- {6 a' n" k<constantname="struts.enable.DynamicMethodInvocation"value="false"/>. ?+ k9 \9 ~3 t ~6 [' M# n% w! e
第八:上传所有文件的总大小限制
! x5 Q! R- j' x+ R8 `+ Q8 Cconstantname="struts.mulitipart.maxSize" value="10701096"/>! @# T3 x; t! r3 I! b N/ u5 p
5 s; K P @( |4 c, u
28.自定义拦截器$ j9 e; d8 R" b! ^
(1).如果用户登录后可以访问action中的所有方法,如果用户没有登录不允许访问action中的方法,并且提示"你没有权限执行该操作"1 A7 f* ?8 Q. p% ]' r6 j# k# K2 Q* n
2 L3 A- O1 m- I* e6 m7 e
(2).
3 [4 n7 a6 C5 P<interceptors>
1 \. P# I6 x9 N) _<interceptorname="permission"class="cn.itcast.interceptor.PermissionInterceptor"/>
$ V# a, B) l) l1 V/ ^% f</interceptors>
! Q' j7 x1 n7 m3 W9 h<actionname="list_*" class="cn.itcast.action.HelloWorldAction"method="{1}">
3 L: P$ x7 o/ y<interceptor-refname="permission"/>! O$ D$ g- {1 x4 p5 D# }# N
如果为某一个Action定义一个拦截器,struts2中对Action的默认的很多拦截器都失去功能,所以要想做到两全其美,需要定义一个拦截器栈:, X% R2 ?5 w3 z! q- `' k
<interceptors>
6 T$ A+ F) z4 F' J) g<interceptorname="permission"class="cn.itcast.interceptor.PermissionInterceptor"/>
% b |. e! r$ h; k' X0 v<interceptor-stackname="permissionStack">3 L' ~7 z) l; a6 l3 [% Q
<interceptor-refname="defaultStack"/>
$ B8 |2 k. y: K6 e* c5 r+ Z<interceptor-refname="permission"/>
# u6 I' r. x! F7 D. {% u</interceptor-stack>
* `6 l! n# Q3 ~6 @</interceptors>. Y; Z) D% c- o2 |" c1 x3 I* Z0 _
因为struts2中如文件上传,数据验证,封装请求参数到action等功能都是由系统默认的defaultStack中的拦截器实现的,所以我们定义的拦截器需要引用系统默认的defaultStack,这样应用才可以使用struts2框架提供的众多功能,如果希望包下的所有action都使用自定义的拦截器,可以通过<default-interceptor-refname="permissionStack"/>把拦截器定义为默认拦截器,注意:每个包只能指定一个默认拦截器,另外,一旦我们为该包中的某个action显示指定了某个拦截器,则默认拦截器不会起作用.5 Q* b" T2 b) S- O0 A
(3).系统默认的拦截器可以到struts-default.xml中查看,很多功能.系统拦截器放在最前面,自定义的拦截器放在后面.8 r4 ?1 G1 d* G3 C4 m7 X* T( ?
, ]4 B- Y& ?( J0 K1 z29.自定义类型转换器
8 y" U' m! Q* l: n(1).struts2中提供了两种类型转换器:局部类型转换器(只对某一个action起作用),全局类型转换器(所有的action起作用)" x0 c0 T/ S7 {9 \" i; o4 ]: k
- `4 u) l, m9 x8 L* W
(2).类型转换器必须继承DefaultTypeConverter最好用xwork2.jar中的,重写converValue(Map<String,Object>context,Objectvalue,Class toType){# W: S- ]$ A/ c& H& t1 }" L
returnsuper.convertValue(context,value,toType);6 N) A) T% l+ _+ r
}6 \( r8 _ K' Q% n. d) [5 @6 a0 j
其中第一个参数和ognl表达式,第二个参数是需要转换类型的内容(是String数组,因为可能有多个值),第三个参数是需要转换成什么类型,要实现双向转换
3 W# B* x Y# H5 l" { 2 T) p: G8 K6 j) `; K B/ x! J$ l
(3).将上面的类型转换器注册为局部类型转换器:
0 P$ s9 p$ f5 A) v3 v" ?在Action类所在的包下放置ActionClassName-conversion.properties文件,ActionClassName是Action的类名,后面的-conversion.properties是固定写法,对于本例而言,文件的名称应为HelloWorldAction-conversion.properties.在properties文件中的内容为:
: S. b& k1 Y* U* v/ _- c需要转换的属性名称=类型转换器的全类名# U5 s; g- o9 ^! K l3 B0 @: B t
对于本例而言,HelloWorldAction-conversion.properties文件中的内容为:
9 u5 v8 ^* ^9 z$ s% U8 O; W, bcreatetime=cn.itcast.conversion.DateConverter |
|