我的日常

登录/注册
您现在的位置:论坛 盖世程序员(我猜到了开头 却没有猜到结局) 盖世程序员 > JavaWeb方面的所有知识听课记录
总共48086条微博

动态微博

查看: 5467|回复: 12

JavaWeb方面的所有知识听课记录

[复制链接]
admin    

1244

主题

544

听众

1万

金钱

管理员

  • TA的每日心情

    2021-2-2 11:21
  • 签到天数: 36 天

    [LV.5]常住居民I

    管理员

    跳转到指定楼层
    楼主
    发表于 2015-06-02 12:42:30 |只看该作者 |倒序浏览
    一、       javaWeb基础
    3 t% ]+ }, R2 ~- T, d8 R第一天:+ U6 X3 _2 B1 z( L( I- k
    1.Eclipse详解:( @6 v. @1 N* G8 N' l* ^
    (1).Bad versionnumber in .class file:编译器版本和运行(JRE)版本不符合。高的JRE版本兼容低版本的编译器版本。( M+ [- K" T$ @( o1 L8 x$ f! @4 v$ o

    ' y2 u! C, }$ e, b) D(2).当程序有错误的时候,使用Debug as 运行程序。双击语句设置断点。程序运行到此处停止。点击跳入方法的内部代码。点击跳过,执行下一条代码,点击跳出,跳出方法。观察变量的值,选中变量右击选择watch. 跳入下一个断点。查看断点,调试完后一定要清除断点。结束运行断点的jvm.
    $ Q( ?* W1 \  m/ p' N: G9 c
    0 m. x0 j6 _* f& p# ~; e, H$ H, I2.HashSet和hashCode和equals方法
    6 c- w! }8 I! z7 _% ?java系统首先调用对象的hashCode()方法获得该对象的哈希吗,然后根据哈希吗找到相应的存储区域,最后取出该存储区域内的每个元素与该元素进行比较.两个equals相等,hashCode()相等。需要重写equals,hashCode()方法.更改数据的值,hashCode()的值也更改了,并未删除.内存泄露.有个东西不在被用,但是未被删除,导致内存泄露.  V! c+ e: B9 {" ]7 b
    4 ~; X0 y$ ?' a1 D- w5 [
    3.Junit测试框架
    0 A  ]! _) n- d; v, j( N: K+ }3 A(1).在测试类,方法前加注解:@Test,不然出现初始化异常。! k8 }& g$ u& r. U! ?

    ! a5 a6 x9 N9 H- e(2).方法before,after前加@Before,@After注解。在测试方法之前和之后运行方法。+ A/ ?/ l/ X. T* h, |# [& x% R5 u

    # E4 I* A* m  I(3).静态方法beforeClass,afterClass方法前加上注解@BeforeClass,@AfterClass,类加载的时候运行
    ! L# c4 _4 Y  T* F4 z4 [7 }! o1 I0 b% Z3 |1 @
    (4).Assert断言。判断两个对象是否相等。期望值和实际值相等。! T3 y5 Y& D& m4 G- l6 C
    / h4 ?8 T; s) I1 Y
    4.得到配置文件的路径
      f% v3 T! n) [  L* y6 [3 }通过类加载器 reflect.class.getClassLoader.getResourceAsStream();在class指定目录下查找指定的类文件进行加载.编译器把src中的.java文件编译成class文件,所有非.java文件,原封不动的搬过去.但是这种方法是只读的.
      z# n; R9 @) u& d2 _通过类的信息reflect.class.getResourceAsStream();相对路径
    7 y) \1 s* b$ m# M9 v0 t一般用绝对路径,使用配置文件告诉用户路径.7 {. h' s/ }0 R2 D" D
    一定要记住要用完整的路径,但是完整的路径不是硬编码的,是计算出来的.
    # ]! O/ |/ u7 [# {
    3 d5 t. r! Y, W1 U! P" v5.反射: v" D# Q1 b% \1 a; i9 T% G
    (1).反射主要用于框架开发
    . ~" f3 K1 f+ y7 m6 U- l# D7 V3 `5 K3 D3 l; e% V- l
    (2).一个类有多个组成部分,例如:成员变量,方法,构造方法等,反射就是加载类,并解析类的各个组成部分。
    1 _- i5 b8 P5 J* p
    5 \4 e( G  Y& r/ B7 U0 a3 f/ z(3).加载类使用Class.forName()静态方法,给类的完整名称,包名和类名。; |: V% \' Q1 p

    " M) _4 x5 M3 C(4).Class提供了解析public的构造方法,方法,字段等方法以及private。字段封装数据,方法执行功能' c2 `3 l  O; H% o; X
    * H! j" h: Y! _. ~3 ~# g) f+ o
    (5).静态方法无需传递对象,method.invoke()% n% F  b! }* ^0 B/ W

      `+ E1 S- G* v% P9 |  e' [(6).升级时保持兼容性,main函数的解析有点麻烦,反射解析数组参变量的时候的问题。启动Java程序的main方法的参数是一个字符串数组,即public static void main(String[] args),通过反射方式来调用这个main方法时,如何为invoke方法传递参数呢?按照jdk1.5的语法,整个数组是一个参数,而按照jdk1.4的语法,数组中的每个元素对应一个参数,当把一个字符串数组作为参数传递给invoke方法时,javac会到底按照哪种语法进行处理呢?jdk1.5肯定要兼容jdk1.4的语法,会按jdk1.4的语法进行处理,即把数组打散成若干个单独的参数。所以,在给main方法传递参数时,不能使用代码mainMethod.invoke(null,newString[]{"xxx"}),javac只把它当做jdk1.4的语法进行理解,而不把它当做jdk1.5的语法解释,因此会出现参数类型不对的问题。解决的方法:
    - U3 g) L  A0 e; YmainMethod.invoke(null,newObject[]{new String[]{"xxx"}});
    # a+ T9 C. z) I% m! H% OmainMethod.invoke(null,(Object)newString[]{"xxx"});编译器会做特殊处理,编译时不把参数当做数组看待,也就不会把数组打散成若干个参数了.
    / _4 t$ Y9 R" W$ g% o
    , C3 b- S6 }) Y(7).对数组进行反射:相同的维数(不是数组元素的个数,例如都是一维数组,不关心数组的大小),相同的8种基本数据类型时数组有相同的字节码." J( V- w3 X2 @5 m$ b* N# X

      k3 e% A+ S. X; O8 E/ f6. 泛型( f* D4 L6 S; V1 N( s
    (1).泛型是对象类型,不能是基本类型,泛型存在源代码级别上,给编译器看的,生成class文件就不存在泛型了
    0 ?/ w; B8 ~! p. Q# d
    8 N( U, I& r( T% r(2).参数类型变量,实际类型变量,泛型类型,参数化的类型
    5 Z+ `: e7 p: N2 @/ i; Q
    " a9 m2 g/ o' N+ y( c: U' x/ e(3).自定义泛型方法:public <T> void method(T args){};public<T,K,V> void method(T a,K b,V c){};
    ! K! m) q* n2 f8 o/ w& a
    - K6 S' \; H6 B0 M7 `8 P* s(4).静态方法public static <T> void method(T t){};泛型类不作用静态方法
    8 j% o3 L8 w* o( O6 c) `
    - f- {8 h, W# S, D) ]/ s  A7.可变参数- v' e3 a( s7 x0 A
    (1).可变参数就看成数组,可以使用增强for循环5 T0 ]9 O& ]$ ^( n. x) }% B# C
    + \1 @9 a( k" r. x# L% |
    (2).可变参数列表为最后一个参数7 x. ?1 L7 H! ^: z

    ) L, g9 w: I2 b, u, U(3).可以给可变参数传递一个数组
    ( U- W9 [* r0 S4 m" T
    & k. y* m  J, N) v(4).可变参数的类型是基本类型还是对象类型  T9 L( `" I6 g& z! f& v
    ) t1 i, d0 p7 I/ v+ P$ s# U
    8.课程介绍
    3 y0 Z4 l+ l5 H- K' O$ c' a(1).在谷歌心目中,“云”必须具备以下条件:数据都存在网上,而非终端里,软件最终消失,只要你的“云”设备拥有浏览器就可以运行现在的一切,“云”时代的互联网终端设备将不仅仅是PC,手机,汽车,甚至手表,只要有简单的操作系统加个浏览器就完全可以实现,由于数据都在“云”端,企业的IT管理越来越简单,企业和个人用户也不同在担心病毒,数据丢失等问题。# u. r# W6 T7 \$ [. ]0 Y, N

    $ N4 G# ^% A8 u, U& _# q& X(2).李开复描述了这样一个场景,只要你的PC或手机等终端里安装了一个简单的操作系统和完整功能的浏览器,开机后输入自己的用户名和密码,你存在“云”中的应用软件和数据就会同步到终端里。
    % J; }; D3 X6 y9 O: H
    % ^3 B9 T  j2 T( ^% f- G9.快捷键% C0 I2 S- f& [) g, z3 }& H
    (1).配置快捷键:window->preferences->key& C8 t% Q1 Q( R5 r
    (2).Alt+/ :内容提示
    " ~- v+ x8 A: M1 b          Ctrl+1 :快速修复8 {$ V9 `: U6 O/ W/ r
              Ctrl+Shift+O :快速导入包6 G. W* C' D! U5 n& L! S  z
              Ctrl+Shift+F :格式化代码
    + j0 P, K1 l6 y          Alt+方向键 :跟踪代码
    ' H1 Q, s& s$ C8 i* S8 g          Ctrl+Shift+/ :添加注释
    1 A1 V6 ?2 J4 x( q          Ctrl+Shift+\ :取消注释7 _# Z" `. h: J- z4 H( g
              Ctrl+Shift+X :更改为大写& }' E! H& P, z# v  K; H
              Ctrl+Shift+Y :更改为小写
    & w7 R0 s  ^9 ]/ `          Ctrl+Shift+向下键 :复制代码) P0 ^0 ]1 u* r2 d0 N
              Ctrl+Shift+向上,向下  :改变代码行的顺序
    0 [% B7 e3 _/ c$ c! [  c8 p% q$ t          Ctrl+T  :查看继承关系
    * u& j% ~4 @! d' ?9 }8 `          Ctrl+Shift+T :查看源代码# l6 `* M6 `' `2 h0 E
              Ctrl+Shift+L :查看所有的快捷键2 h! p' ^5 W8 j
    " y" W0 H% j- O7 t
    10.类加载器及其委托机制的深入分析) v: @9 w. m4 C; {6 e% l; q, [; A
    (1).Java虚拟机中可以安装多个类加载器,系统默认三个主要类加载器,每个类负责加载特定位置的类:BootStrap,ExtClassLoader,AppClassLoader3 T5 n" M( W+ f9 [4 c

    8 i( e: y: F  @6 T6 I(2).类加载器也是Java类,因为其他java类的加载器本身也要被类加载器加载,显然必须有第一个类加载器不是java类,这正是BootStrap(内嵌到JVM的内核中,使用C++语言编写的); f2 D3 w6 R, X' W
    9 d! A$ z: a; c1 k1 N$ x* P
    (3).Java虚拟机中的所有类装载器采用具有父子关系的属性结构进行组织,在实例化每个类转载器对象时,需要为其指定一个父级类装载器对象或者默认采用系统类装载器为其父级类加载器1 v' ]+ ]% E+ f7 {. f2 a4 l, e
    (4).6 D7 V$ P6 r5 j, w4 c& H' Z) t
    public class ClassLoaderTest{2 ~$ p4 R: _: d
    public static void main(String[] args){
      M) B5 `$ J8 t/ b; h- f& }, pSystem.out.println(ClassLoaderTest.class.getClassLoader().getClass().getName());
    1 `. j3 O. W: Q' k9 u//输出为sun.misc.Lanuncher$AppClassLoader;
    ( p% g) y& i8 w3 vSystem.out.println(System.class.getClassLoader());! T7 o9 c$ w  T" Y) |& S# Q8 K
    //输出为null,因为类System是由BootStrap加载的;
    & `  @# F- v4 A" k( U( H}
    1 X9 \: E1 c8 |' x# }; M, C7 F2 _/ ^0 R}
    / w6 p1 R3 c9 x0 O, j
      K; V9 J+ T8 r- m' `9 M' A: {4 h(5).BootStrap->ExtClassLoader->AppClassLoader5 @6 s# S+ t# _5 h, q. M" E
    ClassLoader loader= ClassLoaderTest.class.getClassLoader();$ |; X! B$ ]9 l' M) v! }; A
    while(loader!=null){( ?9 C. L2 ~1 h& G+ k. [2 `& A
    System.out.println(loader.getClass().getName());
    8 Q# i2 F' B8 d$ U3 ]  n+ Aloader = loader.getParent();//往上顺序打印
    , N1 ?& X7 l! ]}
      e3 m: m" _( }  a) L# Z& aSystem.out.println(loader);//最后打印老祖宗
    . S& M5 |2 {- {/ k+ I+ s) c7 s" k; P& m
    (6).. h; t' W- a) E# h) d6 T+ W
    BootStrap------>JRE/lib/rt.jar& @# Z; E# N) {: M4 g
    ExtClassLoader----->JRE/lib/ext/*.jar
    ' ~/ |' H5 G. U/ a5 WAppClassLoader------>ClassPath指定的所有jar或目录
    1 P* z' u& k+ v# {) C6 [0 L2 `: U用Eclipse的打包工具将ClassLoaderTest打包成itcast.jar,然后放在jre/lib/ext目录下,在eclipse中运行这个类,运行结果显示为ExtClassLoader,此时的环境状态是classpath目录有ClassLoaderTest.class,ext/itcast.jar包中也有ClassLoaderTest.class,这时我们在打印ClassLoaderTest类的类加载名称,发现是ExtClassLoader,而不是AppClassLoader.6 O1 m/ Q8 h8 {/ u) H
    8 V3 F8 \9 c1 s# l  [8 J5 G
    (7).类加载的委托机制:, t( i5 k* A. v' ]# Z5 I" Y
    当Java虚拟机要加载一个类时,到底派出哪个类加载器去加载呢:6 h+ s: ]4 j+ K
    首先当前线程的类加载器去加载线程中的第一个类8 O& w- I1 p1 e: ]1 t! J4 ^
    如果类A中引用了类B,Java虚拟机将使用加载类A的类加载器来加载类B
    % ?8 ]2 }7 x; U  w5 l/ M, }还可以直接调用ClassLoader.loadClass()方法来指定某个类加载器去加载某个类* f5 S9 B8 s! b( j& T$ b
    每个类加载器加载类时,又先委托给其上级类加载器8 }! w, S. `: f  p. s. ^
    当所有祖宗类加载器没有加载到类,回到发起者类加载器,还加载不了,则抛ClassNotFoundException,不是再去找发起者类加载器的儿子,因为没有getChild方法,即使有,那么多个儿子,找哪一个呢?
    - }. r; k! [+ ^* N+ ^$ S对着类加载器的层次结构图和委托加载原理1 X+ M% y. {6 p) u

    0 S  U- y2 ?" `4 @7 e(8).Thread类有一个方法setContextClassLoader(ClassLoader classLoader)//加载当前的类.
    1 I, S2 L; N* t/ O' g: f& a* y4 B# Y* B4 {. J) P3 ~/ V
    (9).每个类加载都首先会委托送到BootStrap,那么BootStrap很累,这样,那为什么不多搞几个BootStrap呢,之所以不这样做,是因为,便于统一管理,因为所有的类都会找BootStrap,可能这时有几个相同的类进行加载,那么BootStrap,不会多次将他们的class文件加载内存中,只会加载一份即可.这样效率就高了.
    ) }6 T1 m, ?$ a/ ^8 M7 I1 R8 E9 F& f0 M* p1 c
    (10).
    & V: e0 _: n( ?  I& d% I9 ?- C. F, |public class MyClassLoader{9 w  x/ ?$ i1 {/ S' B  E1 y+ V# V
    public static void main(String[]args){
    8 M7 B9 Z. X1 f/ T7 H4 e  B/ jString srcPath=args[0];3 ^/ J% m* s0 V1 s$ N
    String destDir=args[1];//得到目录
    & S5 k, T& U% s4 |- I5 z8 hString destFileName =srcPath.substring(srcPath.lastIndexOf('/')+1);//得到文件名( D; f% A# L: U7 d' |
    String destFilePath=destDir+"\\"+destFileName;7 |6 R9 g" k3 s1 ?
    FileInputStream fis = newFileInputStream(srcPath);
    9 q0 n! q+ Q" z7 B/ I4 P9 xFileOutputStream fos=new FileOutputStream(destPath);  g5 _% A( L1 y- ^8 g
    cypher(fis,fos);
    , N4 t* S/ S( Q! d7 ~( bfis.close();
    . ^# b& `, B- k6 j  X9 R$ Ofos.close();0 Q, ^; D; E) z  ~$ C9 J
    }( t7 S2 K+ S3 f8 {8 F% w# X1 d
    private static void cyp(InputStreamips,OutputStream ops){5 ~# R# N+ b: T$ w' \
    int b =-1;
    ' C7 r4 ]" ^% F1 u3 x3 L" ^  s" f" Lwhile((b=ips.read())!=-1){% C6 |; P/ s; c6 H3 b  i, ^
    ops.write(b^0xff);//对内容进行异或处理
    . x. N0 `0 Y8 j4 n}- H; ~( M' c0 l+ ]2 u) X
    }: F6 k9 _5 U* m* a* n6 o
    }* A- z9 O5 F# P( E1 s0 c' E$ \9 R
    class ClassLoader extends Date{) y% n: v! U8 ?; O+ \
    public String toString(){
    . Y' b1 s/ z% N" _% i2 O; W: Wreturn "hello,itcast";8 w" ~7 @2 T  s* M% I( U, [
    }& O# i0 i, W9 G8 E
    }
    & D, A5 j2 M( V7 }* Z, e) Qargs[0]:ClassLoader.class的绝对路径
    / N- Y1 s) x6 h' U2 [, Pargs[1]:itcastlib
    9 c* E# n2 e" g8 Q有包名的类不能调用无包名的类.
    " h. K% f8 ]$ T; ~" z' e3 L& [' G' a/ ^; |' D( ~, E" q( d
    (11).编写自己的类加载器:% Y- o# J" \/ {4 f& U* C& s- b
    知识讲解:% v7 t' \8 g5 N6 y; H: C; o1 [6 D
    自定义的类加载器必须继承ClassLoader(抽象类)
    ; Y6 B: B1 ?+ ]/ e9 H1 Q' p覆盖findClass方法
    % c& O  r) w9 \$ |; m( NdefineClass方法:得到class文件转换成字节码/ p" e. H3 g! R1 ^$ c
    编程步棸:0 `' [$ x$ g  w
    编写一个文件内容进行简单加密的程序
    9 A7 i( ]3 R# e; E6 }1 m编写了一个自己的类加载器,可实现对加密过的类进行装载和解密
      y0 h! Y) k- A3 c编写一个程序调用类加载器加载类,在源程序中不能用该类名定义引用变量,因为编译器无法识别这个类,程序中可以出了使用ClassLoader.load方法之外,还可以使用设置线程的上下文类加载器或者系统类加载器,然后再使用Class.forName
    6 ~, l; E* q, i; l4 x; C: u* X. H6 b' V
    (12).
    " M, s- z8 T- U* o/ K) k9 A( a模板设计模式:; B# |. C( i/ \0 |- _) ^9 \
    父类--->loadClass(相同的)
    8 e' I9 [# T: p& o6 g: d子类1(自己干的)
    6 d! d( Q: K* D$ p) W* Q子类2(自己干的)
    0 b4 n* H0 r3 l+ I3 W1 m/ G覆盖findClass方法(自己干)
    6 U' @; N& H: o- O# b9 U
    ; L- Y+ W/ M) B1 p(13).
    . ]6 ]5 Q: k9 Q+ ~% W% }public class MyClassLoader extendsClassLoader{& R- ^% h& ~3 r' @% g' x# x$ L" V/ H
    public MyClassLoader(){# J7 q% n1 `% p5 {; r+ ^
    }
    ' E' q9 w/ L/ O( k# K0 h% apublic MyClassLoader(String classDir){# N9 g% T3 _) @
    this.classDir = classDir;
    + u1 Z; t" Z: Q}8 A. ?5 B7 Z8 R5 ]5 ?# S
    @Override
    $ g1 E" T0 w( s/ Gprotected Class<?> findClass(Stringname){0 H( I% y  |# Q9 i! w0 e
    String classFileName = classDir +"\\" + name + ".class";
    ) x+ G$ m0 C) B- r, r& l, fFileInputStream fis = new FileInputStream(classFileName);
    $ ^% \5 v# B  q9 X. d/ V1 \$ a, A  wByteArrayOutputStream bos = newByteArrayOutputStream();. X$ S8 A& |3 \
    cypher(fis,bos);) }$ F6 N, d/ L
    fis.close();# A% `  ^6 g' {1 O$ ?
    byte[] bytes = bos.toByteArray();: I" O6 L$ [7 X; o5 r# E  m
    defineClass(bytes,0,bytes.length);7 n, B& G- W+ E- w! ~% m
    return super.findClass(name);
    & o& f3 M* _  m' Z, H}
    # o! Y3 ~0 T3 {' zpublic static void main(String[] args){  z3 P1 ^. b2 w
    Class clazz = newMyClassLoader("itcastlib").loadClass("ClassLoaderAttachment");. }& {$ U' t4 w
    ClassLoaderAttachment d1 =clazz.newInstance();
    ) }. T5 _1 U! b1 }% k* p' V0 _8 [}
    " K9 m4 r# r6 `}# Z5 H/ ]3 b; z% x

    : T) W; Z6 d# U3 b# i(14).windows->showview->problem查看错误.8 O% T! G7 ~8 {7 }" Q+ T6 \( `

    : t' f, f4 }4 ]% a8 H(15).可以查看Servlet的类加载委托继承树
    ' q& p& T' a% y 传智博客(JavaWeb方面的所有知识)听课记录(经典).txt (219.71 KB, 下载次数: 1) * L8 |* d) Z5 L; M. u

    1 ]! n/ U1 ]1 l5 A- U
    3 }7 R% k) }8 I4 i1 [: E

    科帮网 1、本主题所有言论和图片纯属会员个人意见,与本社区立场无关
    2、本站所有主题由该帖子作者发表,该帖子作者与科帮网享有帖子相关版权
    3、其他单位或个人使用、转载或引用本文时必须同时征得该帖子作者和科帮网的同意
    4、帖子作者须承担一切因本文发表而直接或间接导致的民事或刑事法律责任
    5、本帖部分内容转载自其它媒体,但并不代表本站赞同其观点和对其真实性负责
    6、如本帖侵犯到任何版权问题,请立即告知本站,本站将及时予与删除并致以最深的歉意
    7、科帮网管理员和版主有权不事先通知发贴者而删除本文


    JAVA爱好者①群:JAVA爱好者① JAVA爱好者②群:JAVA爱好者② JAVA爱好者③ : JAVA爱好者③

    admin    

    1244

    主题

    544

    听众

    1万

    金钱

    管理员

  • TA的每日心情

    2021-2-2 11:21
  • 签到天数: 36 天

    [LV.5]常住居民I

    管理员

    沙发
    发表于 2015-06-02 12:42:55 |只看该作者
    11. 枚举) o* j6 ]5 V- t
    (1).枚举的作用限定指定的值,没有枚举前设计一个类,将构造函数设置成私有的,变量设置成静态常量7 t; s/ O  [4 r% N) e3 ?8 h
    + y. D. V0 H, {6 c! |7 Q
    (2).枚举可以有构造函数(私有的),字段,方法5 L5 J2 X) G/ u" \

    / ^  a' ~, R/ h0 G(3).可以定义set,get方法,获取变量的值。
    & o4 j7 M* a2 L7 O8 D(4).带有抽象方法的枚举,不能new出新的对象了。在new对象的时候就重写抽象方法,使用匿名内部类7 p4 g+ C; A; v1 {+ O# H
    + F' I5 u2 S. b7 i
    (5).枚举中的每个枚举值代表枚举类的一个对象。
    : `4 u. h( o' D; ?; O& o% c
    0 @( }2 D% f/ L- J) j- j(6).枚举也可以实现接口,或继承抽象类
    % W, d% q$ s& S, y5 r$ ?) h 2 u: a3 x( w" r$ a3 Y
    (7).JDK5中的switch拓展为除了接受int,short,char,byte外,也可以接受枚举类型。
    2 F+ o9 p: W; O7 f ( W# `, [) a* T9 C5 X
    (8).枚举的方法,name,ordial,valueOf,将字符串转换成枚举值。表单提交数据的时候。values返回枚举的所有的枚举值
    # [+ c  B" ~$ r# J, z, n; z9 r
    " h" q$ B$ X7 n  T12.内省! d, `: v; m# K' k' Q3 P
    (1).内省:Introspector,专门操作Bean的属性。
    1 d* X/ e7 |/ i+ P9 v4 C6 M$ T2 g/ O % U' v6 |* h+ I/ l0 d6 }& W) k
    (2).Bean的属性:只有字段提供了set/get方法,就是属性。只要有get/set的方法,就有一个属性,所以属性是有get/set方法决定的# F) |4 v( ]3 a  ]
    $ C3 p) y0 ^% H) N9 b4 f$ B
    (3).任何类都继承了Object类,又因为Object中有一个class属性
    : C4 A; g, o8 @; W3 g! H
    % {9 B8 `# J$ S* Y  q$ l5 c1 V4 f(4).内省的入口类:Introspector,方法getPropertyDescriptors()获取Bean的属性
    * a5 w$ n: Y% x& x7 s$ J : i: V) T) N  \+ D4 [4 ?4 q
    (5).操作Bean的指定属性1 F2 z/ d: \  `2 @, l( k/ w+ K

    , P7 |1 y" k+ y9 ?. A. L(6).BeanUtils框架操作Bean) @! N4 [1 V# o9 [

    1 S: \9 m. M( j5 M# E) c6 U(7).在工程中新建一个文件夹,将架包复制到文件夹中,还需另外的架包loging,选中架包,右击build path加入到工程的环境中7 m4 W" J+ F8 U
    & q  H) l- J& r
    (8).BeanUtils使用方便,类型自动转化,只支持8中基本数据类型
    & R. D6 a; u" e( {9 A* w2 k
    " K( d2 r- M! S( {0 a& L4 J- ](9).给BeanUtils注册类型转换器,ConvertUtils.register()3 ~/ ~1 e* o2 n: F6 }$ V
    # i4 Z; i6 h: |6 y# f, ~$ J
    (10).将Map中的数据整合到对象中,BeanUtils.populate()方法
    3 J% C. A3 z' R5 E% ] ! a3 Z/ a  F/ r% M  Q
    13.配置Java模板代码' I( l/ }0 i6 E0 m
    window->preferences->java->Editor->Template:编辑模板代码:line_selection,光标cursor,右击选择source with* z# W0 p/ F( }

    9 C5 g& z9 H- X: g! u' G14.享元模式
    % W8 u+ C+ Y3 s' u" ?* x8 }相同的对象只实例化一个,实例:桌面上的图标,word中字符,有很多小的对象,有很多相同的属性,不同的属性叫做外部行为,相同的属性叫做内部行为integer的缓冲池
    # J# ]/ k3 E# `) a / N4 B- r# w, Q& _
    15.注解9 k: l* b4 {/ ?# O  s( p
    (1).@SuppressWarning("deprecation")过时注解
      F8 M7 R9 }6 I3 o, u! j( T4 X, H" T 3 O. J' y3 J" G9 {: |
    (2).@Deprecated注解,表示该方法是否过时,架包升级时添加的注解* c! _7 B, L0 \/ C0 e$ f* A

    ! G$ j/ ~3 {& ]9 f(3).注解相当于一种标记,通过反射了解你的类及各种元素上有无何种标记
      e& m3 ^# f0 |, F) V; }- } 8 F" E1 g+ ]2 g
    (4).注解相当于一个类:@interface Annotation{};注解类,应用注解类的类,对应用了注解类的类进行反射操作的类
    5 D. B% Q9 X6 d1 W4 b7 \) M 6 c1 A9 {( m4 {1 m: K2 g
    (5).AnnotationTest.class.getAnnotation(ItcastAnnotation.class)得到类AnnotationTest上的注解ItcastAnnotation,注解上使用注解叫做元注解,元数据,元信息# a4 @) K( ~5 R1 U% ?$ Q
    . C+ S8 t" [4 |0 ?3 i3 Q
    (6).@Retention(RetentionPolicy.RUNTIME)(保持到运行阶段),@Retention(RetentionPolicy.SOURCE)(保持在源文件阶 段),@Retention(RetentionPolicy.CLASS)(保持在class文件中):源代码->class文件->(类加载)内存中的文件(字节码)" w$ |1 A5 p; S' g  ]
    . b1 T  S" p. j1 A/ ^3 G
    (7).@Override注解保持到SOURCE,@SuppressWarning注解保持到SOURCE,@Deprecated注解保持到RUNTIME(只有将该类调到内存中才知道该类中的方法是否过时了)
    * X6 o0 x/ M* h. D$ R 0 e; n" _  L3 J
    (8).@Target({ElementType.METHOD,ElementType.TYPE})注解只能标记到方法上或类,接口等类型上 (9).注解的属性:String color();类有个color属性,还有一个特殊的属性value,属性的默认值default,数组的属性值,枚举的属性值,注解的属性值
    / k2 k. o# ~/ \! G; S: k7 b" v! p  T第二天:
    & W' r: j. E0 c- \/ m1.dom4j解析XML文档1 _, ]* T+ H8 y8 ]. z3 c* o
    (1).Dom4j是一个简单、灵活的开放源代码的库,Dom4j是由早期开发JDOM的人分离出来而后独立开发的,与JDOM不同的是,dom4j使用接口和抽象基类,虽然Dom4j的api相对要复杂一些,但是他提供了比JDOM更好的灵活性5 u* _4 L7 a+ C# \% K! h' P
    ) E( U+ r8 j# J" O
    (2).Dom4j是一个非常优秀的Java XML API,具有性能优异、功能强大和极易使用的特点,现在很多软件采用的Dom4j,例如hibernate,包括sun公司自己的JAXM也使用了Dom4j
    6 m2 r; [4 L9 H* t5 O; l% R" H1 ~& B' w
    8 A# O: I( H* o; y" i(3).使用Dom4j开发,需要下载dom4j相应的jar文件
    . T' ^% O6 b( }3 b 8 @" _1 J% K( A) y: x
    2.  XML语法4 E! b4 V- D, x+ _3 J
    (1).编写XML文档时,需要先使用文档声明,声明XML文档的类型,使用IE校验XML文档的正确性.
    1 L: [# f7 {/ D4 [# x9 Q5 O' {+ L
    - h7 x5 m+ `, u! R; U(2).XML文档中的"中国"保存为本地的码表的"中国"# s6 W5 A* j  y# h& y: r% f
    0 _) S" v+ ~" @# C# G
    (3).在XML文档中,空格和换行都作为原始内容被处理
    : W( B8 ?( r% w: j ! t9 M$ r' `$ }
    (4).XML区分大小写,一个标签可以有多个属性,每个属性都有它自己的名称和取值,在XML技术中," ]6 t. \# r7 c8 W. ?
    标签属性所代表的信息也可以被表示子标签表示
    ( }' v. V: m. j' C1 G$ K; h2 |- E
    ) `  r8 t" I& t  H7 i7 \7 V, n1 d8 Y4 W(5).XML文件中的注释采用:"<!--  >"格式
    5 Y$ B' s9 t+ w/ j5 C1 h " G7 z9 O4 ]9 m7 h
    (6).CDATA区域内的内容,XML解析程序不会处理,而是直接原封不动的输出# M7 {+ b% Z+ m

    " A) I' i* ]/ @" i' h# |5 |7 ](7).转义字符: "<":&lt; ">":&gt;. F* x& D; [, r6 U* H  Z

    9 i: T  |- a- A) c- c4 ~" ~+ g  x3.  Xml语言
    2 f, b8 E" \9 T$ v: g在xml语言中,它允许用户自定义标签,一个标签用于描述一段数据,一个标签可分为开始标签和结束标签之间,又可以使用其他标签描述其他数据,以此来实现数据关系的描述.用于配置文件,以描述程序模块之间的关系. H6 x) R7 R! s5 e% W
    4.  XML约束
    ; V  |  |6 R6 _" ](1).在XML技术里,可以编写一个文档来约束一个XML文档的书写规范" W& w- D2 g0 m3 M6 c

    / D5 _/ ]! v2 S
    ; B( }" q) U$ Y: R' d  y# B(2).常用的约束技术:XML DTD 和 XML Schema( m1 Y4 G  \0 c% E1 O

    0 ~5 e) ]/ _4 W( e5 l' T(3).如果dtd文件中含有中文,一定要存为utf-8格式.! E4 F6 j' M& C4 \/ {" V
    # b* v0 l: H9 ^8 H
    (4).IE5以上的浏览器内置了XML解析工具:Microsort.XMLDOM,开发人员可以编写javascript代码,利用这个解析工具装载xml文件,并对xml文件进行dtd验证.创建xml文档解析器对象:
    3 O$ z% ~7 y+ Cvar xmldoc = newActiveXObject("Microsoft.XMLDOM");0 y! y* L, J2 }  I* y" i
    开启xml校验! S9 q0 u7 t/ e% @) m) F
    xmldoc.validateOnParse= "true";' S, n/ n6 {" }2 L( m! M
    装载xml文档6 t4 Y& [. @5 |% g" p# M
    xmldoc.load("book.xml");
    $ c, U1 _  f! q. e, R3 B; w4 _6 G获取错误信息" r5 G8 w& }1 S2 R5 A: ^# L
    xmldoc.parseError.reason;xmldoc.parseError.line! Q' @3 x0 k3 C, U# `2 }7 m
    5 C7 B& `8 c5 Q9 M
    (5).将DTD文件导入到eclipse中,可进行校验xml是否遵循dtd文件
    4 P3 f: K& N. d% h在xml文件中编写DTD# j* ~& M4 C% j; A& c" j$ T
    9 L8 i' r. a  r( a% N8 i/ c
    第三天:
    . Y( `' L5 H+ D( J% G5 X/ K1.HTTP请求头各个头字段详解
    4 J) F2 v+ z. T. k请求头详解:
    8 a2 ~" X9 ~8 {- c  P; O8 rAccept:text/html,image/*用于告诉服务器,客户机支持的数据类型,
    : n; b; M8 t& Q& W/ x2 PAccept-Charset:客户机采用的编码
    ; R2 }1 w" B; X  A/ kAccept-Encoding:客户机支持的压缩格式. ~8 u2 i- |, v, R, }; z7 H
    Accept-Language:客户机的语言环境,不同国家访问的内容也是不同的,就是通过这个头实现的,用于实现国际化0 x8 a# n8 P& A& L3 D
    Host:www.it315.org:8080:告诉服务器,想访问的主机名$ I8 W2 ~' ^) v+ e: W+ F
    If-Modified-Since:客户机通过这个头告诉服务器,资源的缓存时间,提高访问效率
    - m" G- v) ]3 T9 C  t0 [7 T5 G: QReferer:客户机告诉服务器,他是从哪个资源来访问服务器的(防盗链),通过检查该头是否是从本网站点击过来的,如不是的,就让他跳到本网站的首页来. J) o/ h5 d8 Z2 H. D  L( L) E* J3 w
    User-Agent:客户机告诉服务器,客户机的软件环境
    % t# l, p5 ~* n* H, `- UCookie:客户机通过这个头向服务器带点数据$ l+ Y( c$ m6 b: j/ @
    7 A& x4 `4 z4 k) l1 J) D/ L, W3 J
    2.Http请求行和请求方式
    6 R8 n0 w$ l& _- J一个完整的HTTP请求包含:一个请求行,若干个请求头,以及实体内容,请求行,请求头,之后空一行,带有请求信息(如表单提交数据为post方式)如下所示
    % g* a- |$ @1 f4 R) p$ {Get/books/java.html HTTP/1.1   :请求行,用于描述客户端的请求方式,请求的资源名称,以及使用的HTTP的协议版本号  p$ \; F# a! H* L$ \* O
    Accept:*6 o3 z9 S. C9 Z- N
    Accept-Language:en-us
    7 D+ ?' j1 c$ B4 G$ w$ Y, i+ qConnection:Keep-Alive
    + p% P0 ], S9 O, wHost:localhost
    ; s7 d2 N' L$ V, o9 z9 j1 ?Referer:http://localhost/links.asp0 n  k# y# Y1 R2 L6 o, B( v; J+ ?3 f
    User-Agent:Mozilla/4.0
    + [, s6 L2 i7 g: |: I( ~, C9 ]Accept-Encoding:gzip,deflate" C, Z  D+ `: w2 ?3 @
    以上的内容为多个消息头,用于描述客户端请求哪台主机,以及客户端的一些环境信息等
    4 S$ O2 u' a2 i7 Z9 ~+ F# @+ R$ l请求行中的GET称之为请求方式,请求方式:post,get,head,options,delete,trace,put,常用的有:get,post,用户没有设置,默认情况下浏览器向服务器发送的都是get请求,例如在浏览器直接输地址访问,点连接访问都是get,用户如想把请求方式改为post,可通过更改表单的提交方式实现.不管post或get,都用于向服务器请求某个web资源,这两种方式的区别主要表现在数据传递上:请求方式为get方式,则可以再请求的URL地址后以?的形式带上交给服务器的数据,多个数据之间以&进行分割,同时在URL地址后附带的参数是有限制的,其数据容量通常不能超过1k,若请求方式为post方式,则可以再请求的实体内容中向服务器发送数据,post方式的特点:传送的数据无限制! [7 ?- d1 k6 k8 _
    <ahref="1.html?name=aaaaaaa">连接</a>get方式6 U6 `9 X- P$ J. f, Z7 g8 g# P! b

    3 z+ {2 I8 o" j! w3.Http响应头字段详解
    9 M: Z' G4 p# t8 q响应头:7 d# n5 f7 [, H6 u4 ?; y, T
    Location:* w) V8 @! u+ X* T) ?, |+ H' Q: S
    这个头配合302状态码使用,用于告诉用户找谁
    " w' @# \% o1 q2 z, S, C+ z7 I3 Q5 X,response.setStatus(302),response.setHeader("Location","/day04/1.html");请求重定向,判断浏览器的地址栏的地址是否发生变化,实例是用户登录
    $ x) X+ R& k) sServer:服务器通过这个头,告诉浏览器的类型
    / D' E9 M4 r) d8 R" rContent-Encoding:服务器通过这个头,数据的压缩格式,收费是靠数据出口量算的, 所以为了省钱和效率高,要进行数据压缩,jdk中的GZIPOutputStream类,压缩类流,包装流和底层流,最好将包装流关了或者刷新,数据写入到底层流中去,2 \- ~! w- ^% R5 l0 z( @! H
    response.setHeader("Content-Encoding","gzip");response.setHeader("Content-length",gzip.length+"");; b8 S2 S% O3 t1 Y
    Content-Length:服务器会送的数据的长度, K' Y8 ?; W7 a4 U4 `! p# n
    Content-Type:服务器会送数据的类型,response.getOutputStream().write();服务器会送数据都是二进制,通过这个头,可以告诉浏览器这个二进制是什么类型,this.getServletContext().getResourceAsStream("/1.bmp");intlen = 0;byte buffer[] =new byte[1024];OutputStream out =response.getOutputStream();while((len=in.read(buffer))>0){out.write(buffer,0,len)},在服务器的目录下的web.xml中查看各个数据类型的respose.setHeader("content-type","")的写法.
    , K; {% H5 _7 h/ ]2 Y2 |4 N3 oRefresh:告诉浏览器隔多长时间刷新一次,response.setHeader("refresh","3;url=""")控制浏览器隔三秒跳到指定的网页上: d1 z: ~* d# u& X9 M
    Content-Disposition:告诉浏览器以下载的方式打开文件,response.setHeader("content-disposition","attachment;filename=3.jpg);9 }) [8 R! S! o' }* Z9 R
    Transfer-Encoding:告诉浏览器数据的传送格式
    ' B- B% x5 n0 \8 s; n; R3 oETag:缓存相关的头,服务器根据数据的内容生产一个字符串,客户机第二次访问服 务器时带来的ETag的值和服务器的值一样,就拿缓存给客户,实时更新,) D2 `, N: |) m) }, \6 i. B3 y
    Expires:高速浏览器,把会送的资源缓存多少时间,-1或0,则是不缓存的) k/ s0 U% ]9 }
    Pragma:no-cache6 |4 ~- z3 F& }$ r. X; P
    Cache-Control:no-cache
    , m# y: f: b% G' B: o控制浏览器不要缓存数据,当数据不发生改变时,就要缓存,当实时性要求很高的数据不能缓存.
    2 {* K2 M5 W( c9 W$ [ * {" J2 N7 q! E! S9 K7 t
    4.Http响应状态行4 U* A5 w* ^4 B# P4 L6 V- D4 U
    HTTP /1.1 200OK  :状态行,用于描述服务器对请求的处理结果8 E& V4 t- D2 U
    Server:Microsoft-IIS/5.0 0 J$ U- d! s" X" S/ T; \
    Date:....: L# r% c- C7 P
    Content-Length:2291
    + ]* D2 X0 ?+ K# PContent-Type:text/html
    5 r+ X' c+ f8 I5 \% K; rCache-control:private
    , I( m. A( X; L1 `; K* b/ k, Y多个消息头:用于描述服务器的基本信息,以及数据的描述,服务器通过这些数据的描述信息,可以通知客户端如何处理等一会它会送的数据
    + f7 s: \; \% F8 i$ m. q<HTML>3 ]" |0 j$ P/ @2 u( I
    <BODY>" y) f; z" g1 A
    实体内容:代表服务器向客户端会送的数据( y4 \3 E/ T( H; h, y
    具体:
    $ O/ v4 T! U" y$ U! x1 G! S) k2 C* h状态行:
    $ ^  L5 d3 [5 s6 h1 G格式:HTTP版本号  状态码  原因叙述<CRLF>% z8 B* N( l+ I
    举例:HTTP1.1  200     OK7 b9 c: ?0 l; g8 Y8 G9 }
    状态码用于表示服务器对请求的处理结果,他是一个三位的十进制数,响应状态码分为5类,
    + f+ S. J  S7 z7 \  Z100-199:表示接收请求,要求客户端继续提交下一次请求才能完成整个处理过程
    , q" W% ]: X) S, m% o3 P% [200-299:表示成功接收请求并已完成整个处理过程,常用200
    ( |' P1 R9 {8 }) }, @5 v300-399:为完成请求,客户需进一步细化请求,例如,请求的资源已经移动一个新地址,常用302(你请求我,我叫你去找别人),307和304(拿缓存)% n8 K. N3 ?% P7 A
    400-499:客户端的请求有错误,常用404,403(没有权限访问)
    / J2 d5 K. c2 c+ C500-599:服务器端出现错误,常用500  E9 Y* [' S, m" Y: ]
    " r5 g# L" M  Q4 C. ]; q
    5.http协议概述* Q! G5 V! e: \
    (1).浏览所有的服务器:$ ?! ^- [1 K. Y0 e
    在命令行中敲入
    5 h# @- `( f  j! o: @4 S3 Ctelnet localhost 8080
    % A5 B/ }8 k2 V# y) KGET /aa/1.html HTTP/1.16 v- A' y' t3 Z7 ?% L/ F; G7 `
    Host:
    - A* o! S; h) j- G(2).在HTTP1.0协议中,客户端和Web服务器建立连接后,只能获得一个web资源。在HTTP1.1协议,允许客户端与web服务器建立连接后,在一个连接上获取多个web资源。9 B9 T+ P$ C8 G" A7 Y; i
    (3).在命令行中敲入以上的命令,HTTP/1.0获取一次资源,就断开连接,而HTTP/1.1不会断开连接' O7 a) E: Q+ i( R/ T3 u
    (4).一个web页面中,使用img标签引用了三幅图片,当客户端访问服务器中的这个web页面时,客户端总共会访问几次服务器,即向服务器发送了几次HTTP请求:- |4 Y  b8 p( E0 l4 U7 t
    <img src="1.jpg">  V8 ~% j4 J- x! h" K. I
    <img src="2.jpg">
    ) R/ X( i. ^3 u. @) A' b( i9 \<img src="3.jpg">
    3 {! X4 T6 h- j共四次请求,浏览器认识img标签,浏览器解析img,则img一次请求,最好将这三张图片整合到一张图片上,这样就只有两次请求了! ]0 E2 c. D' [/ a8 ?
    (5).<scriptsrc="1.js"># e! |" C2 n) v( H, E
    <scriptsrc="1.js">" }4 U$ C+ f! T+ @5 M" u
    共三次请求,可以将多个js文件整合到一个js文件,避免多次向服务器发出请求' A9 j: A0 K3 q5 [. d
    6. tomcat服务器目录结构! P# u3 _" d: o# h7 V, v. g4 p2 ?
    bin:存放启动和关闭Tomcat的脚本文件(命令)
    2 l5 T: p4 V) z7 a+ x$ o8 q' Q6 yconf:存放Tomcat服务器的各种配置文件(核心文件:server.xml)( [" r/ B0 ^& z& A! T
    lib:存放Tomcat服务器的支撑jar包
    7 P# j3 v$ t2 e* Plogs:存放Tomcat的日志文件(catalina文件)  m& J0 q: y0 x1 w- K
    temp:存放Tomcat运行时产生的临时文件
    & [! ?, W& \( ]" ?( Owebapps:web应用所在目录,即供外界访问的web资源的存放目录
    * i" @# ~: K. P9 `/ |9 Wwork:Tomcat的工作目录,jsp翻译成servelet文件
    5 {" N  t- z  j( E9 E
    ( a# {$ X. r) X3 D7.web服务器启动问题
    1 i( @* }+ z7 \) n$ }& }(1).tomcat是使用Java编写的,启动的时候需要JVM,java环境设置不正确的话可能导致服务器不能启动,还有就是8080端口被占用了.
    + H, U: p6 B$ f/ z ' ~0 F* ~. T" U- ^9 F+ t
    (2).在conf->中server.xml中对tomcat服务器进行配置; O& v! g$ g2 E9 m6 S; d4 {
    $ z  |# F; A' P* V0 R0 @( M" S
    (3).我的电脑->计算机管理->服务->停止windows服务1 }! w* J& O% z6 A% a

    * |2 R+ k1 k1 ^4 c(4).http://www.sina.com/是url;www.sina.com是主机名;sina.com是域名,在域名下可以搭建多个网站* o$ }% V4 d  V

    1 c7 `/ x, ]( V/ ~6 i2 b(5).tomcat文件下的webapps下新建一个web资源
    - F7 N( ^4 C$ r5 ^2 y& f$ U . E. w8 k, I% B. ~3 h! h
    (6).Cataline_home环境变量的设置问题,服务器的目录,启动的服务器的目录要和服务器存放的目录一致/ q' X: W  E8 L6 u7 K3 G( N
    2 b# N& \4 y5 a% Q$ |8 k/ W8 \) c
    8. web开发相关概念和常见web服务器
    1 ?0 H! f  K+ Q* T- f(1).自己开发服务器:
    1 D8 |3 @! r! ~public class Server
    5 Y# h1 x$ q- b" B, N; r{3 u0 I/ b$ l- g) |- z3 O
    public static void main(String[] args)throws Exception{- A1 X4 u) n6 e4 O4 m
    ServerSocket server = newServerSocket(9999);//服务器建立端口2 z  J0 b( E( C. q# l. e
    Socket sock server.accept();//客户端访问
    ' V( R6 p8 N3 K% F/ hFileInputStream in = newFileInputStream("c:\\a\\1.html");2 b, }" m: }* ~; t. L. h
    OutputStream out = sock.getOutputStream();% ]& F, V5 F1 ^. V5 H
    int len = 0;  b7 A. P* t2 g% j( E) O/ e
    byte buffer[] new byte[1024];
    0 j7 ^9 g; z3 S  M6 x. ^6 b* jwhile((len = in.read(buffer))>0){' j* Y# i! _' {* K
    out.write(buffer,0,len);4 \/ [3 r( D6 A8 v
    }in.close();
    0 U! P6 k* d5 v4 t+ n6 z& Lout.close();
    6 Z5 `0 o! D$ Q" o% x8 X1 Ysock.close();
    6 a, \0 Z# B9 V9 }server.close();
    7 z1 x% J3 Y$ l+ q}
    * F4 @' Q: r, l& n. \  |}
    $ a( l; G+ D  @% e0 a8 ?(2).WebLogic是BEA公司的产品,是目前应用最广泛的Web服务器,支持J2EE规范,是商业产品,需要收费: s( s  A2 p( V# o6 n$ X4 S
    (3).常用的Web服务器IBM公司的WebShpere,也遵循J2EE规范,也要收费. T- X7 i, T9 S& I3 n7 f
    (4).Tomcat服务器只支持JSP以及Servlet规范,不支持EJB,如果要支持可以使用JBOSS
    - H& ~. \! ^* j5 p; ?- H9 c(5).http:80,smtp:25,pop3:110,ftp:23,https:443. D0 K# y' f+ O

    5 K6 d3 x. r, E0 d1 D( o9.  web应用和虚拟目录4 Y- v9 p7 U) b0 P$ R1 H
    (1).web应用虚拟目录映射:自己开发好的web应用项目到部署到服务器上的web应用项目的关系目录,在服务器的配置文件中配置,虚拟目录,<Context path="/itcast"docBase="c:\news"/>,"/itcast"就是虚拟目录,"c:\news"是磁盘上web应用的目录,修改配置文件后一定要重启服务器.http://localhost:8080/itcast/1.html,浏览c:\news\1.html文件.当path=""时,就是缺省设置
    7 _0 [* s( @2 \, @! Q$ H
    8 `# B3 q. {6 L& R0 f& {" z& Z(2).tomcatdocument中reference中观察tomcat的配置文件设置方法.在Catanena\localhost\新建一个context.xml文件.这种方法无需重启服务器.( W6 U+ U$ ]7 A% c! P1 H- O' c% l" n
    <ContextdocBase="c:\news"/>,http://localhost:8080/1.html;多级虚拟目录:文件名:a#b#c.xml,http://localhost:8080/a/b/c/1.html;缺省的虚拟目录,把文件名改成ROOT.xml重启服务器,覆盖原先的缺省设置.3 h( K# a! Y& O0 D0 Q# D
    5 j1 k5 f! a% C8 f% O( w0 o. S
    (3).webapps文件夹中的web应用可以让外界访问,又服务器自动映射目录,把news文件放到webapps中去,http://localhost:8080/news/1.html,当服务器和webapps不在一个目录时,此方法不可以.! X8 g% Q7 Y: ~- Y; l

    " L3 X! B. k: }9 P8 b10.  web应用组织结构
    9 M: W2 j' l/ ~) m(1).mail:Web应用所在目录
    3 z; m3 s2 z3 J8 mhtml,jsp,css,js文件等,这些文件一般存在web应用根目录下,根目录下的文件外界可以直接访问,java类,jar包,web应用的配置文件存在这个目录下WEB-INF,这个文件夹的文件名一个单词都不能写错,这个文件要新建,在这个文件夹下载新建一个classes文件夹,放置java程序,新建一个lib文件夹,放置架包,该目录下的文件外界无法非法直接访问,由web服务器负责调用.每个web应用应该有个web配置文件,是最重要的一个配置文件,一切配置可以通过这个文件进行配置web.xml1 x! N: X6 p$ G! n: U7 U2 u; W
    : A, a/ X+ T) M' ]4 B, G2 T
    (2).对于web.xml文件,可以从server.xml中拷贝,拷贝头和尾部分.
    红红火火恍恍惚惚
    回复

    使用道具 举报

    admin    

    1244

    主题

    544

    听众

    1万

    金钱

    管理员

  • TA的每日心情

    2021-2-2 11:21
  • 签到天数: 36 天

    [LV.5]常住居民I

    管理员

    板凳
    发表于 2015-06-02 12:43:13 |只看该作者
    11.  web资源访问流程5 o1 i& }4 `8 {+ M
    (1).服务器认识war包,自动解压,放到webapps中,自动生成应用.( K) F% ^0 g' |

    1 C  W- v6 ]. l# `(2).配置context元素的reloadable="true",应用程序修改,不需要重新发布应用,建议不要配置,因为只要一修改,就要重新加载,出现问题.) y+ a# ?$ L# H2 O- Z# }0 A

    9 [; v/ R$ L0 ]+ {9 N3 [(3).context.xml中<Context>中配置,被所有的web应用使用.( U9 n2 l# k# b. c) F# c1 L) N0 w
    & v- S4 T2 `" r' v( C5 h, T
    12.  配置虚拟主机
    1 P' M) y) m- c8 L7 e$ e+ P(1).在一个tomcat服务器中可以放置多个网站,所谓配置虚拟主机,就是在tomcat服务器中配置一个网站
    : _) a/ z3 \4 j# j3 ]) w+ M
    . _5 F1 R6 |. R(2).如需在Web服务器中配置一个网站,需使用Host元素进行配置,例:<Hostname="site1"appBase="c:\app"></Host>,配置的主机(网站)要想被外部访问,必须在DNS服务器或windows系统中注册.
    : R# m9 |! O- `1 g2 z4 W- k8 O2 h
    7 w  _( _, B$ P& c& X(3).在server.xml新建一个主机,新建一个Host标签,<Host name=www.sina.com appBase="c:\sina"><Contextpath="/.mail" docBase="c:\sina\main"/></Host>
    ' r) w' \) @# I) f$ e
    1 n' u. V/ n; Y* D1 a3 M(4).互联网访问流程:ie中输入错误!超链接引用无效。 返回给ie,接着使用ip地址访问sina.com+ _2 _* l8 h# Z! @

    ; b2 |$ Q- e+ h7 I7 B(5).ie开始会访问windows的host文件,如果这个文件知道这个地址,就不会去问DNS,没有则会去访问DNS服务器:/windows/system32/drivers/etc/hosts文件进行配置.
    1 v2 L( w9 j" @+ @' P0 B$ h; _
    6 L- T) }, \: {7 A) W' B# j13. 软件密码学和配置tomcat的https连接器
    8 A& o. x% O/ m4 m(1).对称加密:加密和解密的密码是相同的.4 R6 u: g9 C) {! x
    $ {0 z- A3 V/ S
    (2).非对称加密:接受者生成一对公钥和私钥,公钥加密的数据只能私钥解,私钥加密的数据只能公钥解,然后接受者将公钥传给发送者,用接受者的公钥进行加密,然后接受者用私钥解密.然而,在传送公钥的途中可能被黑客拦截,然后黑客自己生成自己的公钥和私钥,然后将公钥送给发送者,发送者如何知道公钥是不是接受者的公钥?这是在互联网上有个叫CA的机构,接受者产生一对公钥和私钥,然后让CA认证,CA给接受者一份数字证书,这时发送者受到的是数字证书,发送者验证数字证书是不是接受者的数字证书.信任点就是CA.网上银行出示一份数字证书(公钥),浏览器可以验证数字证书正确,注册,当你填写信息时,浏览器就是用这份数字证书验证.但是数字证书可能被黑客拦截,怎么确定数字证书是接受者发的还是黑客发的,数字签名(私钥加密),发送者自己生成一对公钥和私钥,发送者想让接受者相信是自己加密的,发送者就需要签名(使用自己的私钥),然后只有接受者使用公钥解密,就相信是发送者将内容加密的.MD5算法得到数据指纹.然后对数据指纹加密.将密文和指纹加密信息传给接受者,可以判断在传送过程没有被篡改,用密文生成数据指纹,和传过来的数据指纹对比.
    6 }" _% z) s8 |& g , i: C  N7 F. F- ^9 ?
    第四天:
    5 O5 E4 i% J/ I* G+ I1.Eclipse开发servlet
    4 i3 c2 @6 }8 A( q/ S" y(1).web工程的名称,该工程部署时,在webapps目录下就会有一个example的web应用% ^$ h$ y& k2 k; ^7 g
    # N0 e+ c1 y: u: h% e5 ]3 z( `
    (2).src文件夹,Java程序的开发目录,该目录下编写的所有程序在部署时,会自动部署到example/web-inf/classes目录下
    ) x3 d( o, {% ]8 x" k, u' X9 ? : K! F$ |% T  w0 N  ]  t
    (3).WebRoot文件夹,webroot对应于web应用的根目录,该目录下的所有子目录和子文件夹在部署时,会原封不动的发布到web应用目录下2 D/ `( Y( S, h+ U

    - A/ i. A4 V$ R" h! q(4).Web rootfolder:WebRoot应用的根目录,Context root URL:映射到服务器的目录,在WebRoot目录下新建Web资源,应用发布时,将src文件夹中的源文件编译成class到webRoot\web-inf下的classes文件夹
    3 J- {' ^2 ~5 x ( U- A# M' b0 F$ @: q% i
    (5).servlet导入源码,到tomcat服务器的目录下查找文件夹
    / S- c$ m( }: F
    + O& H. E3 L' A0 F: A0 j(6).为servlet配置对外访问路径,% H, P( \4 O2 M3 u) H
    <servlet><servlet-name>! J- Z# e2 x; n
    别名
    0 Y( m6 @' a+ }. ~, P4 V: u' i9 J5 T# [</servlet-name><servlet-class>( C. e- b; s2 u/ t2 X' X
    类名(全称)
    1 K; G, L0 }& _8 g% a! z</servlet-class></servlet> 3 a0 ?) K& J; z8 ?
    <servlet-mapping><servlet-name>  H. z+ A2 ~% I* w2 m" Q4 p& z
    别名
    , ]5 w3 Q( G5 d</servlet-name><url-pattern>/aa<url-pattern></servlet-mapping>
    1 m3 u* O( A8 J, z( ^+ y& k, V 9 S* s: [' l( C6 y" V, ]) G
    (7).还可以更改tomcat的运行java环境, H6 V. s+ q& i9 m$ k

    2 h" ^$ K1 |7 Q% h% Q2. HttpServlet和一些开发细节6 G+ n$ L8 K# q# K
    Servlet接口sun公司定义了两个默认实现:6 }: Y# K, j" V  D: J( y
    GenericServlet,HttpServlet
    6 C) B" Y+ U( I- j8 o' X$ Phttpservlet指能够处理HTTP请求的servlet,它在原有的servlet接口上添加了一些与http协议处理方法,它比servlet接口的功能更为强大,因此开发人员在编写Servlet时,通常应继承这个类,而避免直接去实现Servlet接口; m7 C* i, A/ Q& g$ S( X. L
    httpservlet在实现servlet接口时,复写了service方法,该方法体内的代码会自动判断用户的请求方式,如为get请求,则调用httpservlet的doGet方法,如为post请求,则调用doPost方法,因此,开发人员在编写Servlet时,通常只需要复写doGet或doPost方法,而不要去复写service方法.可以查看httpservlet的源码,
    0 s5 ?1 K; f  v修改servlet的模板代码.进入myeclipse,搜索Servlet.java文件,打开修改文件.
    : V$ `7 U* x1 ]! D) N$ w2 ^) m
    2 S3 w; R  y  _3.ServletConfig对象以及它的开发场景
    5 W8 ^1 x5 R2 F$ n- C% ~0 ]' H3 v(1).在servlet的配置文件中,可以使用一个或多个<init-param>标签为servlet配置一些初始化参数,当servlet配置了初始化参数后,web容器在创建servlet实例对象时,会自动将这些初始化参数封装到servletconfig对象中,并在调用servlet的init方法时,将servletConfig对象传递给servlet,进而,程序员通过servletconfig对象就可以得到当前servlet的初始化参数信息.+ N$ {! b3 x" A
    5 `; i% n, k& N. F& \
    (2).web服务器创建的容器对象4 C/ S8 u4 S% s- z
    request,response,servletconfig,servletcontext,session,cookie4 O( x! c! K! a

    8 H% P6 e6 o- P9 `2 W- _, [, j2 A(3).servletconfig
    / e, w0 K5 P- k7 T5 _( P配置参数信息.
    : S: @; _7 {, h( z+ y3 q<init-param><param-name>data</param-name><param-value>xxxx</param-value></init-param>& h4 c2 J6 |0 P5 a
    , J9 j& M+ g* w# v" ~
    (4).覆写init()方法带有servletconfig参数的,服务器自动将参数信息封装到servletconfig对象中.调用init()方法时,servlet可以得到这些参数信息.在servlet中定义一个servletconfig对象,将init()传递过来的servletconfig接收到.private ServletConfig config;public voidinit(ServletConfig config){this.config = config;}String value =config.getInitParameter("data");System.out.println(value);参数信息不在程序中,而是在配置文件.同时this.getServletConfig();代替privateServletConfig config;因为父类已经定义了ServletConfig对象,只需调用父类的getServletConfig方法即可.( z) X# D% U: p* r( L

    $ z( A0 o5 f, \(5).可以配置多个参数信息,同时为得到所有的参数信息:
    0 x; r1 o# x3 |* C1 c0 NEnumeration e =this.getServletConfig().getInitParameterName();
    ' d, X3 b' [, n) A, s. nwhile(e.hasMoreElements())
    9 l4 ?# x7 n3 K1 p% O6 R% H6 C{
    ( V# J! I/ s# u, o2 i, PString name = (String)e.nextElement();& w' b8 V2 ^  V! v& U
    String value =this.getServletConfig().getInitparameter(name);0 _3 N8 z! c/ z; D7 V9 d
    System.out.println(value);5 ^: ^9 l- Z% ]! L
    }0 `4 A# B5 N* b3 S
    (6).开发过程中,参数不能在程序中写死,需要在配置文件中给出参数信息.实例:字符集,连接不通的数据库.读取不同的配置文件." |; c* Q! I3 S/ B- s

    2 @1 t& t  m4 S2 w  Q(7).struts就是个实例.可以到struts实例中查看struts的web.xml文件" z, w/ {! J- O; Y  f
    # l8 b& Z4 I* w0 ]. W# y* |
    4.  ServletContext读取web应用中的资源文件
    4 j0 N3 ]* \8 @1 u! f% V3 W( h$ x(1).在web开发中,用于资源文件的类型一般为两种xml,property,如果数据之间是没有关系就是用property,有关系就使用xml,数据库的连接关系没有关系,一般用property文件,
    8 d( t+ P: V5 b" g
    , X! I7 d' }; G. D, S  f, Q(2).db.properties文件,url=jdbc:mysql://localhost:3306/test;username=root;password=root;读取db.properties文件.通用代码:
    ; i1 o# ~$ F& t5 \InputStream in
    1 e& X) w- W2 k9 N) B# O  g= this.getServletContext().getResourceAsStream("/WEB-INF/classes/db.properties");
    7 d' y% K- I6 |) F& [* r斜杠就代表web应用根目录,Properties props = new Properties();props.load(in);把文件load到Properties对象中,用Map对象存储String url =props.getProperty("url");
    / e& q  I2 |% ^  PString username =props.getProperty("username");
    * [" [% g2 O* ]String password = props.getProperty("password");
    ( R; |1 f7 B" G3 b- X+ ^) ~4 Q8 E " g' ~8 J9 w0 y# Q, T; }6 x! a% A
    (3).web应用中不使用FileInputStream in = newFileInputStream("src/db.properties");采用的是相对路径,相对路径中有这个资源,这是相对路径相对的是jvm的启动目录.当时jvm的启动目录是服务器的启动目录:服务器/bin7 a9 E1 B* h; O- j# R; ?. r
    - [; D6 k$ O5 i
    (4).另外的读取资源的方法:- b2 {  a! ~# Z; `6 U; v1 k! `2 j
    String path = this.getServletContext().getRealPath("/WEB-INF/classes/db.properties");得到硬盘的绝对路径 此时就可以使用FileInputStream in = new FileInputStream(path);需求:下载,可以得到资源的名称,读取文件时客户机带来的,当前读取的资源文件的名称.可以从path中截取出来./ V3 ]* c" Q4 b! T: K/ y
    String filename =path.substring(path.lastIndexOf("\\")+1);( {( R4 D  u6 v
    ; z# Y4 N& G0 Q% K4 W  p2 a
    (5).在servlet中用dao对象操作数据库,dao对象需要properties文件的内容即数据库的连接信息.dao一般放到cn.itcast.dao包中.不能将servlet中的context对象传递给dao对象,web层和服务层不能有交织了.如果读取资源文件的程序不是servlet的话,此时要通过类装载器读取资源,在UserDao对象中InputStream in =UserDao.class.getClassLoader().getResourceAsStream,资源文件("db.properties");一般只需读取一次,有个习惯:将读取文件代码放在static{};静态代码块中.通常抛出初始化错误throw new ExcepitonInitaliterError(e);文件不能太大,类加载器要装载到内存中% `) X) P+ [& O' v0 C2 N& q0 j

    . J/ I( B4 }6 Y3 a5 y4 Z$ V(6).有人可能把properties文件改了.然后读取的内容是原始的,还是修改的,读取的是原始的,因为类装载的方式,但是类只装载一次.但是想读取更新后的内容:String path =UserDao.class.getClassLoader.getResource("db.properties").getPath();然后用FileInputStream去读取.; \8 \# ]7 X( |

    # N5 ?' I$ H4 o/ r3 I5.  ServletContext对象
    5 A4 E5 i+ ^, Q8 c- u5 |8 I(1).web容器在启动时,它会为每个web应用程序都创建一个对应的ServletContext对象,它代表当前web应用,servletconfig对象中维护了servletcontext对象的引用,开发人员在编写servlet时,可以通过servletconfig.getServletContext方法得到servletContext对象.同时也可以context = this.getServletContext();得到父类的context查看servletContext api 文档了解servletContext对象的功能.
    6 |1 U: |5 k' e$ B$ v7 i0 I0 K, E6 c * `) e+ a6 s2 [8 {
    (2).servletContext为所有的servlet对象实现数据共享.是全局概念.方法getContext()为了得到别的web应用.getInitParmeter()方法可以得到web应用的配置信息.getNameDispatcher()得到一个转发对象.getRealPath()方法得到资源的硬盘真实路径,方法getResource()得到资源的路径,只是返回的是URL地址.getResourcePaths()方法返回给定目录下的所有资源.getServletContextName()方法返回web应用的名称.setAttribute(),getAttribute(),removeAttribute();
    . I$ D2 w7 A) ^  O
    ( N: o# c& b: x, n' O8 T(3).由于一个web应用中所有servlet共享同一个servletcontext对象,所以多个servlet通过servletcontext对象是实现数据共享。servletcontext对象通常也被称之为context域对象.可以获取web应用的初始化参数,实现servlet的转发,利用servletcontext对象读取资源文件:得到文件路径,读取资源文件的三种方式,.properties文件(属性文件),context域,request域,session域,page域.
    # ]. h' n5 u& D; s: P
    6 i& o9 {& w# z; l( E: f(4).实例:聊天室,资源共享.开个浏览器说,开个浏览器读! i, d& V  C, U: T( {  a8 l# R

    5 C) I( i& s4 L% A' m& P# Q% S4 u(5).<context-param><param-name>data</param-name>
    - y% ^+ Q2 p. @: F<param-value>XXXX</param-value></context-param>; u4 G, j7 {. q1 h1 Q
    和config不同,config是给某个servlet初始化参数,而context是所有的servlet初始化参数
    6 G& @  I* k" J9 @" t
    # p) w& S5 C9 m% F& ]8 n1 _(6).servletconfig中的初始化信息中连接数据库信息,应该写成context的初始化信息,因为可能有多个servlet需要数据库连接; d7 f" x; B9 d$ j
    * n& [7 I9 `1 P4 N1 b
    (7).转发和重定向:转发:你找我,我帮你去找他,重定向:你找我,我告诉你,你自己去找他;如果是转发:只有一次请求,地址栏不发生改变重定向:两次请求。servlet中不适合浏览器的数据的输出,因为要数据的排版,应该在html,jsp中输出数据.servlet要将输出数据要转发给jsp显示.
    ' e" ^- n6 s( U1 yRequestDispatcherrd = this.getServletContext().getRequestDispatcher("/1.jsp");
    + {  o+ {; C0 ~# g8 Y6 o* i* \" Ard.forward(request,response);
    5 u/ P9 h; n" L; Z0 Y1 L% r2 D+ d在jsp中取数据,
    : e* T. D9 |0 P<% String.data=application.getAttribute("data");out.write(data);%>servletcontext可以,但是在开发过程中是有问题的:有一个人来访问数据,把它的数据存到context中,当他在jsp中取数据的时候,此时又来一个人,也来存数据,可能覆盖前一个人的数据,那前一个人取得数据可能是后来的那个人的数据,因为context是共享资源,存在线程安全问题.在开发过程中一般用request域带数据.
    ! b! c' U% \) t8 X. b
    6 D3 V% ~9 R1 U(8).服务器启动时,为每个web应用创建servletcontext对象,当服务器停止时,servletcontext对象被销毁.+ o$ K. C6 @1 l

    " n2 p: i" o' B6.  Servlet的生命周期, a9 j  \% h4 ?$ z% ?7 |' C
    ie浏览器,web服务器,Servelet程序; F8 X( z0 D0 O9 E5 r& F
    浏览器输入地址->连接web服务器,发送HTTP请求,解析出客户机想访问的主机名,解析客户机想访问的web应用和web资源-->使用反射技术创建一个servlet实例对象(servlet第一次访问的时候被创建)-->调用servlet.init()完成对象的初始化->调用servlet.service()响应客户端的请求,服务器创建request,response容器,service方法执行->向response容器中写入客户机请求的数据->服务器从response中取出数据构建一个HTTP响应回写给客户机-->回写http响应
    ; M- [% u/ T# c2 M1 V客户机第一次访问服务器,生成一个servlet,第二次在访问同一个servlet就不会再创建了,当关闭服务器时servlet就消亡了
    4 C' ]/ m- l, D( V2 L. U) g5 _ ; m7 H# }8 H$ x! L/ j
    7.  Servlet的线程安全& L' @$ ]$ R' {0 f5 k
    (1).当多个客户端并发访问同一个servlet时,web服务器会为每一个客户端的访问请求创建一个线程,并在这个线程上调用servlet的service方法,因此service方法如果访问了同一个资源的话,就有可能引发线程安全的问题,如果某个servlet实现了SingleThreadModel接口,那么servlet引擎将以单线程模式来调用其service方法。SingleThreadModel接口中没有定义任何方法,只要在servlet类的定义中增加实现SingleThreadModel接口的声明即可.对于实现了SingleThreadModel接口的servlet,servlet引擎仍然支持对该servlet的多线程并发访问,其采用的方式是产生多个servlet实例对象,并发的每个线程分别条用一个独立的servlet实例对象。实现SingleThreadModel接口并不能真正解决servlet的线程安全问题,因为servlet的引擎会创建多个Servlet实例对象,而真正意义上解决多线程安全问题是指一个servlet实例对象被多个线程同时调用的问题,事实上,在servlet api2.4中,已经将SingleThreadModel标记为Deprecated(过时的).标准的解决方案是同步方式sychronized
    8 H4 L9 R. N, S   K) p) U! G& z$ |
    (2).如果线程向person的静态list集合中加入了数据(aaa),数据用完后,一般要移除静态集合中的数据(aaa),否则集合中的数据越来越多,就会导致内存溢出。对象销毁了,静态资源的字节码仍然驻留在内存中.) ^: @! h' S$ _" U

    8 R4 j+ Q8 j" ~# N(3).web中的异常一般不能抛,要catch住.% W5 a) h4 w9 O0 P+ f% k/ X

    + Y/ B' ?5 L# p8 a" Q8.  Servlet开发的一些重要细节6 Y: S# M: c: n- |8 S) B" p
    (1).由于客户端是通过URL地址访问web服务器中的资源,所以Servlet程序若想被外界访问,必须把servlet程序映射到一个URL地址上,这个工作在web.xml文件中,使用<servlet>元素和<servlet-mapping>元素完成  }' v. x+ K( c: h  M6 v3 W

    - V. N$ P: m* l(2).<servlet>元素用于注册Servlet,它包含有两个主要的子元素:<servlet-name>和<servlet-class>,分别用于设置Servlet的注册名称和Servlet的完整名.. ^3 h! L# F) }; X0 B
    一个<servlet-mapping>元素用于映射一个已注册的Servlet的一个对外访问路径,它包含有两个子元素:<servlet-name>和<url-patteren>,分别用于指定Servlet的注册名称和Servlet的对外访问路径3 p* K# H+ p5 v7 U
    (3).进入myeclipse->Web更改Web-Content-root
    ( D+ K9 Q" I, m3 w/ m, w - n; D4 ]8 e" s! D7 ~; D: J  T
    (4).一个servlet可以映射到多个对外访问路径.只需多复制个
    ' X- c1 n" ~! l+ Y<servlet-name><url-patteren>(伪静态)2 Y  |  C  N6 m5 Y4 z, J2 y

    2 e# d8 w0 p! r1 @(5).同一个Servlet可以映射到多个URL上,即多个<servlet-mapping>元素的<servlet-name>子元素的设置值可是同一个servlet的注册名1 n( c, h9 E6 P9 S
    5 b- a5 I( s" O( M
    (6).在servlet映射到的URL中可以使用*通配符,但是只能有两种固定的格式:一种格式是:"*.扩展名",另一种格式是以正斜杠/开头并以"/*"结尾的./ X4 s; h( I8 _  e
    <servlet-mapping><servlet-name>anyName</servlet-name>* c& U+ Y2 b7 O8 ~; L
    <url-pattern>*.do</url-pattern></servlet-mapping>
    * k: k2 E* ^" g# f- F0 i$ b3 k/ |匹配所有后缀名为do的访问路径(也可以不需要后缀名, 匹配所有的文件)<servlet-mapping><servlet-name>anyName</servlet-name><url-pattern>/action/*</url-pattern></servlet-mapping>服务器在更改web.xml时,无需重启,<WatchedResource>WEB-INF/web.xml</WatchedResource>在服务器的conf文件夹下的context.xml配置文件中,监视web.xml是否发生改动
    . }4 ?8 Y5 n( W3 N  _, O$ v/ h; C
    2 ~# M* h2 \" `5 @: U(7).servlet1 映射到 /abc/*
    8 H9 M6 K& I9 C# Z% N$ ]          servlet2 映射到 /*6 y: Q( m2 Z- X
              servlet3 映射到 /abc- X# z# D; R  k
               servlet4 映射到 *.do
    9 n7 h- @  Z/ S- K& D: |# |# }           问题:
    0 @. {" C% H3 G: L8 R1.当请求URL为"/abc/a.html","/abc/*"和"/*"都匹配,但是servlet引擎会调用servlet1
    % y9 A$ _% H- b# ^2.当请求URL为"/abc"时,"/abc/*"和"/abc"都匹配,但是servlet引擎会调用servlet3* o& x# _3 [% q  u* w
    3.当请求URL为"/abc/a.do"时,"/abc/*"和"*.do"都匹配,但是servlet引擎会调用servlet1
    : y  U- E. k2 V% [2 V4.当请求URL为"/a.do"时,"/*"和"*.do"都匹配,但是servlet引擎会调用servlet26 ^: M; X- E  y5 f9 ^5 P
    5.当请求URL为"/xxx/yyy/a.do"时,"/*"和"*.do"都匹配,但是servlet引擎会调用servlet2
    3 L! M/ H: E8 d/ z* g/ Y总结:谁长的最像,谁先匹配,同时*的优先级最低.1 g- E) J, D7 Y+ W* u8 k6 e1 Z6 b
    ; X% J8 J: |* t8 ]
    (8).Servlet是一个供其他Java程序(Servlet引擎:服务器端调用servlet的程序)调用的Java类,它不能独立运行,它的运行完全由servlet引擎来控制和调度.针对客户端的多次servlet请求,通常情况下,服务器只会创建一个servlet实例对象,也就是说servlet实例对象一旦创建,他就会驻留在内存中,为后续的其他请求服务,直至web容器退出,servlet实例对象才会销毁.在servlet的整个生命周期内,servlet的init方法只会被调用一次,而对一个servlet的每次访问请求都导致servlet引擎调用一次servlet的service方法,对于每次访问请求,servlet引擎都会创建一个新的httpservletrequest请求对象和一个新的httpservletresponse响应对象,然后将这两个对象作为参数传递给她调用的servlet的service方法,service方法再根据请求方式分别调用doXXX方法.
    1 r' K0 D1 I' ]; ^! s+ i
    , i' x8 U( u+ X/ \& i(9).右击,可以选择父类的方法,进行重写父类的方法
    ! }9 j9 w6 R5 j1 t6 O/ v# `) y/ ?
    ( k3 j" v( w/ H+ H" K0 ?) _(10).服务器启动时,servlet并未创建,只有当访问web资源时会创建一个servlet,调用init()方法.一个servlet为多个请求服务,当服务器停了,servlet就被摧毁了,调用destory()方法.$ M) F: C- P* W" F  c2 t7 b6 w
    ' {; [: f, J% b; }
    (11).针对客户端的每一次请求,服务器端都会为用户创建一个request和reponse,他们的生命周期很短,如果是并发的请求,大量的请求可能创建大量的request和reponse对象,可能导致服务器运行变慢,但是不是并发的话,可能不会创建那么多个对象( D; f) {, P" Y$ g" g3 Y

    , j% d) c% q' a. A* h( D4 C(12).如果在<servlet>元素中配置一个<load-on-startup>元素,那么web应用程序在启动时,就会装载并创建servlet的实例对象,以及调用servlet实例对象的init()方法,<servlet><servlet-name>invoker</servlet-name><servlet-class>...</servlet-class><load-on-startup>2</load-on-startup>;用途为web应用写一个initservlet,这个servlet配置为启动时装载,为整个web应用创建必要的数据库表和数据.(即启动服务器时就创建了servlet),struts就是一个非常大的servlet,其中数字的大小就是启动的优先级,数值越小优先级越高.struts中的数值为2,担心用户需要扩张,所以不使用1.
    3 J( q) m! j3 V- S, l  _* M
    . F" q  A; b: H7 v4 O1 C, D(13).如果某个servlet的映射路径仅仅为一个正斜杠/,那么这个servlet就成为当前web应用程序的缺省servlet;凡是在web.xml文件中找不到匹配的<servlet-mapping>元素的URL,它们的访问请求都将交给缺省servlet处理,也就是说,缺省servlet用于处理所有其他servlet都不处理的访问请求.在<tomcat的安装目录>\conf\web.xml文件中,注册了一个名称为org.apach.catalina.servlets.DefaultServlet的Servlet,并将这个servlet设置为缺省servlet;当访问tomcat服务器中的某个静态html文件和图片时,实际上是在访问这个缺省servlet。这个用途主要用于:用户请求一个数据(html,jpg,avi等)服务器将启动一个缺省的servlet提供给客户数据.在浏览器看到的任何内容都是数据都需要由一个servlet来为客户提供数据,包括错误信息.- L8 c0 G' n/ _9 Y8 U
    / J- @! j! q9 ^; L" B
    (14).配置文件修改了,不需要重启服务器,但是源程序修改了,要重启服务器.- E+ E/ X+ Z+ Y8 I4 x. w8 H
    9.  Servlet开发入门( j8 C, g" m1 r0 n% x; k! W+ l
    (1).jsp就是servlet,servlet是sun公司提供的一门用于开发动态web资源的技术,编写一个java类,实现servlet接口.获得servlet启动信息,通过getServletConfig(),service的两个参数,ServletRequest req,ServletResponse res,这个方法是服务器调用的,获取客户机的对象,只需找request,输出信息找response.getOutputStream();6 D7 y3 o- K4 K/ C! U' A
    (2).进入服务器的目录,新建一个web应用,在应用中新建一个java,class,lib,在文件夹java下新建一个servlet文件,. [' q/ @! a1 J, l
    package cn.itcast- b- s% p1 j) M6 h# _
    public classFistServlet extendsGenericServlet{
    $ S! S2 w$ _9 ?! O* U- U* Ipublic void service(
    2 O/ S: b# T& a' K! \7 M9 b8 qServletRequest req,ServletResponseres)throws ServletException,java.io.IOException{
    / N! a" f* O9 tOutputStream out = res.getOutputStream();
    * |) u  s7 ^: _5 V  o3 B% }- }* A( M' Qout.write("Servlet".getBytes());& v  y' G  Y! o% v( O; w0 k
    }# f0 s! z+ V: E9 J
    }
    / `) l! @: a. N4 a& B& w5 H(3).servlet所需的架包导入,tomcat能运行servlet,则在服务器的lib目录下含有架包,为servlet配置对外访问路径web.xml,代码可以到服务器的web.xml文件中拷贝
    红红火火恍恍惚惚
    回复

    使用道具 举报

    admin    

    1244

    主题

    544

    听众

    1万

    金钱

    管理员

  • TA的每日心情

    2021-2-2 11:21
  • 签到天数: 36 天

    [LV.5]常住居民I

    管理员

    地板
    发表于 2015-06-02 12:43:29 |只看该作者
    第五天:. O8 z& w" x- U. ?
    1.     HttpServletResponse简介
    . S2 ~+ u$ }2 g(1).web服务器收到客户端的http请求,会对每一次请求,分别创建一个用于代表请求的request对象,和代表响应的reponse对象.request和reponse对象既然代表请求和响应,那我们要获取客户及提交过来的数据,只需要找request对象就行了,要向客户机输出数据,只需要找response对象就行了.9 ~1 ?3 P: o; g4 `' ^. d9 Z

    & e5 A; a; A5 k3 B(2).reponse的相关方法,构建一个http响应的方法,getStatus(),setHeader(),getWriter(),getOutputStream();字节流可以写任何数据
    & y# A- q) ]8 O+ M2 }+ e0 ]
    5 W1 }" ^1 o5 h+ t0 Q% J1 Z2.request获取请求头和请求数据
    + l( v/ D$ x6 j! S! T(1).用户点击链接和表单提交向服务器提交数据9 ^8 m5 S8 W1 v1 H- K: }2 c0 K: [; a0 |
    : f2 y0 g* Q7 p7 L5 O& B
    (2).在服务器端获取数据:& d$ }8 `4 ?" C- t6 t* U, l) B( @
    request.getParameter("username");5 @' T3 `- D6 a7 N& V5 R
    request.getParmeterName();. X( q+ D" p7 p8 E: Z
    得到所有的名称的枚举.) w8 A6 n. N* i8 n# y  H- v
    request.getParameterValues();, D' H- ?1 t# ^& [/ l% O: }
    得到名称相同的值
    # \& x' H7 A& A* }# l9 n. f(3).在用户提交的数据中,可能抛空指针异常,加强代码的健壮性.
    5 a3 {; v/ |+ K
    # C- Z4 \" O- r. x2 w- o(4).Mapmap=request.getParameterMap();将用户的数据封装到对象中User,先获取Map,再用JavaBeans将Map转化成对象User,其中map关键字是请求参数名称,关键字的类型是String,值的类型是String[],因为可能有多个相同名称对应的值就有多个,所以是个数组;Map将值填充对象User,迭代Map,得到关键字和对应的值,利用反射技术将这些数据填充到User对象中.设计一个对象代表表单.BeanUtils.copyProperties(user,formbean);将表单beans拷贝到user对象beans中;BeanUtils.populate(user,map);对象上的字段必须是八种基本类型,不是的话,要自己写个转换器.
    ) E0 \1 E. F8 s  L/ f
    & D" z  }' J; v( c(5).服务器以断点的方式运行进行调试,1 {  i2 N+ s$ _) N# u  |! ]$ q2 y2 I

    4 e$ n" L0 ~* }" m( I(6).request.getInputStream();获取输入流,进行获取数据,进行文件上传.
    / v1 j1 p! k" D( }) ^1 z3.  request简介.txt& a7 ~# X- ~% y% N9 t1 ~
    (1).HttpServletRequest对象代表客户端的请求,当客户端通过HTTP协议访问服务器时,HTTP请求头中的所有信息都封装在这个对象中,开发人员通过这个对象的方法,可以获得客户这些信息.
    9 R* Z9 l- _# |* E( X
    % C! r6 `, j: d(2).getMethod();获取请求方式,getRequestURI();获取客户想我访问的资源;getHeader();getDateHeader();getHeaderNames();获取所有的头信息,getHeaders();与getHeaderNames();不同的是:可能有同名的头信息,此时就要用getHeaders()方法,a-xx;y-xx;getParameter获取客户带来的数据name="aaa";password="rot";getParameterNames();getParameterValues();getInputStream():将用户的数据当做流读入,如:客户进行文件上传;
    : @, O9 ~& n: ]8 W0 Z9 }/ {
    5 ?; h; W: h! ?0 s# G0 p% w  s(3).getRequestURL():/day07/servlet/demo1;2 l6 k4 X8 I$ z. v
    getRequestURI():http://localhost:8080/day07/servlet/demo14 t0 G- u. |, \3 {& }
    用在权限拦截上,不同的网页有不同访问权限,做个拦截器,过滤器,还有是在记录页面的访问次数.getQueryString()客户携带的查询信息,地址栏后的类容,getRemoteAddr();得到访问者的ip地址;getRemoteHost():主机在DNS注册了,打印了主机名.getRemotePort();来访者的端口,ie也是一个程序,需要一个端口getLocalAddr();返回web服务器的ip地址,getMethod():获取请求方式,默认是get方式- G, C' h" s/ O- P  U2 @
    ( k) ~! E8 @( e0 }
    4.request乱码
    0 l/ p0 G* ?, d# G! I" B" M$ j  i(1).在文本框中敲入中文,浏览器提交数据的码表就是浏览器使用哪个码表打开.request容器也是默认码表是iso8859码表,所以在获取数据时,request.setCharsetsEncoding("UTF-8");这种改变码表的方式只对post方式有效,对get方式无效.这种方式解决的是post方式提交的数据.
    0 z- f/ B: P2 w9 {/ N3 j3 J$ L , T; S5 x, S) ?- ^
    (2).但是这种方式不能用于get方式,如果是get提交,必定是乱码,所以必须在servlet中手动的改变乱码,newString(username.getBytes("iso8859"),"UTF-8");7 G" \# E7 z! u" Q/ h
    & j9 @$ w2 q1 S/ E, W, v  E
    (3).用户使用超链接提交数据时,带有中文,其实超链接就是一个get方式,所以要使用get方式去解决乱码问题.3 p$ A3 d5 m" T+ {/ q' V

    ( Y) e' W! s/ ](4).更改服务器的配置,解决乱码问题,但是在开发中严禁这种方式的使用,从浏览器中进入tomcat,修改连接器,HTTP->URLEncoding,然后到tomcat目录下的server.xml中修改连接器Connector,在其中添加一个属性:URLEncoding="UTF-8";还有一种方法,在连接器中添加一个属性,useBodyEncodingForURI="true";是将post方式解决乱码的方法可以用于解决get方式乱码.所以还要事先设置好request容器的码表.7 W; T/ W1 J4 K, @3 O0 K! s3 H
    : F7 m: N7 D' M
    (5).request.setCharacterEncoding("UTF-8");. o# u" Q% c/ N7 r, U
    String username =request.getParameter("username");  r0 c5 A- `# g0 w$ z. @
    response.setCharacterEncoding("gb2312");
    # K2 t6 ~; `- D- `6 D& R2 W# D" Hresponse.setContentType("text/html";charset="gb2312");
    4 Y* m1 [3 C+ h$ cresponse.getWriter.write(username);不会有乱码; ]- `3 k! Y) y5 _
    5.request实现请求转发和mvc设计模式.txt
    2 Z$ E+ G6 S6 k) D' W! k  u4 Y: [(1).request是个域对象,request作用的范围是请求范围中,主要用于设计mvc中.
    ( r+ {2 C- k, e. `4 a. r
    0 Z7 j! Z  X( Q(2).servlet中转发到jsp显示数据,
    . Q- Y7 R$ g# Qrequest.getRequestDispatcher("/mesage.jsp").forward(request,response);
    1 \' K2 L, F# v: A& D6 q在开发中一般都是用这种方式,而不是用ServletContext的转发方式.每一次请求都对应一个request,正好,request的作用域是一个请求,所有request的转发,都是一个请求,都在这个作用域中./ D/ z0 a0 m4 h/ ?

    ; m" \; \. d& W/ b* O(3).request.getParameter()方法是获取客户机带来的数据,而request.getAttribute()方法是获取request域中获取数据.
    / o/ }9 h4 s/ f8 `- N , B' ~5 K8 C* m" x- j( r" c
    (4).forward方法用于将请求转发到requestDispatcher对象封装的资源,如果在调用forward方法之前,在servlet程序中写入的部分内容已经被真正的传送到了客户端(关闭response流),forward方法将抛出IlegalStateExcepiton异常,如果在调用forward方法之前向servlet引擎的缓冲区(response)中写入了内容,只要写入到缓冲区中的内容还没有被真正输出到客户端,forward方法就可以被正常执行,原来写入到输出缓冲区中的内容将被清空,但是,已经写入到HTTPServletResponse对象中的响应头字段信息保持有效.
    1 D3 ?. t" O: }: x* g" R! {: a" P % p1 \# |/ {1 a% b2 `
    (5).if(true){request.getRequestDispatcher("/index.jsp").forward(request,response)}) ?5 X" Z( U0 w2 G, H( P4 L9 r6 [
    跳转到index.jsp向客户机写数据, M# ?  f5 I' [( Q5 u- X, |
    request.getRequestDispatcher("/index.jsp").forward(request,response)也要向用户写数据。所以一个良好的习惯是跳转之后一定要return;
      h- E! ~; D) o1 L; w& }7 {
    " t8 k+ Z0 I) o8 u(6).String data ="aaa";response.getWriter().write(data);不要关闭这个流,这是可以request.getRequestDispatcher("/index.jsp").forward(request,response);理论上应该是两部分内容的组合,即data和jsp中的内容,但是实际上只显示jsp中的内容,因为jsp的内容将覆盖data内容,但是消息头不改变,在这个过程中创建了两个servlet,因为jsp也是一个servlet,但是只有一起请求.不会创建两个容器.
    8 N: m* D' V( ]# K. F/ P* v 2 c$ c! G% ^, e+ @
    6.     request实现页面包含
    ) P% U7 Z" N3 M; q所有的网页都有相同的网头和网脚,此时只需要将网头和网脚设置共同的部分html,request.getRequesetDispatcher("head.jsp").include(request,response);response.getWriter().writer("hhh");request.getRequestDispatcher("foot.jsp").incude(request,response);被包含页面不能出现全局架构标签,因为可能重复.实现页面包含的作用% U$ n5 z# a% d) ~: w2 G
    2 `7 t0 t: Q) Q
    7.response的outputstream输出数据的问题) \$ q/ T9 v4 X( X
    (1).在程序中的"中国"变成了UTF-8码表,但是浏览器默认的是gb2312打开内容,在程序中发送一个头,告诉浏览器用哪个码表打开.
    ( g5 j/ m8 {0 R6 U( H' V" [response.setHeader("Content-type","text/html;charset=UTF-8");
    ' b, {" R" x- u* T6 j3 u! G. N  j
    . W; @! L/ J  ^8 h; z(2).html中的标签<meta>可以模拟一个http响应头.<meta http-equiv='conten-type'content='text/html;charset='UTF-8'>;: i: n! P% R! \9 G3 P8 s
    - N' k6 u( g! c" N& i7 M6 M
    (3).当Content-type写成content-type就变成下载了., w; X) g- D2 {* j  e
    , b% B& T7 R; m6 p( E
    (4).out.write(1);在浏览器上显示的并不是1,因为1对应的码表.
    ) e$ {/ p& ~* J7 c7 Y% {' T$ A
    $ y& {# }1 @0 F. r  Z1 P$ Q8.response的writer输出数据的问题.txt$ O4 H3 H! ?% \3 c% V' z
    (1).String data ="中国"rintWriter out =response.getWriter();out.writer(data);有问题显示乱码- z7 A) y, w7 n
    & H. _( {( K- i2 o% R
    (2).servlet将"中国"写入response中,但是response用的是iso8859码表,对应没有" 中国",所以在浏览器中看到乱码,response.setCharacterEncoding("UTF-8");设置response的码表为UTF-8;同时也要告诉浏览器以utf-8码表打开.
    $ p5 Q% U0 c) F9 _! o. i/ m ) b5 A# O4 }% B6 N  Y
    (3).简便的调用response.setContentType("text/html;charset="UTF-8");可以代替以上两条设置码表的代码.
    # s# L8 g: Y. `3 e8 n6 c, E5 @$ e
    % W5 `0 M. p9 ]. U6 R( K0 O  {3 H9.response实现请求重定向和response的一些细节
    1 \' d  {- F% J8 \(1).怎么实现请求重定向:发送状态码302,
    4 U$ M; Y! ?% y4 W& iresponse.setStatus(302);response.setHeader("location","/day06/index.jsp");
    0 t: z% L9 w+ w! L" ]同时sun公司提供了一个方法response.sendRedirect("/day06/index.jsp");重定向的特点是地址栏发生改变.可能加重服务器的负担,因为有多次请求,一般用于用户登录,因为转发,地址栏不变,用户看不到是否跳到首页了,重定向的地址栏变化,可以告诉用户跳到首页,还有一种是购物,点击购买,服务器的一个servlet帮你买成了,servlet会跳到购物车显示页面,显示你购买的东西.当使用转发时,用户点击刷新,可能又买了一次,重新干一次就不好了,所以用重定向,其他情况下都最好使用转发,减轻服务器的负担! S3 F4 J% T" }9 u3 G

    * Y8 A/ W' K$ A" w9 N* y(2).getOutputStream和getWriter方法分别用于得到输出二进制数据,输出文本数据的servletoutputstream,printwriter对象,但是这两个方法是互斥的,条用了其中的任何一个方法后,就不能再调用另一个方法,servlet程序向ServletOutputStream或printwriter对象中写入的数据将被servlet引擎从response里面获取,servlet引擎将这些数据当做响应消息的正文,然后再与响应状态行和个响应头组合后输出到客户机,servlet的service方法结束后,servlet引擎将检查getWriter或getOutputStream方法返回的输出流对象是否已经调用过close方法,如果没有,servlet引擎调用close方法关闭该输出流对象/ R) l  R+ p. u8 Y7 a' X
    / [. _9 _1 C. y/ h8 `8 ^4 L6 \
    (3).当在demo1中使用了getOutputStream()方法,转发到demo2,但是在demo2中使用了getWriter();就是在一条调用链中不能存在这两个方法,当使用重定向了,有两次请求,产生了两个response对象所以不会有问题,但是转发就有问题了.
    ( M1 D+ z- N( c* K) z7 G, y3 B 6 B* U8 q# h/ H  Y7 n  r; ?
    (4).无需关心流的关闭,servlet引擎会自动关闭,但是你自己定义的流需要自己关闭,引擎不会去关闭的.
    , |4 \1 D% O4 f; x; u1 O% h0 j: V * z7 y- {% ]8 E( s7 R; s
    10.response实现文件下载* Y0 Q/ w7 _6 m5 d1 L
    (1).在webroot应用下新建一个download文件夹,存放下载资源.
    % B2 _, H/ w" b4 ?: l当下载的文件是中文文件,则文件名需要经过url编
    8 ^2 G( u( y1 P5 J码,URLEncoder.encode(filename,"UTF-8");filename是下载文件名.
    5 O' a& i/ t9 g
    0 m) {8 y% A3 S0 O& ?5 j" Z5 C11.web工程中各类地址的写法
    3 A; y( K6 S) `2 }(1).request.getRequestDispatcher("/form1.html").forward(request,response);给服务器用的
    * l) ]' i9 C, b5 @4 J, v3 }" e
    * u; ?9 v, U5 C- W: \  e" ](2).response.sendRedirect("/day05/form1.html");给浏览器用的
    5 g, U9 ^( E) Q / k% n7 [4 @% a; a! M1 z' R
    (3).this.getServletContext().getRealPath("/form1.html");给服务器用的! W# F( t8 k8 Y& ]2 `, U8 Z

    + a" v) r. D5 G(4).this.getServletContext().getResourceAsStream("/form1.html");给服务器用的: D7 _3 V% G2 R' d. {

    5 _3 q5 }- C. e# R(5).<ahref="/day05/form1.html">ddd</a>给浏览器用的
    , e) @, \) x( E: U : y# Q+ B( y* s6 ^4 o
    (6).<formaction="/day05/form1.html"></form>给浏览器用的
    $ i* d9 `" a) K4 W6 f  W
    + s8 T3 G3 F. ?4 z! M总结:写地址以斜杠开头,如果地址是给浏览器用的,这个斜杠代表一个网站,如果是给服务器用的,这个斜杠代表web应用.浏览器向服务器发送请求,就是给浏览器用的
    , x( S/ ^' r9 @6 |; T
    ' B# p+ L' y2 J8 K$ l2 D1 Z* n12.   防盗链
    7 u1 y# k( i' K+ i, M首先检查你是否从我指定的网页来的,没有的话,String referer =request.getHeader("referer");直接在地址栏上粘贴地址,还有就是refer.startsWith("http://localhost");检查是否从首页跳转过来的.首页上放广告,再有凤姐日记的链接.网站就是靠广告运行的.
    8 W: u( r0 Z: u& e& {" @
    1 J6 @$ a) n. c13.输出随机认证码图片. b9 d  A  F. U/ k  u- |* _
    (1).BufferedImage:内存中的一幅图片.构造函数指定了图片的长和宽,像图片上写随机数,getGraphics()方法返回一个图形的对象,然后向该图像上写数据.在把内存中的这幅图片输出到浏览器,ImageIO图像的输入输出;write();方法,指定一个和浏览器相关的输出流.
    / S1 V- [' C/ y* t. u
    5 D! D1 o+ ]* n(2).告诉浏览器以图片的方式读取数据.
    0 N& |  a  c4 _6 U! N9 n, T . Z4 @/ z) m6 t$ @0 w* f. o5 c
    (3).<imgsrc="/day04/servlet/ResponseDemo3">;' N: {+ \: X+ R9 V$ g
    & p2 \/ u+ g0 }: Y
    (4).刷新后图片为改变,没有控制浏览器是否进行缓存,浏览器默认是进行缓存的,所以每次拿的都是缓存图片,没有改变,所以要控制浏览器不要缓存,% T: v5 |7 p* h, Y$ u. k
    response.setDateHeader("expries",-1);response.setHeader(Cache-Control","no-cache");response.setHeader("arma","no-cache");必须三个都要设置好,才起作用.刷新有两个作用:重新向服务器发出请求,还有就是把上一次的事情在干一次.(防止表单重复提交);
    / f+ p; t, @& Q' \& J
    + W6 @0 H: I: m5 ]! J$ |9 {(5).<imgsrc="/day05/servlet/ResponseDemo4" onclick="change(this)"alt="换一张"style="cusor:hand"><br/><scripttype="text/javascript">function change(img){img.src = img.src+"?"+new Date().getTime()}</script>如果后面不跟随机数,还是拿缓存了.
    - C# x8 T; F! \ 0 r. G2 ^* ?  @. h4 A; h% ]
    14. 用Expires头控制浏览器缓存
    7 \3 e: b0 a; ?$ p+ `3 y" z(1).控制浏览器缓存,在servlet向客户会送的数据不变,就需要进行缓存,不需要向服务器发送请求,只需要到硬盘的缓存存放文件夹中读取缓存
    ) }3 H5 `) B' [0 t# ^: d 9 v& p7 t0 U  s0 u
    (2).Internet选项->常规->点击设置->查看文件,浏览器把index.jsp缓存了,缓存时间是当前时间值:System.currentTime()+1000*3600;
    - N# C9 L( w( ?6 ?2 L  d : c6 Z6 h: @, \+ O+ d+ y5 _- [
    15. 用refresh控制浏览器定时刷新2 W2 b7 x& u# p9 c. ]/ k
    (1).刷新头:response.setHeade("refresh","3;url='/day04/index.jsp'");控制浏览器每隔3秒刷新一次,
    . Q8 T  |5 n' {) W2 a" s
    8 E7 P' `1 P# G1 \! @' Q2 H% f& [(2).假设是一个用于登陆的servlet:提交用户名和密码,需要到数据库查询是否登陆成功,response.getWriter("恭喜你,登陆成功,本浏览器将在3秒后,跳到首页,如果没有调,请点击<a href="  ">点击</a>");
    ( h  ?9 D2 @# b3 I& ^  `' H " I5 ?6 F% ^: G
    (3).this.getServletContext().getRequestDispatcher("/message.jsp").forword(request,response);
    * q7 K" C  w9 ?1 y2 d* M 0 ]9 M, o/ [3 W2 a
    (4).servlet怎么告诉jsp定时刷新.字符串写给jsp"<metahttp-equiv='refresh'conten='3;url=/day08/index.jsp'>"
    红红火火恍恍惚惚
    回复

    使用道具 举报

    admin    

    1244

    主题

    544

    听众

    1万

    金钱

    管理员

  • TA的每日心情

    2021-2-2 11:21
  • 签到天数: 36 天

    [LV.5]常住居民I

    管理员

    5#
    发表于 2015-06-02 12:43:43 |只看该作者
    第六天:
    . y! H+ L+ }2 D; b( `  J& O1.  Cookie的细节# a0 X; C+ ?/ l) B. c8 u6 ~
    一个Cookie只能标识一种信息,它至少含有一个标识该信息的名称(name)和设置值(value),一个web站点可以给以一个web浏览器发送多个Cookie,一个web浏览器也可以存储多个web站点提供的Cookie,浏览器一般只允许存放300个Cookie,每个站点最多存放20个Cookie,每个Cookie的大小限制为4KB,如果创建了一个cookie,并将他发送到浏览器,默认情况下它是浏览器之后即被删除,若希望浏览器将该cookie存储到磁盘上,则需要使用maxAge,并给出一个以秒为单位的时间,将最大失效设置为0,则是命令浏览器删除该cookie,注意,删除cookie时,path必须一致,否则不会删除.cookie不能太大,不然可能造成网络拥堵,也可以将cookie删除.当两个cookie的名称相同时,删除其中一个,就是删除两个.即这两个cookie必须相同,setPath();也要设置.可以到ie保存Cookie的文件中查看文件.javascript也可以删除cookie
    $ @/ l' q0 d& R$ t3 @% Z$ Z + U( e" O! k' s6 F! j% z$ H
    2.Cookie显示用户上次访问网站的时间+ F  H( v; d) Y( t, M4 K3 w4 z) ]
    (1).javax.servlet.http.Cookie类用于创建一个Cookie,response接口也定义了一个addCookie方法,它用于在其响应头中增加了一个相应的Set-Cookies方法,它用于获取客户端提交的Cookie,使用Cookie封装用户的数据.
    $ a8 W: a& b3 a/ T  U  U/ J( {' p
    , R" ]2 n; E# n* k(2).构造函数,Cookie(name,value);给Cookie一个名称name,用户下一次访问时,带着上一次访问的Cookiet,所以在Request.getCookies();返回一个Cookie数组,即用户的所有Cookie,Cookie的有效期默认是浏览器的进程过程,可以通过setMaxAge()方法设置有效期,案例:用户在一段时间内自动登录。方法setPath();用于设置访问哪个web资源时携带Cookie,方法setDomain(),设置访问哪个域名时携带Cookie.IE默认禁止这种第三方Cookie,你访问我的网站,我给你带回一个的网站的Cookie.
    ) H6 U1 S: s2 v% T: R / B1 U* ^! n1 }2 }9 y4 l
    (3).用户第一次访问时,没有携带Cookie,Cookie cookie[] =request.getCookies();cookie可能为空,% b; \& t+ F+ Q6 ?0 I/ D# X
    首先要判断一下是否为空,. K& D% m* Q0 X" \! d9 t  m" G
    for(inti=0;cookies!=null&&i<cookies.length;i++){if(cookies[i].getName().equals("lastAccessTime")
    2 ~& w+ n2 o6 }. F5 @- m{long cookieValue= Long.parseLong(cookies[i].getValues()}};! }$ @: ]6 l+ P, D1 j$ \
    新建一个
    * V2 S+ Y2 m+ D- CCookie:Cookiecookies = new Cookie("lastAccessTime",System.currentTimeMillis());% T) @, g' v5 R2 M) G6 @' D  ^/ J
    设置Cookie有效期:cookie.setMaxAge(,,,);
    ! X' Y6 H7 ]7 c设置有效路径(默认的是"/day04/servlet");cookie.setPath("/day04");7 |# T; F% w# r% {3 }

    : w* E0 L* @1 @& ]3.session的工作原理* [9 P7 k; w! e/ A* N
    (1).如果把Cookie禁止,则不能购买东西,Internet选项->隐私->高级
    7 f" o( K' r7 Y) J. p& v) _5 y
    . _5 z4 d8 V5 d(2).URL重写,你访问我的服务器,都是超链接点击过来的,不同用户访问首页时,就帮用户创建session得到session的id号,然后将id号放在超链接URL中携带过来,不是Cookie携带了,所以所有的URL要重写,
    # ]) b6 t$ |7 Q; n4 [2 a6 p& b  Irequest.getSession();Stringurl = response.encodeURL("/day03/servlet/SessionDemo")此方法自动将session的id号加入该地址中.<a href = '+url+'>点击</a>
    " d3 g) s$ H! E! F
    " X( M4 e2 J% [0 F* W7 s, R. g(3).如果没有禁止Cookie,浏览器带一个Cookie来了,就不会进行URL重写
    # \. c. ?0 m1 Q* o+ E! O; x
    : R  S3 P* T# H& G1 F9 \4.session的一些细节问题
    + H! L* O3 v) p+ j; b# D3 \(1).一个浏览器就占用一个session,开一个新的网页,不创建session,超链接弹出的新窗口也是共享一个session,同时开多个浏览器,创建多个session,这个问题也不是绝对的,不同的浏览器,功能不一样,将session的id号写到浏览器进程,+ G% r/ G8 |6 v5 T( P) N9 G

    8 a1 B  b. P3 \# J5.Session简介0 J, v5 {, X. c( M4 g; a
    (1).默认情况下,一个浏览器独占一个session对象$ _: l6 N, T) }& c* c
    ; M: o% h- [# @, ^+ u
    (2).HttpSessionsession = request.getSession();* [, X5 X5 [# B/ O" h/ f: E; @+ T% n
    session.setAttribute("name","value");
    & P: X: j5 ^$ i! A4 Usession.getAttribute("name");
    ) G, p1 f; N5 Y - P% i& ^' I5 I' x  d$ g* k1 G& U$ Z2 [
    (3).写好过滤器,就可以解决web应用中的乱码问题
    * c. f: ?0 i, ]( a& |
    . Q: u9 v3 C1 [(4).一个session只为一个会话服务.(只打开一个浏览器窗口)% V! n. }- h* M* U, N3 c9 V
    , i/ P( B' z  y5 W$ j
    (5).session的生命周期,第一次访问getSession()方法时,session就被创建,session是30分钟没有人用了(即使浏览器不关闭),服务器就将其删除.而不是结束会话服务.当然这个时间可以实现的.
    # |6 t8 }' l+ p/ O( `4 b4 s<session-config><session-timeout>10</session-timeout></session-config>单位是分钟,同时在代码中也可以控制,session.invalidate();request.getSession(true)方法是指:有session,不创建,没有session就创建,request.getSession(false)方法是指:只获取session,不创建session,显示购物车的时候,没必要创建session只需要获取session.0 h% t5 L2 @) u1 y' k' S5 a

    - M. i" L  b# V9 o+ R" S. [) g(6).request.getSession()方法怎么知道获取指定用户的session,即session的工作原理:servlet1中:session=request.getSession(),产生一个session的id号,以Cookie的形式回写给用户,下一次用户就带着session的id号来,所以该方法就知道获取指定用户的session,但是这个Cookie没有写有效期,会话结束后,这个id号就没有了,下一次访问,服务器又要创建session,买东西的过程中,关闭浏览器,下一次再接着买,上次买的东西,都没有了,所以解决这个问题,只需设置到这个Cookie的有效期即可.sessionid = session.getId();回写一个Cookie,名称为JSESSIONID,值为sessionid,覆盖默认的Cookie.
    1 F) Z% t$ j: v) \
    0 }2 b. I# N9 E+ k0 m6.防止表单的重复提交
    " N1 ~& F5 Z: A7 C(1).用javascript防止表单的重复提交,<script type="text/javascript">variscommitted = false;function dosubmit(){if(!iscommitted){iscommitted =true;return ture}else return false;}</script>) P: u" M6 e! N  w0 R" ^) |
    <formaciton="/day01/servlet/Demo1" method="post"onsubmit="return dosubmit()">客户端阻止,第一个问题是,用户可以查看页面的源代码,修改js代码,用户也可以点击刷新,把上次干的事在干一次,也可以点击后退按钮.但是在开发过程中还是用js,: P) S6 z$ w. }. _% j

    # W3 L6 @. W' O% b6 O" l, [) B6 ?(2).在服务器端防止表单提交,表单是由程序给出,为用户提交的表单提供一个随机数(表单号),服务器端,查看表单的id号,检查表单是否提交过了,如果提交了,删除该表单的id号,产生令牌的发生器class TokenProcessor{}为了保证令牌是独立的,一般讲其设置为单例模式,是一个对象创建的随机数重复性低,还是多个对象创建的随机数重复对象低,是后一种重复性低,为了得到随机数的长短一样,就要用数据的指纹(摘要),数据的指纹都是一样大的,是128位,类MessageDigest dis =MessageDigest.getInstatnce("md5");使用md5码,byte[] md5 = dis.digest(input);input是传过来的数据,用字节数组变成字符串返回,用到新的算法:base64编码,任何数据的base64码变成ASCII码,原理:将数据字节变成四个字节,0011 0010 1100 1101 0010 1001,这三个字节变成四个字节,每六位装到一个字节,少两位,就在前面补两个零,为:00001100 00101100 00110100 00101001,最小数为0,最大数为63,不会超过64,base64自己定义了一个码表,比如:0:a,1:b,3:c ........63: 所以任何数据都可以变成键盘上对应的字符,人们熟悉的字符,应用在数据的传输过程中,需要给数据的带一个结束符号和开始符号,所以把要传输的数据变成base64,开始符号和结束符号使用base64中没有的字符即可,BASE64Encoder encoder=newBASE64Encoder();return encoder.encoder(md5);这个方法可能在api中查找不到,
    . c- C$ J4 X1 B. N* J7 { 0 G% X# {# S' |# g, R, M3 }
    (3).在servlet中产生一个令牌String token 然后跳到jsp中,这是将token存到session域中,而不是request中,因为token是id号,以后还要使用,在jsp中<input type="hidden"name="token" value="${token}">在用户不知道的情况下将令牌带过去,
    2 j6 h, A% B3 d! k9 V, C
    1 J3 e* p- F3 d9 h* ?6 S(4).将已经提交过的表单号删除,struts中就是这样防止表单的重复提交的.
    # _1 R: u( @/ b " \5 L1 @, W$ Z; }+ D- v
    7.会话管理% M; X4 J5 t) V) {
    (1).会话可简单理解为:用户开一个浏览器,点击多个超链接,访问服务器多个web资源,然后关闭浏览器,整个过程称之为一个会话,- i4 w1 Z  Q% ]: h
    % e9 j  ^# `- N' a4 o# g1 ?# Q/ ^
    (2).会话过程中要解决的一些问题:每个用户与服务器进行交互的过程中,各自会有一些数据,程序想办法保存每个用户的数据,例如:用户点击超链接通过一个servlet购买一个商品,程序应该保存用户购买的商品,以便用户点结账servlet时,结账servlet可以得到用户商品,为用户结账。
    - z+ d' W9 E  Y; V: V ! G) G% K. \- ^$ `- G  ]
    (3).如果用户的数据存到request中,结账servlet是一个重新的浏览器请求,所以结账servlet与购买servlet是两次请求,不是共享request,所以结账servlet不能得到购买servlet的内容.ServletContext存在并发问题* k# U* m( D) I6 o/ A* u

    + j, \1 p* N: X( N; q& a3 L(4).一般使用Cookie和Session;Cookie是客户端技术,程序把每个用户的数据以cookie的形式写给用户各自的浏览器,当用户使用浏览器再去访问服务器中的web资源时,就会带着各自的数据去,这样,web资源处理的就是用户各自的数据,你找我买个的东西,买完了东西,你带回去(可能将东西cookie放在ie缓存,在本地硬盘中),当你来结账时,把东西在带来,即访问结账请求,浏览器拿缓存(cookie);* N5 R, E, n, k4 F4 J% z

    1 Y. w/ A" k8 a; `- B; v, T: m(5).Session是服务器端技术,利用这个技术,服务器在运行时可以为每个用户的浏览器创建一个其独享的Session对象,由于session为用户浏览器独享,所以用户在访问服务器的web资源时,可以把各自的数据放在各自的session中,当用户再去访问服务器中的其他web资源时,其他web资源再从用户各自的session中取出数据为用户服务.用户的数据存在服务器端.购买Servlet:session =request.getSession();Session.setAttribute(name,object);结账Servlet:Session=request.getSession();obj=Session.getAttribute(name);不会再为这个用户创建一个session,而去拿服务器中该用户的session
    9 c9 Q) k% [8 n6 E5 g/ }) C' w
    9 {- m1 M# ^% O( z6 `( v8.三个域对象的总结
    / \% l8 v' |( [( [% f4 |: G(1).request,session,servletContext三个域:request:程序显示完,数据没有用了,就用request(转发);session:程序显示完数据后,该数据还有用,就用session,servletContext:显示完了,除了该自己用,还要给别人用(聊天室)6 i, I3 ~. p5 {7 E  ~+ }
    $ `7 X" K& r, w& y4 e6 M0 I
    (2).md5码,数据指纹,保存用户名的密码,为了安全性,将密码的指纹保存,根据md5码反推,需要时间很长,也可以暴力破解,因为密码的长度是一定的,所以一般把数据加上一个随机数,还有一种用途是进行数据完整性的校验,
    回复

    使用道具 举报

    admin    

    1244

    主题

    544

    听众

    1万

    金钱

    管理员

  • TA的每日心情

    2021-2-2 11:21
  • 签到天数: 36 天

    [LV.5]常住居民I

    管理员

    6#
    发表于 2015-06-02 12:43:56 |只看该作者
    第七天:5 I/ c! }* X  K% y
    1.     jsp入门和jsp运行原理
    * e2 u7 B8 R. D. K" e$ h(1).<%Date date= new Date();out.write(date.toLocaleString();%>显示当前时间.1 o+ r/ f) @; C4 b" y4 p

    8 k3 X9 w' E+ b8 ?(2).jsp的运行原理:首先将jsp中翻译成servlet放在服务器的work目录下,调用servlet的service方法,在这个方法中将jsp中内容通过流out.write()写到浏览器中,jsp中的java代码将原封不动的运行,可以查看jsp的servlet代码,查看所有定义的对象,不仅有out对象,还有其他对象,JspWriter就相当于PrintWriter3 v7 j# s. {5 i

    1 \1 M4 L) T( T9 {7 }8 E2 I(3).最佳实践:一种技术可以用在多个地方,但是只有一个最佳的地方就是最佳实践.& C' Y: i2 d8 ~' ~1 h
    ! p6 J/ D  ]4 w9 k8 L5 t! E* d+ }0 `
    2.     jsp语法page指令详解
    % ~, H9 [$ w" S5 T(1).修改jsp的模板,进入到MyEclipse,搜索servlet.java,在这个目录下,点开jsp找到Jsp.vtl
    ' a- e- ]. L/ H 0 A1 S5 N; z* x3 @8 T' |
    (2).jsp指令是为jsp引擎,引擎就是将jsp翻译成servlet的程序,他们并不直接产生任何可见输出,而是告诉引擎如何处理jsp页面中的其余部分,在jsp2.0规范中共定义了三个指令:page指令,include指令,taglib指令& o. \" D9 b: Y" @7 J  ]& C0 c
    - x4 ~! u( x8 v) @) r" s
    (3).指令的语法:<%@ 指令属性名=“值”%>,不会产生任何输出,举例:<%@ pagecontentType="text/html;charset=gb2312"%>,如果一个指令有多个属性,这多个属性可以写在一个指令中,也可以分开写,例如:<%@ pagecontenType="text/html";charsetgb2312"%><%@ pageimport="java.util.Date"%>也可以写作<%@ pagecontentType="text/html;charset=gb2312" import="java.util.Date"%>' ?, Q! b' u5 J* ^
    9 E! T6 B+ j1 I8 U% |
    (4).page指令用于定义JSP页面的各种属性,无论page指令出现在JSP页面中的什么地方,它的作用都是整个JSP页面,为了保持程序的可读性和遵循良好的变成习惯,page指令最好是放在整个JSP页面的起始位置.
    : V" g8 P( T& w( k . _' C4 V! P- X1 T
    (5).JSP2.0规范中定义的page指令的完整语法:
    5 P/ e: g1 ]- l4 |- V9 t& w<%@ page7 m3 ^# K1 e; o# Q- V
    [language="java"]0 ?9 g7 ~3 p" i( T' m  A
    [extends="package.class"]7 k6 Z0 W8 |' t  j2 ?7 z
    [import="{package.class|package.*},..."]
    / u+ ^/ f- `/ Y. h: F! MJSP引擎自动导入下面的包:java.lang.*;javax.servlet.*;javax.servlet.jsp;javax.servlet.http;可以在一条page指令的import属性中引入多个类或包,其中的每个包或类之间使用逗号分隔:<%@ pageimport="java.util.Date,java.sql.*,java.io.*"%>
    * b5 a- V) h/ ~[session="true|false"]:session是否创建,如果为true,则翻译成servlet时,就创建一个session,否则不创建,设置true,后就可以在JSP中片段中使用session对象,否则不能使用,默认值为true,因为session的周期比较长,为了减轻服务器的负载
      m  r( M$ u( @  }" G  q- @2 y* O[buffer=none|8kb|sizeKb]:默认值为none,jsp的out直接写到response中,没有缓冲了3 `3 u7 ?# a; _% R, S
    [autoFlush="true|false"]:自动刷新,默认值为true
    / X) K( o+ a) \8 s[isThreadSafe="true|false"]:JSP是否是线程安全的,设置true后,将发现翻译后的servlet就继承了SingleThreade接口,一个servlet只为一次请求服务,不会出现线程安全问题
    8 b: |2 @- \" B+ y7 I: k[info="text"]:带信息3 P/ J2 I4 o! R7 d! T+ l" P9 `
    [errorPage="relative_url"]:指定JSP错误的处理页面,errorPage属性的设置值必须使用相对路径,如果以"/"开头,表示相对于当前web应用程序的更目录(注意不是站点根目录),否则,表示相对于当前页面,errorPage="/errors/error.jsp";也可以在web.xml文件中使用<error-page>元素为整个web应用程序设置错误处理页面,其中的<excepiton-type>子元素指定异常类的完全限定名,<location>元素指定以"/"开头的错误处理页面的路径.作用域不同一个是在一个页面,一个是全局的配置,<error-page><exception-type>java.lang.ArithmeticException></exception-type><location>/errors/error.jsp</location></error-page>;<error-page><error-code>404</error-code><location>/errors/404.jsp</location></error-page>;如果设置了某个JSP页面的errorPage属性,那么在web.xml文件中设置的错误处理将不对该页面起作用,全局性的优先级高,会覆盖页面的配置, ~- ?8 T/ R8 H9 R7 m: o7 `
    [isErrorPage="true|false"]:这个页面是否为错误处理页面,默认值为false,好处是:服务器在翻译成servlet时将异常封装成一个对象,九大隐式对象中包含一个Excepiton,但是这个对象并不是任何时候都有,只有设置这个参数后,才有,记住
    7 i6 [/ {* |, w+ u[contentType="mimeType[ ;charset=charachterSet]"|"text/html;charset=ISO-8859-1"]页面的类型
    / A4 p  n' m6 F0 m% l7 a[pageEncoding="charachterSet|ISO-8859-1"]:页面的编码8 M; n8 S6 a% {
    [isELIgnored="true|false"]:是否忽略EL表达式,EL表达式时JSP2.0的新语法,使用EL表达式时需要将其设置为false.默认都支持EL表达式
    & i. u: d8 e& `6 d2 |/ q+ Y5 u" z
    & c- v/ m" n; W6 j7 S* m; u(6).jsp乱码问题:Jsp程序存在有与Servlet程序完全相同的中文乱码问题,输出响应正文时出现的中文乱码问题,读取浏览器传递的参数信息时出现的中文乱码问题,Jsp引擎将JSP页面翻译成Servlet源文件时也可能导致中文乱码问题,jsp引擎将jsp源文件翻译成servlet源文件默认采用UTF-8编码,而jsp开发人员可以采用各种字符集编码来编写jsp源文件,因此,jsp引擎将jsp源文件翻译成servlet源文件时,需要进行字符编码的转换,如果jsp文件中没有说明他采用的字符集编码,jsp引擎将把它当做默认的ISO8859-1字符集编码处理,通过page指令的contentType属性说明JSP源文件的字符集编码,page指令的pageEncoding属性说明JSP源文件的字符集编码
      l( P8 h1 R5 h4 N& i * I: r: M; i$ l8 `8 e5 C" m
    (7).tomcat6已经不存在乱码问题了,在jsp中是"中国",将"中国"保存到硬盘中,默认的查本机的码表,web服务器,翻译jsp的时候就是用iso8859-1码表,所以翻译后的servlet中的"中国"将是乱码,所以要改变服务器的编码,所以就用pageEncoding="gb2312"设置服务器翻译的码表,翻译到servlet后"中国"是正常的,但是此时还要设置浏览器以哪个码表打开所以需要设置:contentType="text/html;charset=UTF-8",其实最后的一个设置不需要设置了,因为response是以UTF-8编码的1 g" h/ |* Y1 c+ ^  b. d
    1 p1 W; g" p; D9 T4 o9 D3 m; M
    3.     jsp中常用标签
    1 W3 m0 `4 O4 f4 _/ t(1).jsp标签也称之为Jsp Action(jsp动作)元素,它用于在jsp页面中提供业务逻辑功能,避免在jsp页面中直接编写java代码,造成jsp页面难以维护.& i1 q& K! f% a5 g; s  K
    6 x  b1 N* s6 f" q: F5 a
    (2).<jsp:include>,<jsp:forward>:<jsp:forwardpage="/index.jsp"></jsp:forward>应用场景:通常为网站配置首页,首页是个servlet,显示是jsp,配置首页的时候,servlet是无效的,但是jsp是有效的,所以可以使用这个标签从jsp跳转到servlet.<jsp:includepage="/1.jsp"></jsp:include>是动态包含,翻译成两个servlet,在运行时进行合并,静态包含是在源代码的基础上进行合并<jsp:param>标签是设置参数,<jsp:forward page="/servelet/ServletDemo"><jsp:paramname="username" value="xxxx"/></jsp:forward>,跳转到servlet时带数据过去,value可以是一个脚本表达式<%=x%>,x为变量.! C. J+ Z" c! W* y, F! l

    + q. [1 a% d3 t5 V8 y5 I1 x; }- b% q4.     div+css
    / z; |" h2 s" L2 Z! g0 K2 i+ U(1).盒子模型:每一块数据都用div套起来,可能控制div的边框,属性border,上边框:Border-top;下边框:Border-bottom;左边框:Border-left;右边框:Border-right,Padding-top,Padding-bottom,Padding-left,Padding-right;控制数据在div盒子中的位置,Margin-top,Margin-bottom,Margin-left,Margin-right:控制一个盒子在页面的位置,
    2 z* h- Z9 L& u8 l" V ! `' T) d0 ]& `$ a% [3 d# ~
    (2).div是个行级元素,默认的是div单独排在一行,有时想将两个div排成一行,这是就要用到定位技术:Float和Position两种技术
    " G! U2 H/ \6 D& K- o: ?1 H% e 4 J( @$ z' ?9 U) [3 J
    (3).<style>" r% @& x# J4 N" f
    body{margin:4px 3px 2px 1px;}2 t  R3 @: W/ F, I" q
       #father{5 M$ n  D/ S) _& W6 j% T# _2 Q* ?
    background-color:#FFFFEE;
    , X( J$ m* h  @# x9 `' ]width:100;: }7 ^( n8 y% u2 _! h! x$ t
    height:100px;
    ; h2 D3 i0 o) u2 E( m; @! Eborder:1px dashed green;//边框是绿色的虚线,
    7 Y0 W) i, `) n, U: {- `% i}
    ; h% q6 v7 l% q% R8 M6 [, F1 b#son1{float:left;}# {8 w2 X/ N7 D/ b9 |( a
    #son2{float:left;}5 n  H) U7 [. r4 C
    #son3{float:left;}
    ) f1 d: z) B. P, N: \% }3 d</style>+ R( T) [, o0 C1 _$ @2 f, f
    <body>8 w7 a3 n2 E9 k
    <div id="father">( U; `* u, [: w) b( N  D
    <divid="son1">aaaaaaa></div>% a" p7 x5 @3 V, c: I5 H4 u
    <divid="son2">bbbbbb</div>
    / m7 i) Z0 t/ L2 b8 D<div id="son3">ccccc</div>5 g/ o* P# H# g# m
    </div>2 I& t/ b0 |- n
    </body>8 T/ o. E& {1 k2 |7 G% G
    这个代码是有问题的,当添加<div id="son4">ddddd</div>时,也会浮动,受到前面几个div的影响,所以要清楚浮动效果<divid="clear"></div>,#clear{clear:both;}* j# f3 S; a9 C1 ^: M; b

    2 R) }& x8 \$ }' u- J! I(4).position技术就是将#son1{position:relative;left:60%}
    % _) b2 n) i' L
    4 @& S* d* B$ G4 I2 z5.jsp九大隐式对象! x* p: s$ z4 ~% ], X
    (1).每个jsp页面在第一次被访问时,web容器都会把请求交给JSP引擎(就是一个Java程序)去处理,JSP引擎先将JSP翻译成一个_jspServlet(实质上也是一个servlet),然后按照servlet的调用方式进行调用,由于JSP第一次访问时会翻译成servlet,所以第一次访问通常会比较慢,但是第二次访问,JSP引擎如果发现JSP没有变化,就不在翻译,而是直接调用,所以程序的执行效率不会受到影响,JSP引擎在调用JSP对应的_jspServlet时,会传递或创建9个与web开发相关的对象供_jspServlet使用,JSP技术的设计者为便于开发人员在编写JSP页面时获得这些web对象的引用,特意定义了9个相应的变量,开发人员在JSP页面中通过这些变量就可以快速获得这9大对象的引用,
    + @$ M1 I8 ?+ b% Q
      F2 t7 Z2 w+ `# r! e2 s' w1 @(2).HttpServletRequestrequest,HttpServletResponse response,ServletContextapplication,ServletConfig config JspWriter out,Exception,PageContextpageContext,Object page,Session session,具体的可以查看文档
    3 c. s8 b% ^$ f( g- ^5 E2 v6 A& y . @8 D- j* I9 _3 \, b' @: B3 x
    6.jsp映射和jsp常见错误处理. `5 ^/ ^& h" Y2 k5 `9 O; G6 c
    在web.xml中
    $ r0 m( U$ e, N" n, m<servlet><servlet-name>xxx</servlet-name><jsp-file>/1.jsp</jsp-file></servlet><servlet-mapping><servlet-name>xxx</servlet-name><url-pattern>/12.html</url-pattern>
    * ]* T5 t0 b" X( H3 y6 Y% o' l% N   p3 e4 u8 T# L) q; Z1 s
    7.jsp语法8 d! `0 f! L$ n2 m6 k; y. [6 S8 [; Y
    (1).jsp语法:jsp模板元素,jsp表达式,jsp脚本片段,jsp注释,jsp指令,jsp标签,jsp内置对象,如何查找jsp页面中的错误0 e" B# a( x( i, E+ |4 U+ z

    ( M2 B5 F+ q* }(2).jsp脚本表达式(expression)用于将程序数据输出到客户端:语法:<%=变量或表达式%>举例:当前时间:<%=new java.util.Date()%>;jsp引擎在翻译脚本表达式时,会将程序数据转成字符串,然后再相应位置用out.print(...)将数据输出给客户端,jsp脚本表达式中的变量或表达式后面不能有分号(;)/ [8 w- A' i3 Z$ E' t7 A
    & i, O0 Y5 A0 q( }
    (3).jsp脚本片段<% %>,中写入java代码,原封不动的变成java代码+ u8 `8 K- Z5 W. X2 G! R( p
    , _# y$ n' F7 V4 k/ b
    (4).变量可以再多个脚本片段中是可以相互访问的.
    ! A( d, U0 i# Q5 S
    ) w" e7 E8 ?, }  h$ u) i( _(5).jsp声明:jsp页面中编写的所有代码,默认会翻译到servlet的service方法中,而jsp声明中的java代码被翻译到_jspService方法的外面,语法:<%! java代码%>(多个感叹号),所以,jsp声明可用于定义jsp页面转换成的Servlet程序的静态代码块、成员变量和方法,多个静态代码块、变量和函数可以定义在一个jsp声明中,也可以分别单独定义在多个jsp声明中,jsp隐式对象的作用范围仅限于Servlet的_jspService方法,所以在jsp声明中不能使用这些隐式对象,<% public void run(){} %>这种语法是错误的,<%! public void run(){}%>这种语法是正确的.- N, `3 y0 i) o( V
    / \$ i# f) S& z8 b
    (6).注释:<%-- --%> jsp中的注释不会打给浏览器,但是html注释会打给浏览器.$ {! b& m% |9 t  j& T

    " t: V: A) i* r5 N8. out隐式对象
    9 ~1 y( G9 w- S2 n" ~3 z2 F. ~(1).out隐式对象用于向客户端发送文本数据,out对象是通过调用pageContext对象的getOut方法返回的,其作用和用法与ServletResponse.getWriter()方法返回的PrintWriter对象非常相似,JSP页面中的OUT隐式对象的类型为JspWriter,JspWriter相当于一种带有缓存功能的PrintWriter,设置JSP页面的page指令的buffer属性可以调整它的缓存大小,甚至关闭它的缓存,只有向out对象中写入了内容,且满足如下任何一个条件时,out对象才去调用servletresponse.getWriter方法,并通过该方法返回的PrintWriter对象将out对象的缓冲区中的内容真正写入到servlet引擎提供的缓冲区中:设置page指令的buffer属性关闭了out对象的缓存功能,out对象的缓冲区已经满了,整个JSP页面结束.
    3 L& `3 l; d) `7 l+ R % `- p8 }2 W1 q8 Q
    (2).out.write("hahahahhahha");response.getWriter().write("wowowowo");理论上是先看到hahahah在看到wowowowowwo,但是实际是wowowowowowo hahahahhaha,原因就是out有缓存,JspWriter向缓冲区中写入内容,respose的缓冲中是wowowwowowwo,当jsp结束后,response缓冲区发现JspWriter的缓冲区中有内容,就将其刷新到response,所以先看到wowowowowwo,然后就是hahahahahhaha,就是response写入的内容最先显示,所以在开发过程中不要两种方式都使用,一般使用out.write();
    , A- W' \2 S) {" Y
    6 w; b* r; o) c2 M! e9.pageContext对象
    4 w4 P, _* J) c0 R2 r, d+ ?) C(1).pageContext对象是JSP技术中最重要的一个对象,它代表JSP页面的运行环境,这个对象不仅封装了对其他8大隐式对象的应用,它自身还是一个域用来保存数据,并且,这个对象还封装了web开发中经常涉及到的一些常用操作,例如引入和跳转其他资源,检索其他对象中的属性等
    # ^  B. F/ {" T9 a1 C8 r# k
    0 v$ n1 s$ n( i3 ~( w! |9 G0 y(2).它的方法:getException()返回exception隐式对象,getPage()方法返回page隐式对象,getRequest()返回request隐式对象,getResponse()返回response隐式对象,getServletConfig返回config隐式对象,getServletContext返回application隐式对象,getSession()方法返回session隐式对象,getOut()方法返回out隐式对象,pageContext封装了其他8中内置对象,用途是在自定义标签中,jsp中最好不要出现java代码,有时想显示数据,又不能有java代码,String data -(String)request.getAttribut("data");out.write(data);在页面中写入一个标签<flx:viesData/>传递一个pageContext对象就相当于传递8大隐式对象.
    # m. `- H* H3 w$ T2 B3 y
    6 r! ~/ R" }$ w' o0 b(3).pageContext也是一个域对象,pageContext.setAttribut("data","aaa");域的生命周期是整个页面的周期,作用域是整个页面,四个域中,作用范围最小的一个,request就是一个请求范围内,session就是在一个会话范围内,servletContext整个web应用
    5 z+ \7 }. Z; _6 y) X* g9 ~# B 2 z5 b" _4 `( E# j
    (4).代表各个域的常量:0 r. ]% M9 k9 V& U; O# s9 Z# z/ _
    PageContext.APPLICATION_SCOPE,PageContext.SESSION_SCOPE,PageContext.REQUEST_SCOPE,PageContext.PAGE_SCOPE
    5 Q0 i$ E9 m$ B3 x9 f: ^% p1 \ 3 T8 {2 S1 j3 |( P, G) R
    (5).封装了访问其他域的方法,方法public java.lang.Object getAttribute(java.lang.Stringname,int scope)是个重载的方法:Stringdata=(String)pageContext.getAttribut("data",PageContext.REQUEST_SCOPE);管理所有域的入口
    % I  O* o) G: M7 g ' l; x' U" Z3 v& z
    (6).最重要的一个方法:findAttribute:查找各个域中的属性,pageContext.findAttribut("data")首先从page,request,session,application逐级查找,找不到返回空.用于el表达式,el的执行就是按照这个方法的原理执行的.
    : b( }8 d& e9 L1 A+ ? ' @) L8 |8 u) B4 L+ L
    (7).引入和跳转到其他资源,PageContext类中定义了一个forward方法和两个include方法来分别简化和代替RequestDispatcher.forward方法和include方法,方法接受的资源如果以"/"开头,就代表当前的web应用; |+ I2 d& E. X0 i  ~, I. S3 P
    3 {  O& p. J. w" N8 k( R: Q
    10.jsp语法include指令详解5 ^2 N8 ?( U- G+ e" v: s
    (1).include指令用于引入其他JSP页面,如果使用include指令引入了其他JSP页面,那么JSP引擎将把这两个JSP翻译成一个Servlet,所以include指令引入通常称之为静态引入
    - u# E& v! w, Q! ` 1 U% R& b6 h7 G' s* Z
    (2).语法:<%@ include file="relativeURL"%>其中的file属性用于指定被引入文件的路径,路径以"/"开头,表示代表当前web应用,细节:被引入的文件必须遵循JSP语法,被引入的文件可以使用任意的扩展名,即使其扩展名为html,JSP引擎也会按照处理jsp页面的方式处理它里面的内容,为了见明知意,JSP规范建议使用jspf(JSP fragments)作为静态引入文件的扩展名,由于使用include指令将会涉及到2JSP页面,并会把2个JSP翻译成一个servlet,所以这2个JSP页面的指令不能冲突(除了pageEncoding和导入包除外);3 i% u% Q7 u* q4 ~
    3 d8 r1 h7 U% c) \/ @; ^
    (3).查看源文件,不是格式良好的html页面,就将被包含的页面的头删除
    / h5 G- h7 O6 `7 P3 N
    0 W; O) V) v7 x2 b9 i9 a1 W(4).动态包含:request.getRequestDispatcher("/public/head.jsp").inlude(request,response);它会把三个JSP翻译成三个servlet,静态包含只翻译一个Servlet,
    - k, j( `. V) W% K3 {; s 0 Y3 q: K1 v3 I
    (5).在开发过程中两种包含都用到:静态包含编译时包含,性能高,动态包含是运行时包含
    红红火火恍恍惚惚
    回复

    使用道具 举报

    admin    

    1244

    主题

    544

    听众

    1万

    金钱

    管理员

  • TA的每日心情

    2021-2-2 11:21
  • 签到天数: 36 天

    [LV.5]常住居民I

    管理员

    7#
    发表于 2015-06-02 12:44:08 |只看该作者
    第八天:0 i& q$ ?3 h- J0 p$ e
    1.  sun公司的jstl标签库, ^0 H4 O# f9 j' h9 f5 X9 z* V7 [, ~
    (1).JSTL标签库:核心标签,国际化标签,数据库标签,XML标签,JSTL函数(EL函数)
    $ f% M/ j6 g7 @' m ) @( ^4 H/ I3 H
    (2).sun公司的所有标签都放在standard.jar架包中,6 s; G: ]7 M' N3 M( q- W3 P2 c

    1 ^2 o7 W$ U1 [! I( W(3).<cut>标签用于输出一段文本内容到pageContext对象当前保存的"out"对象中,  b4 Q, A' z' K, h
    ; x/ ^* U; ^3 y7 Q
    (4).<c:set>标签用于把某一个对象存在指定的域范围内,或者设置web域中的java.util.Map类型的属性对象或JavaBean类型的属性对象的属性.
    % \$ A6 f6 [& F7 B6 j
    3 ~, |7 j; |( E3 U(5).<c:remove>标签用于删除各种web域中的属性
    $ v& v; b8 Z$ [, |7 c2 W: D
    ! [. M+ x2 Q5 Q# C' _2 }(6).<c:catch>标签用于捕获嵌套在标签体中的内容抛出异常4 N5 \3 `6 Z+ t( A8 V+ t4 {; P' }

    + H) b  R- [$ _. W$ k(7).<c:if>,<c:foreach>标签" ]( H& t( Q+ {. O9 y( Q2 k
    9 X" f" Z. N$ D+ ~0 x$ J
    2.标签案例-打包自己的标签库
    * m  b1 y/ n# H+ }" a5 C3 E(1).首先建一个java工程,将所有的标签处理器类考到这个java工程中,然后将tld文件存放到meta-inf文件夹中,然后将java工程导出为jar4 `% Z- X  G3 J' C2 e

    4 b  S' o% P1 Z5 f, V3.标签案例-开发foreach" d1 \( y7 F1 q: l4 _+ v
    (1).List list =new ArrayList();然后迭代输出list中的数据
    7 S% p" Z. z7 g: F
    & O) H' l( K1 Z5 J3 i; g3 U1 x8 z4 G5 L(2).<c:foreachvar="str" items="$(list)">$(str),var属性记录list中数据的类型.var的类型为String
    0 h4 E$ c6 B8 ~% f
    4 D$ A/ y  p! M" m0 X7 n" E# T(3).在doTag()方法中,List list = (List)items;Iteratro it =list.iterator();while(it.hasNext()){Object value =it.next();this.getJspContext().setAttribute(var,value);this.getJspBody().invoke(null);标签中的内容执行回到pagecontext的域中拿去属性var的值,然后输出}
    - I# W- k6 {/ @- L1 D 5 n* X; a0 s7 p$ m
    (4).将foreach功能加强可以迭代数组,Map,Collection判断是不是Map,还是Collection,还是Object[],但是要进行类型转换,可能有很多重复代码,所以都直转型Collection collection即可,if(items instanceof Collection){collection =(Collection)items;}if(itemsinstanceof Map){Map map (Map) items;collection = map.entrySet();} if(itemsinstanceof Object[]){Object obj[] =(Object[])items;CollectionArrays.asList(obj);}当碰到基本数据类型时,就需要迭代数组,将数组中的数据放到collection中,; [! P& q8 `" H  J5 z  o/ G

    , @/ B! M! L- z# ^(5).sun公司的foreach标签在standard架包中,在web-inf,c.tld,查找forEach标签对应的类
    8 P6 g; G0 }8 m8 }' u
    " G) P% N( f) j(6).反射技术,Class类中有一个方法isArray()判断是否是数组类型,不仅适用于任何类型,在反射类中有一个类Array,它有个方法
    & a$ F/ f* ?% I+ q/ }% W: ?0 pgetLength(),for(inti=0;i<length;i++){Object value =Array.get(items,i);this.collection.add(value);}
    , e+ c; o: B) }9 o9 I9 p对任意数组进行操作.
    : d; a0 X" }) Q" M9 W: H 0 Q4 c& F9 `9 g  j4 h% e% \
    4.标签案例-开发if-else标签
    ; T1 z7 N! A+ L- m& n(1).<c:iftest="$(user!=null)">aaa</c:if><c:else>bbb</c:else>这两个标签是共享一个变量,即让他们两个共享同一父标签<c:choose></c:choose>,共享的变量在父标签中定义,则此时需要写三个标签处理器类,
    $ P. L$ H/ W% ~8 N
    ! Q# s1 ]( X  C8 B(2).同时在父标签中要让其标签体执行,所以在父标签中的doTag()方法,this.getJspBody().invoke(null);
    5 g5 F. U+ w; j5 {2 j
    ) L# [8 s' n9 F* d# H; v(3).得到父标签的变量,所以在子标签的doTag()方法中Choose parent = (Choose)this.getParent();就可以得到父标签的处理器类5 H: {& E1 \- c3 Y7 T. b( u0 |: r
    / m# G4 Z, e& A/ L( O0 u4 p* [
    5.  标签案例-开发防盗链标签
    / ^3 ^% s  c6 B* y0 D就是将前面的request的课程中的方法定义为标签处理类即可
    3 c5 y- C9 _0 Q% s; W / w: P. T4 l9 L, I0 s$ k
    6.  标签案例-开发转义0 e$ q2 i' \0 L0 o
    (1).比如超链接的原始输出,9 Y& q+ d  P* r  [3 Q9 h- W
    StringWriter sw =new StringWriter();JspFragment jf = this.getJspBody();jf.invoke(sw);Stringcontent = sw.getBuffer().toString();服务器的webapps/examples/web-inf/htmlFilter,java中的方法filter();+ ]+ ]  V, q. f
    7 x# Y; ~- O3 U, h/ \' j* D
    7.  标签简介和开发第一个标签) w) N: g4 Q/ B0 Z: C4 q
    (1).自定义标签主要用于移除jsp页面中java代码,使用自定义标签移除jsp页面中的java代码,只需要完成以下两个步棸:编写一个实现Tag接口的Java类,把页面java代码移动这个java类中(标签处理类),编写标签库描述符(tId)文件,在tId文件中对标签处理器类描述成一个标签,
    2 g; F  Q% N* u) d8 T: ^! Z
    ; E% w6 q5 w, ^3 u% b' I. \/ x+ C(2).<%request.getRemoteAddr();out.print(ip);%>,显示用户的IP,使用自定义标签,首先将java代码移动到java类,继承Tag接口,标签有开始标签和结束标签,doEndTag()方法,doStartTag()方法,标签可能有爸爸,所以方法getParent()将父标签作为一个对象传给jsp引擎,方法setPageContext()方法最重要,就是传递一个PageContext对象,他同时也携带了其他8个对象,所以将java代码移动到doStartTag();Tag是个接口,所以使用其继承类,TagSupport,自己编写的类只需要覆盖doStartTag();准备好request,response,所以要得到PageContext,又因为服务器在执行doStartTag()方法之前,默认执行了setPageContext()方法了,所以可以写为:HttpServletRequest request =(HttpServletRequest)this.pageContext.getRequest();还有获取out对象,用于输出.移动完java代码后,第二步:就是在标签库中将标签处理器描述标签,可以到tomcat的webapps中自带的一些例子,web-inf/jsp2/taglib.tld.
    2 n) L4 k+ ], o( C. ] # y7 `) [) F+ B) V, S+ h
    (3).<tag><name>viewIP</name><tag-class>cn.itcast.web.tag.ViewIPTag</tag-class><body-content>empty</body-content></tag>.<url><http://www.itcast.cn</url>0 C4 b6 W0 o: `6 S, q% e
    , t% Y6 X. u/ h
    (4).导入标签<@taglib url="http://www.itcast.cin"prefix="itcast"%>:prefix是前缀,可以容易定位到描述标签的文件. u/ Y& g1 j1 |0 P' ]( r

    8 i, y( e+ `; T1 t' c(5).<itcast:viewIP/>即可itcast就是前缀prefix,viewIP就是tag的名称name
    8 o* a7 p2 f2 R3 c2 O
    0 Z8 `8 o! ^* U% Z(6).tld文件的位置必须放在web-inf中.
    $ ?( n! s- q0 t# ]$ l  @1 f
    ; I- z0 \) @7 F4 i  Y4 o7 L% U/ r8.  传统标签库功能详解
    0 k6 {5 x5 Z, B( O(1).控制标签体内容是否输出:覆写父类的doStartTag()方法,public int doStartTag(){returnTag.EVAL_BODY_INCLUDE}执行标签体,然后再添加tld文件,<body-content>JSP</body-content>,如果不执行只需要return Tag.SKIP_BODY即可- Y; H2 d* @3 W+ i: g

    0 F+ c0 N/ g/ h) x(2).控制整个JSP是否输出,就在JSP开始头定义一个标签,doEndTag()的返回值是否继续执行余下的JSP内容,只需覆盖父类的doEndTag()方法,public int doEndTag(){return Tag.SKIP_PAGE}不执行JSP内容,然后添加tld文件,一定要将该定义的标签放在JSP开始的部分.返回Tag.EVAL_PAGE就可以执行余下的JSP内容.
    7 ^; T6 J: w6 `0 K0 f. v* n( Z 1 _6 [2 f4 D& V$ |/ J
    (3).控制某一部分内容重复执行,只需将重复类容套在自定义的标签中,Tag接口找不到指定的方法,必须要使用其子接口IterationTag接口,其内部中有个方法:doAfterBody(),这个方法在标签体结束后执行,如果返回EVAL_BODY_AGAGIN,就接着执行,直到返回SKIP_BODY,所以继承IterationTag接口,但是TagSupport继承IterationTag,所以还是继承TagSupport即可,首先覆写方法doStartTag()方法,返回Tag.EVAL_BODY_INCLUDE;然后覆写方法doAfterBody(){x--;if(x>0)return IterationTag.EVALBODY_AGAIN;elsereturn IterationTag.SKIP_BODY;}其中x应该定义为成员变量,而不是在doAfterBody的局部变量,因为doAfterBody()方法是多次执行的,不是只执行一次.
    7 m/ \: k: Z1 F- f
    5 B: R4 V) S3 D9 e4 R(4).修改jsp页面的内容,将显示的内容修改成大写的,将内容套在自定义标签中,接口BodyTag接口,此接口继承了IterationTag接口,覆写BodyTagSupport方法doStartTag()返回EVAL_BODY_BUFFERED,会将标签体最为一个对象通过方法setBodyContent传给标签处理器类,然后在doEndTag()方法中BodyCotned bc = this.getBodyContent();得到标签体,String content = bc.getString();content=content.toUpperCase;然后通过pageContext输出,最后要记得返回EVAL_PAGE,执行余下的jsp内容,就拿到标签体中的内容,然后对内容进行操作.5 W2 J& ~4 ^7 d4 S+ u

    $ l/ ?- ^. h, F; X( U: X! H. E& C/ U(5).IterationTag接口的子类为TagSupport类,TagSupport类的子类为BodyTagSupport类,同时BodyTagSupport类是实现BodyTag接口,同时BodyTag接口继承了IterationTag接口,IterationTag接口继承了Tag接口,同时Tag接口和SimpleTag接口都继承JspTag接口,但是Tag接口时JSP2.0以前定义的,现在不用了,而现在主要用的就是SimpleTag接口,其子类为SimpleTagSupport类.所以称Tag接口为传统标签,SimpleTagSupport为简单标签
    8 }2 w/ K7 N% ~  Z& k 6 W' D  s' ~/ v3 e+ W
    9.  简单标签库功能详解
    + R7 |# b2 N+ H8 j(1).SimpleTag接口集成了以前的Tag,IterationTag,BodyTag三个传统标签的功能,所以SimpleTag有个默认实现类SimpleTagSupport2 _+ D# h1 _1 Y" x+ l

    / `2 F0 k6 D1 q' O: e(2).JspContext就是PageContext,方法,setJspBody(JspFragment jspBody),服务器通过这个方法将PageContext对象传递给标签处理器,然后调用doTag()方法,没有开始和结束标签,doTag()方法中抛出个异常,返回SKIP_PAGE控制整个页面不输出,
    ( |! y% o( [3 O 3 S1 X8 i4 k' {& p
    (3).简单标签的一般逻辑都在doTag()方法中,JspFragment jf = this.getJspBody();jf.invoke(this.getJspContext().getOut());简单标签中写成<body-content>scriptless</body-content>,JSP2.0规定了jsp中不能写脚本代码了,所以不死JSP,而是scriptless,jf是得到了一个标签体,jf.invoke()标签体运行,在this.getJspContext().getOut()的out对象写入浏览器
    * U2 i. {! Y* b! k- Z: W, M4 D" L
      m% F: k9 h) w7 _% J6 S9 a(4).如果想让标签体不执行,jf.invoke()方法不执行就可以了
    3 a  L- v3 S8 q* o* L2 L* v : j' \( L) e: s% S0 f+ [! _" s( C
    (5).jf就是标签体,可以写成:for(int i=0;i<5;i++)jf.invoke(null);这样写也行,就是默认的也是写给浏览器,重复将标签体- J6 R9 m- A9 w; I* d% |: y
    , ^/ s' @# z# G1 Z% \4 O/ B
    (6).StringWritersw = new StringWriter();jf.invoke(sw);将jf的标签体的内容输入到自己的缓冲,可以自己定义一个缓冲流,String content = sw.toString();content =content.toUpperCase();+ S) W& {# j: d7 k2 U; g1 M

    5 m8 b3 ^; z7 _* n. y(7).控制整个标签是否执行,只需在doTag()方法中抛出异常,throw new SkipPageException();即可.同时这个标签也要放在jsp页面的开始处,jsp余下的内容就不执行了" f" r8 e0 [# h* `3 e
    4 b8 k% i. b  t* P0 O$ F
    (8).jsp运行时遇到简单标签,实例化标签处理器类,然后通过调用对象setJspContext()方法,将PageContext对象传递给标签处理器类,接着调用setParent()方法将父标签传递给标签处理器类,然后把标签体的内容封装JspFragment对象传递给标签处理器类,然后调用doTag()方法执行自定义标签,执行完后,标签就变成垃圾了,等待垃圾回收器回收,这一点和传统的自定义标签不同
    . }/ I. W. ?& s, B  | . [* K6 e6 B/ \( H
    10.开发带属性的标签
    ) [; q4 I! l1 ?1 @3 Z) a5 v(1).要想让一个自定义标签具有属性,通常需要完成两个任务:在标签处理器中编写每个属性对应的setter方法,在TLD文件中描述标签的属性,为自定义标签定义属性时,每个属性都必须按照JavaBean的属性命名方式,在标签处理器中定义属性名对应的setter方法,用来接收JSP页面中调用自定义标签时传递过来的属性值,例如属性url,在标签处理器类中就要定义相应的setUrl(String url)方法,在标签处理器中定义相应的set方法后,JSP引擎在解析执行开始标签前,也就是调用doStartTag方法前,会调用set属性方法为标签设置属性
    7 r# {  s0 ~: G& R3 d " V6 p( T- [4 e6 A5 X) N3 q" b
    (2).在tld文件中声明:
    9 e1 {: w3 K8 ^# M" I5 ~) {<body-content>scriptless</body-content>,<attribute><name>count</name><required>true</required><rtexprvalue>true</retexprvalue>这个设置为true说明不仅可以传递变量,也可以传递脚本表达式</body-content>
    # U( Z6 y9 r, ~  \7 @+ \ : Z$ `& H7 r$ @; v( x/ x
    (3).count="5"属性传递过去的是字符串,但是count是int型的,服务器自动转型,但是只支持8中基本类型
    9 W% @+ i8 g: Q& L/ M; m7 q ; h, ]8 L1 ^3 z8 \+ w6 o) [
    11.自定义标签功能概述4 n2 i$ _* u7 V& e' j
    (1).自定义标签功能扩展,开发人员在编写JSP页面时,经常还需要在页面中引入一些逻辑例如:控制jsp页面某一部分内容是否执行,控制整个jsp页面是否执行,控制jsp页面内容重复执行,修改jsp页面内容输出,, |. c* O$ E4 ?; N

    * F; G$ O* f- d  q9 G# P(2).希望有段jsp代码只有权限的人才能访问,就是控制某一部分内容是否执行,格式化页面输出的内容.
    3 [  a. V; {  o5 y, I
    ( ^# I2 _9 V& q' [0 f* r12.自定义标签运行原理3 N5 k! I! Y2 _1 o. u! N
    (1).ie->web服务器->1.jsp(1.servlet)[<itcast:viewIP/>]首先实例化标签处理器类,然后调用自定义的继承Tag接口的类的方法,调用setPageContext方法,把页面的pageContext对象传递给标签处理器类,如果有父类标签,就把标签对象通过setParent方法传递给标签处理器类,完成以上初始化工作后,就开始执行开始标签,doStartTag()方法,如果标签有标签体,服务器一般会执行标签体,然后执行结束标签.然后标签调用release()方法,释放标签占用的资源,一般是web应用停止时调用,服务器接着执行jsp余下的内容,web服务器会将web-inf中的tld文件加载到内存中,
    8 ]9 f/ U9 v2 S" g' C- f/ G
    - \0 C+ D0 w# W(2).jsp翻译成Servlet中可以看到service方法,调用自定标签的动作放在方法中,首先获取pageContext,将标签处理器类的对象创建出来,
    回复

    使用道具 举报

    admin    

    1244

    主题

    544

    听众

    1万

    金钱

    管理员

  • TA的每日心情

    2021-2-2 11:21
  • 签到天数: 36 天

    [LV.5]常住居民I

    管理员

    8#
    发表于 2015-06-02 12:44:34 |只看该作者
    第九天:/ k$ k; l7 o" y7 o" W6 T
    1.     el表达式和jstl快速入门4 b) w- [3 f) W' a9 t
    (1).EL表达式用于获取数据,在jsp页面中可以使用${标示符}的形式,通知JSP引擎调用pageContext.findAttribute()方法,以标示符为关键字从各个域对象中获取对象,如果域对象中不存在标示符所对应的对象,则返回结果为""(注意不是null),EL表达式中也可以使用$(customerBean.address)的形式来访问JavaBean对象的属性,结合JSTL标签,EL表达式也可以轻松获取各种集合中的元素,EL表达式也可以使用类如${1==1}的新办公室进行简单的逻辑判断# u8 k0 f9 Q1 p/ _% m
    & t1 e! m# U9 |8 a( ]
    (2).<% Person p= new Person();p.setName("aaa");request.setAttribute("person",p);%>  ${person.name}! }5 L% x" l  T
    $ g0 H# w$ o2 C* G$ U; t  ^
    (3).用el表达式在取数据时,通常用.号,若取不出来,就用[],${map['aaa'].name};
    2 c1 u$ Z* `7 `2 ~. \: W: N
    . ]9 X& M  F' r$ U2 j9 ^/ c3 k# ^(4).${pageContext.request.contextPath}获取当前web应用的路径,<a href=${pageContext.request.contextPath}>点击</a>
    ( t% C9 D" r/ a0 Q# I 5 R) n* R+ u; M( E, `
    (5).<c:forEachvar="entry" items="${map}">${entry.key}{entry.value.name} </c:forEach>
      l( x  E: f; a+ ]   Z2 r/ O" b+ d' a
    2.     jsp和javabean
    - R$ z6 c1 e6 X& P$ p( c(1).javabean是一个遵循特定写法的Java类,它通常具有如下特点:这个Java类必须具有一个无参的构造函数,属性必须私有化,私有化的属性必须通过public类型的方法暴露给其他程序,并且方法的命名也必须遵守一定的命名规范  l" B% z% K6 @/ x
    $ Y- l' B* @$ H/ k7 E# v( L
    (2).javabean在j2ee开发中,通常用于封装数据,对于遵循以上写法的javabean组件,其他程序可以通过反射技术实例化javabean对象,并且通过反射那些遵守命名规范的方法,从而获知javabean的属性,进而调用其属性保存数据1 s7 J0 R7 T- c% v: _1 s

    9 Y7 h. b" [/ M+ [(3).属性的修改器和访问器就是set,get;2 Y: k4 R! t9 [7 v
    ; B5 h3 w: o/ U7 z: m' J
    (4).JSP技术提供了三个关于JavaBean组件的动作元素,即JSP标签,他们分别是:<jsp:useBean>标签:用于在JSP页面中查找或实例化一个JavaBean组件,<jsp:setProperty>标签:用于在JSP页面中设置一个JavaBean组件的属性,<jsp:getProperty>标签:用于在JSP页面中获取一个JavaBean组件的属性
    ( f% ^+ s8 m/ h8 D+ T
    6 a4 B7 Y" h4 m# I* Y% b  m( o3 Z( N(5).<jsp:useBean>标签用于在指定的域范围内查找指定名称的JavaBean对象:如果存在则直接返回该JavaBean对象的引用,如果不存在则实例化一个新的JavaBean对象并将它以指定名称存储到指定的域范围中,常用语法:<jsp:useBean id="beanName"class="package.class"scope="page|request|session|application"/>;id属性用于指定JavaBean实例对象的引用名称和其存储在域范围中的名称,class属性用于指定JavaBean完整的类名(即必须带有包名)scope属性用于指定JavaBean实例对象所存储的域范围,其取值只能是page,request,session和application等四个值中的一个,其默认值是page,从page中查找,查到不到就创建一个javabean,查找到了就返回定义的javabean,反正最后肯定有一个javabean  c4 n4 d! T/ C5 \. l( L% a* Q& G

    ; k" G, f" I  O5 v6 x& n/ h& b& p(6).<jsp:useBean....>body</jsp:useBean>标签体只会在javabean实例化时执行.! `- D2 C3 K; A2 F; l* a3 c, r
    : t" C/ z  c) B+ N, X
    3.     mvc开发模式0 s' [( v- [5 I+ i5 s4 q6 q
    (1).按照三层结构开发流程:servlet,jsp是web层,service,javabean是业务逻辑层(service层),dao是数据访问层(dao层),通常在层与层之间定义接口,所以在dao层和service层之间定义dao接口,同理在web层和service层之间定义一个service接口,dao层使用jdbc,hibernate编写,当dao层修改了,service的代码不需要修改,因为在service中使用的的是dao接口
    ; q) `/ w) C0 [: V ; Z4 }& \/ x) g6 Z$ p9 V
    (2).组织包接口:Cn.itcast.domain:存放javabean类,cn.itcast.dao存放dao层的类,cn.itcast.dao.impl:存放dao层接口,cn.itcast.service:存放service层类,cn.itcast.service.impl:存放service层接口,cn.itcast.web.controller:存放servlet的类,cn.itcast.web.listener:存放监听器类,cn.itcast.web.filter:存放过滤器类,cn.itcast.web.util:存放工具类,cn.itcast.juit.test:存放测试类,jsp页面放在web-inf下的jsp文件夹下.
    ; C- j8 m2 V$ s4 T$ }5 h6 G
    4 M% M9 X& v& n 7 r9 ?/ i. M; |

    6 W* [. [# H" H  p- t  z( o7 M二、       Struts19 R4 g# i6 }* ?
    1.     ActionForm的工作流程分析- E9 p2 E4 r6 I9 u; c( ?
    (1).ActionForm的工作原理:处理ActionForm的一般步骤:
    6 `; b" h. s& W; i( A* H第一步:检查Action的映射,确定Action中已经配置了对ActionForm的映射
    ! Q( }8 }; l0 U2 D  z8 w第二步:根据name属性,查找form-bean的配置信息- H( ]% u. q' X' N* Y1 f; t+ ]
    第三步:检查Action的form-bean的使用范围,确定在此范围下(request,session),是否已经有此form-bean的实例
    3 W( }; e$ o" {" i4 g第四步:假如当前范围下,已经存在了此form-bean的实例,而是对当前请求来说,是同一种类型的话,那么就重用
    8 {4 h* e( ~# C2 o2 ?( w9 i$ l+ R 8 ^" |+ F/ {8 d; G, J+ O+ A( V% D! f; O
    第五步:否则,就重新构建一个form-bean的实例(调用构造方法),并且保存在一定作用范围
    0 y' u0 ?/ {/ G* Q第六步:form-bean的reset()方法被调用8 p. b3 U, P/ t! Q
    第七步:调用对应setter方法,对状态属性赋值
      C* \4 t2 G  N2 ]( ~  T6 O第八步:如果validate的属性设置为true,那么就调用form-bean的validate()方法
    7 [) B! m! S% ~; W, H& T: g第九步:如果validate()方法没有返回任何错误,控制器将ActionForm作为参数,传给Action实例的execute()方法并执行
    3 n8 y" Y. L) [, u注意:直接从ActionForm类继承的reset()和validate()方法,并不能实现什么处理功能,所以要自己重新覆盖方法0 c9 W6 p; n: Z& U8 e+ Q- U5 `

    9 x/ B2 R. o# o% P0 y8 F2 P2.     ActionForm相关的attribute属性
      L8 t' Y" \6 X0 i(1).配置文件简介:使ActionServlet,ActionMapping,Action,ActionForm这几个不同层次的组件相互协调工作,这些配置文件是在系统启动的时候,读入到内存中,供控制器使用的7 p- i) _# F$ |  p8 n, P  B

    * W9 ~) [5 j1 l% n  H(2).<action-mappings>元素帮助进行框架内部的流程控制,可将请求URL映射到Action类,将Action对象与ActionForm对象相关联,<action-mappings>元素内可定义多个<action>子元素,<action>元素,所描述的是特定的请求路径和一个相应的Action类之间的映射关系,有以下属性:attribute:设置和Action关联的form bean在request/session内的属性key,通过request/session的getAttribute(attribute)方法返回该form bean实例,用来存取form的关键字,缺省值与name一样.而不是attribute的值.(ActionForm),input:当表单验证失败时将请求转发的URL1 P# T- h& Z, _# z) x8 ~: u

    . l$ m( u# C+ ^9 T(3).ActionForm相关的input属性
    $ B9 b/ q; ^. T" {1 X$ P3 [& ]input属性是用来记录路径:当validate校验不通过,即ActionForm的validate方法返回为ActionErrors对象,且该对象携带一些错误信息,就跳转到指定的错误页面(action)的路径,一般是结合validate=true是结合使用的,当为false时,input属性就没有意义了.  J! n7 T3 [: I/ Y6 v- a
    ActionForm获取用户的请求参数,其属性的名称必须要和请求参数名相同,必须要定义对应的get/set方法- I5 Y7 I$ u1 e* I

    0 M* f) `$ `& h( L: Q3.     ActionForm相关的validate属性
    $ Y& x+ V! _8 T9 o6 A' S  z* a(1).数据校验:判断用户带来的数据是否符合规定的格式.- h, P& j' H7 ^* D/ {

    4 ~5 B2 w6 w0 k(2).服务器端的校验:看ActionForm有没有调用validate(),可以返回ActionErrors对象,此对象返回一个错误封装对象,没有错误就返回null,缺省的情况是返回null,所以子类要覆盖这个方法.也可以通过在actionForm子类的配置中设置validate="false"的值,使该方法不调用- t, E9 k0 W' u
    复位:是总控制器恢复bean属性的默认值
    - M, t5 \8 n- W- x: L" r% M0 S- k
    : }  s6 [) p4 L5 I7 E1 U4.     ActionForward的有关问题( g, v( U$ H# C( H* y2 C# R/ D
    (1).ActionForward对象的配置对象,这些配置对象拥有独一无二的标识以允许他们按照name属性等来检索,ActionForward对象封装了向前进的URL路径且被请求处理器用于识别目标视图" c! C3 D% }$ ~$ V3 i8 O/ J
    & h# ~& x4 w" [* g  G  P) d& |$ ~
    (2).其内部的方法就是两种方法:request.Dispatch().forward();和response.sendRedirect(),前面讲到的重定向和转发.8 Q% y$ h, b# x( B2 b8 h( b
    ActionForward:Redirect是false,就是容器内跳转,就是Request.Dispathcer.forward();在这种情况下path为相对路径是true,就是容器外跳转,就是Response.sendRedirect(),在这种情况下path为绝对路径,要在域名前加上协议头错误!超链接引用无效。- f2 U0 f& m8 ~% D
    0 [8 h! z- @7 l( I" v! \, z
    6.ActionMapping的深入研究
    8 ]) U3 q1 \5 P* D' k# s+ d(1).ActionMapping:每个<action>元素都与类ActionMapping的一个实例对应,代表着请求信息,该类就是对<action>元素中的信息进行封装保存.该类含有这些信息的访问器和修改器.
    4 b6 D1 k! b1 R  P" t9 D& l 3 G- z0 R) E7 s6 Z
    (2).mapping.getName();mapping.getPath();mapping.getType();mapping.findForwards();返回局部跳转的页面信息,即就在一个action内部.9 X  x0 G7 @' E  M+ D: k  o; _
    4 l4 k, W: F0 d- h4 s
    7.Action的深入研究和分析0 a% j0 T" r$ ?2 P6 g- W
    (1).怎么测试Action实例化了,根据构造函数即可,public Action(){System.out.println("Action isrunning!");}Action在发出请求时初始化,不是在读取配置时初始化,每个action只会初始化一次,即使在不同的会话中(打开两个浏览器).内存中只有一份,资源可以共享和重用,但是不安全.对以多个请求只创建一个action,含有并发问题.需要进行同步.struts2.x是安全的,1 w; a! ]  U) a7 @+ ]$ \8 ^; Q
    : e2 q) q" Q( X  d. U
    (2).安全的话注意两点:第一、不要用类变量或实例变量共享只是针对某个请求的数据,第二、注意资源操作的同步性4 ?( H0 G- i8 k* B! Q0 ?

    2 T# ]6 Y; z5 f" Q0 A(3).一个action被调用多少次?这是就可以使用action的不安全性,定义一个变量count即可  N" n  R. N5 V" A3 h/ i9 M' Z

    4 u- S; G. [7 N( t8.bean-message标签的讲解+ P# U# F) B9 z$ n
    bean-message:国际化信息,key是信息的关键字,其对应value是资源文件2 y2 a" T, a( D# V
    bean-message的使用:5 \5 q! a2 l5 \: H: \4 _. r
    第一步:定义资源文件,
    . ?; W1 B: L1 Y, Z# H  Q4 m  u- ^com.itcast.ApplicationResources.properties;com.itcast.ApplicationResources_zh-cn;ApplicationResoruces是基名,后面可以跟不同国家的名称; ?* e* ^  ~  _
    第二步:在struts-config中添加:<message-resourcesparameter="com.itcast.ApplicationResources"key="myKey"/>
    # j, X8 N- H& a& F7 u例如:<message-resourcesparameter"cn.itcast.ApplicationResource" key="myKey">写的是基名,在资源文件中定义greeting=welcome username=root password=root键值对key属性是防止所有的properties文件混淆
    + B! A8 M5 }8 ]( p0 W$ I( f第三步:在页面中使用:bean:message test<br> <bean:message bundl
    0 \/ w0 B( L3 n: M" g' t6 @e="myKey"key="userName"/> <bean:message bundle="myKey"key="password"/>
    8 ]$ y/ c, m* b: k例如:
    9 E  i1 Y6 u* N' q# |1 l% p<tr>& [6 \4 Z: A0 k
    <td colspan="2">) Q( b7 R8 b4 c" b
    <bean-message bundle="myKey"key="greeting"/>
    6 H* C5 c1 G6 D0 z% c</td>
    & i3 K  D. f3 Z; ]+ u</tr>
    2 p$ m8 o4 u0 \  C. r + i- b3 \1 A: H8 N: R4 h2 w' P$ R
    <tr>
    3 N3 t+ k* c" l  n4 u<td colspan="2">
    - a' |) \% C8 v! O6 u<bean-message bundle="myKey"key="username"/>
    9 J  u) v/ W9 `2 u) D6 d7 i. k9 j</td>4 c0 H: z1 |9 a3 Y, {
    </tr>- o+ f7 g1 r8 Y- e7 B
    / q& y& D% a+ w3 j! \4 F
    第四步:切换浏览器的语言类型,工具-->Internet选项->常规->语言
    6 l$ y4 \9 p$ @
    - Q0 H% c( |  _/ a% B7 h在struts-config.xml文件内的<message-resources>标签中也可以不指定key属性,那么,该<message-resources>标签指定资源包将成为struts的默认资源包;响应的,在JSP文件内的<bean:message>标签中特可以不指定bundle属性,这表示要在struts的默认资源包中去查找信息.
    ) c4 u( [  \) l) N$ G7 f  U2 o 3 P% L; L1 C' \& G' x( N) L, j
    9.bean-write标签的讲解  ^. b) ?! ^/ k- b  `
    <bean-writescope="request" name="user"property="username"/>
    6 t4 I$ n9 ?; I$ N- b1 Q从哪个域中输出指定bean以及bean的一些属性/ c: L2 e* v3 n: R8 P& I

    8 d* [" l5 k( A10. DispatchAction的讲解; \+ [2 ?7 S. O) {
    (1).实现一个模块,需要对学生信息进行CRUD操作:& d& h; ~, M" B+ ~, `- N; E# ?
    AddStudentAction
    . R1 C: u/ @8 k+ W5 r: u# KDeleteStudentAction
    % u. e" \7 u3 }8 @( Y: c- j# sUpdateStudentAction
    * O4 Z: p) B3 L7 c# \* eQueryStudentAction1 w/ J8 V( F( n
    分别调用四个类的execute方法,但是四个定义四个类有点过多.所有可以合并相关的Action0 H1 M" s6 l. F" y/ K; D
    8 a: z3 r# w* g0 @" d
    (2).特殊的Action的使用ispatchAction,起作用是可以减少Action的数量,使用方法:继承DispatchAction,在里面添加上需要使用的所有方法,参数和返回值与原来的execute方法完全一样,配置文件中带上parameter属性,如parameter="opertationType";使用的时候才用如下形式:/*.do?operationType=add
    " }# c) Y5 K0 J
    . f# n& s4 a. Y5 t/ j! p; g9 ?(3).一个Action可以调用CRUD的方法,DispatchAction中的四个方法的参数和返回类型都和原先的execute方法一样,就是将execute复制四份,同时将方法的名称改为addStudent,deleteStudent,updateStudent,queryStudent即可,然后在这四个方法中编写响应的代码.
    : m: V1 A) z  f' z & F& b$ z) i5 ]
    (4).在地址栏中输入..../DispatchAction.do?operationType=addStudnt,就是访问增加学生页面
    3 ^4 A+ e7 _" k+ |
    " B& M2 T; M+ o% J(5).第一步:可以开发一个DispatchAction的子类,这里的方法注意必须与原来的execute方法同参数同返回值
    ' R9 {9 Y9 _5 G0 N6 a( B: `1 P第二步:配置parameter属性; J: e( W8 M0 G4 K- p

    1 {5 D- N0 y, D, m- m2 b$ I: O11.ForwardAction的讲解
    7 C8 J9 k, a9 k& c(1).访问jsp访问形式:需要统一访问模式*.do;所有的jsp在web-inf下,所以请求jsp之前需要先访问action8 y, Y: m$ |9 s3 x6 c: c2 K

    4 S+ _0 [4 _2 m+ q(2).因为每个jsp对应一个Action,可能要许多Action,所以我们只定义一个跳转Action:ForwardAction 访问:/testForwardAction.do,目的是统一以*.do形式访问所有的模块;<action path="/testForwardAction"forward="/TestForwardAction.jsp"/>9 ?6 p7 o, i, H7 k) b' ^& s! H
    : N" e+ Y8 V( T% M' \4 q1 J' \
    12. logic_iterate标签的讲解
    ' W0 b* _3 g, Q9 p  e(1).逻辑标签:逻辑库的标记能够用来处理外观逻辑而不需要使用scriptlet,struts逻辑标签库包含的标记能够有条件的产生输出文本,在对象集合中循环从而重复的产生输出文本,以及应用程序流程控制,它也提供了一组在jsp页面中处理流程控制的标记,这些标记封装在文件名为struts-logic.tld的标记包中,逻辑标记库定义的标记能够执行下列三个功能:条件逻辑,重复,转发/重定向响应+ b0 y+ g4 q+ r, `
    : I2 q# Y& i) S' R/ P! i( C( f
    (2).Logic:iterate的使用:单重循环,双重循环,对指定的集合的循环,这个集合必须是一个Iterator,Collection,Map,Array;
    - ^; C' |  }# \" h' [0 E
    % [! o  v7 q6 W9 {& U  `(3).
    5 g/ Z" S, z6 u& M6 \) {$ R& C' h<%String[]usernames={"aa","bb","cc","dd"};3 o8 G' p- J9 Q; `0 l: d  G
    request.setAttribute("usernames",usernames);%>
    0 S. A! H5 |7 i <logic:iterate id="username" scope="request"name="usernames">
    1 J: w, f* Y* K. O4 F( E2 U6 ~  ${username}8 [: A+ b9 X4 r7 x$ M& T
    </logic:iterate>5 @1 i6 x3 u* P; D
    name所指代的bean必须是一个集合类型9 z, b) o; W2 J4 a! Z
    name+property:每一个人的所有爱好:4 M% q5 t  e$ Z( i: _' ^( p
    User mengfanlong = new User();
    5 }3 i3 m7 t( P: F5 G$ }mengfanlong.setUsername("mengfanlong");( m, [7 U, Q+ y1 H3 m
    String[]mengfanlongFavorites={"sports","sleep","study"};
    2 i( a' W1 R& J/ g& V  q+ nmengfanlong.setFavorites(mengfanlongFavorites);
    - I- {1 q0 }# [, S0 Z7 R( E  [: b* G用ArrayList存储用户
    $ I0 a7 B/ E4 G# L( C<logic:iterate id="user"scope="request" name="userList">4 U' ?! b) E. w% }$ _
    ${user.username}, d2 [$ U) A+ ?1 {( @" A/ v; j( t
    <logic:iterate id="favorite"name="user" property="favorites">. R- s. Z2 @" e) E4 D+ s. I
    ${favorite}$ j4 G% U+ d( J* ^9 a
    </logic:iterate>
    $ }- Y7 z# P) Z( j, i8 w- E6 R</logic:iterate>- D% C" M# D  l% V& H, k
    2 ^+ V5 Q4 w9 T0 s& [+ G0 Q$ b
    (4).id是迭代时的临时属性,还有其他属性:length,offset,indexId等控制循环变量
    / }( g, s# _* M+ I% w " G& @0 _% G2 ]+ n7 ~7 `# x
    13.struts插件的讲解
    , h( z; p, `4 ~1 y(1).插件(Plugin)生命周期方法:init,destroy,用户提供setter方法,告诉ActionServlet中心控制器把属性设置
    8 K1 x* P# o- b4 n: M 3 O' _0 x/ p+ q+ w
    (2).应用:在struts启动时把hibernate加载进来,就是要把hibernate的配置文件读进来,同时打开hibernate的sessionfactory
    - B6 s' x3 [/ D需要struts.jar包+hibernate.jar包
    + L3 p: B1 X# A' G! Q设计一个类 HibernatePlugin,实现plugin接口,读取hibernate配置文件,打开SessionFactory;
    # g0 J; e( O, E4 Z( q* @; t# l在struts-config.xml配置文件中添加一对<plugin>标签,在plugin中加上子标签/ E- s% O* Q/ p+ _# N& \0 M
    & M! s4 H& G& _) W* z" }
    (3).
    8 R  j. {8 q( X/ Z9 S: @2 B9 D<plug-inclassName="cn.itcast.HibernatePlugin">& u! T0 I! q3 q9 A' Z+ J: X6 \
      <set-property property="hibernateConfigFile"value="/WEB-INF/hibernate.cfg.xml"/>
    ; `; c5 K! G8 n- B' a' o% H </plug-in>
    " d; J. i- x2 C) {! e0 e$ l5 F
    8 C; n7 A9 @' C# V/ ]5 j( n" c# s(4).struts当服务启动时启动,插件在struts启动时读取配置文件时启动.可以在中心控制器ActionServlet的init方法对plugin初始化,destroy销毁了
    - `! ~, Z: w; u. D2 b5 m" _. s 7 G( U; r1 f" a1 ^3 |/ N. W* v
    (5).服务启动-->ActionServlet->读取struts-config.xml->根据各种标签的内容进行一系列的初始化(plugin,ActionMapping): H# O# j" D9 k& }6 j8 |8 L

    $ i. `+ _" T$ }* |14.struts的MVC组件6 c. ^2 m9 B' X* z
    (1).组件:ActionServlet,ActionClasses,ActionMapping,ActionForward,ActionFormBean
    1 K9 j0 [" d7 {+ A5 s
    $ v% Y6 Q# s: Z( g3 q" m, a5 O(2).struts中的MVC:
    & b+ T* s( U3 H7 \第一:模型:本质上来说在struts中model是一个商业逻辑类,开发者实现商业逻辑
    8 @/ O# X/ e+ s4 G第二:视图:View是由与控制器Servlet配合工作的一整套JSP定制标签库构成,利用他们可以快速建立应用系统的界面8 }6 U' X- m4 T! l  }# h" w
    第三:控制器:前端控制器是一个Servlet,它将客户端请求转发到相应的后端控制器Action类
    % \3 M$ s% H3 H$ H0 a* n3 A0 } - I" e. _, ^4 r( a% _
    15.struts的工作原理
    & R" t/ I$ w, S# J9 E第一步:初始化,读取struts-config.xml,Struts框架总控制器(ActionServlet)是一个Servlet,在web.xml中配置成自动启动的Servlet,读取配置文件(struts-config.xml)的配置信息,为不同的struts模块初始化相应的ModuleConfig对象:ActionConfig、ControlConfig、FormBeanConfig、ForwardConfig、MessageResourceConfig( o5 P% W- p3 j3 e3 k6 T
    第二步:等待Http请求:用户提交表单或调用URL向Web应用服务器提交一个请求,请求的数据用HTTP协议上传给Web服务器
    8 E% X# T% f! H. ~# }  y第三步:填充FormBean:实例化、复位、ActionServlet拿到用户的数据填充FormBean、校验、保存等,(*.do)从ActionConfig中找到对应该请求的Action子类,如没有对应的Action,控制器直接转发给JSP或静态页面,如有相应的Action且这个Action有一个相应的ActionForm,ActionForm被实例化并用HTTP请求的数据填充其属性,并且保存在ServletContext中(request或session中),这样它们就可以被其它Action对象或者JSP调用6 V. E$ t+ @, A0 |+ l
    第四步:将请求转换到具体Action处理:控制器根据配置信息ActionConfig将请求派发到具体的Action,相应的FormBean一并传给这个Action的execute()方法.(后台控制器)
    0 ?$ G% [; Q8 o第五步:调用后台的业务功能类完成商务逻辑:Action一般只包含一个execute方法,它负责执行相应的业务逻辑(调用其他业务模块),完成后返回一个ActionForward对象,控制器通过该ActionForward对象来进行转发工作.
    & S8 m( P; x( W% W) [7 {& f0 i3 q第六步:返回响应对象:Action根据业务处理的不同结果返回一个目标响应对象给总控制器(前端控制器actionservlet),该目标响应对象对应一个具体的JSP页面或另一个Action
    4 U  Z' Q( ]: a# z# E, o第七步:转换Http请求到目标响应对象(jsp):总控制器根据业务功能action返回的目标响应对象,找到相应的资源对象,通常是一个具体的JSP页面: N  A! v) P/ Q* s$ ]1 E
    第八步:Http响应:前几步都在服务器端,这一步在客户端,目标响应对象将结果展现给用户目标响应对象(jsp)将结果页面展现给用户9 b7 F; _( V, j8 n# x; P
    控件:总控制器ActionServlet的作用最大,Action(execute)ActionForm,配置文件,将各个模块连接起来
    9 |- s) P$ |5 O/ a3 @/ i注意web.xml和struts-config.xml的区别,总控制器ActionServlet要在web.xml中注册,ActionForm中的属性,Action等的信息都在struts-config.xml中注册.
    3 A! o" o  \! ~' J7 b
    : [! F4 Y4 C% o16.Struts相关基础理论介绍
    8 s& I8 X5 Q" u# p) }- P(1).案例:实现一个用户登录功能,如果用户输入的用户和密码都正确,就跳转到登录成功页面,否则跳转到登录错误页面
    % I( B. [+ }4 _, C , B7 E; v. A( i8 P( W6 U3 e, Q
    (2).知识点:为什么要使用Struts,FrameWork的概念,Struts的概念和体系结构,Struts的工作原理,Struts的组件,Struts配置文件简介,Struts标记库,错误处理框架,校验框架,高级特性,Struts优缺点- D+ W3 M3 f' m  @  J

    ; K" A& N9 k1 d5 d- U(3).框架(Framework):人们用于解决相同或者相似类型问题的方案,可重用性,可扩展性,可收缩性
    $ K3 l- Y2 |. l8 t * N  k  i' ^+ z
    (4).Struts是Apache组织的一个开源项目,主要是采用了servlet和jsp技术来实现的,是基于Sun JavaEE平台开发的6 c) J8 h% u) z6 v% o
      X+ Q2 ^% {! ?+ ]% f7 B0 N
    17.struts中的异常处理- |+ `% M; a, X; N, P  s
    (1).异常的作用:增加健壮性,模块间传递信息2 r3 a3 V. f( o7 B, [7 F0 F/ z; ?
    (2).配置异常:定制异常有两种:全局异常(所有Action使用的)和局部异常(一个Action使用的),9 ?9 T& \7 Q  x* i5 J* P  x6 l. I
    (3).全局异常的定义方法:; B9 B7 J: R. K8 J9 j
    <global-exceptions>" U3 S. u# `( t( J. C& a  O
    <exceptionkey="user.login.invalideUser" path="/Login.jsp"type="com.future.struts.MyException"/>, w" w; f; B0 n2 w
    </global-exceptions>
    8 Y+ J$ o( ~. P5 A; z7 w9 M局部异常的定义方法:
    $ Y  e  [3 _0 R! {<action-mappings>) c" t. S9 Q. l; O6 C
    <action attribute="loginForm" name="loginForm"path="/loginAction" scope="request"type="com.future.struts.LoginAction" validate="false"/>
    9 g0 o2 w' }, _! r3 b7 a<exceptionkey="user.login.invalideUser" path="/Login.jsp"/>2 o2 r" H' g* v# J' O+ C( m! e8 Z
    </action-mappings>7 C5 l: t* K4 I0 C# e, P
    3 F( }% j4 ~" X
    (4).当Action的execute方法抛出异常时,调用异常处理器类ExceptionHandler
    5 m9 B5 {# F# Y; n! s
    + F  e4 w6 b& m' u4 S, \) m(5).怎么使用:
    + u1 `% ~1 Q% R% {% h; e1 z第一步:配置<exception>可以指明path,key,type,Path是指出现异常后的跳转页面,Key是指异常信息的键,对应的值在资源文件当中,Type所要处理的异常
    6 j* I- S$ ~2 L1 x第二步:在相应的action中的execute方法抛出异常' g3 k# O6 y/ M8 `: j0 L) g
    第三步:在异常处理页面(path所指页面)使用html:errors标签打印提示信息( k' E; Z$ R' A  K& w+ N# D

      j5 `9 D5 M: O' b" L* ~18.struts注册程序的组件设计和流程分析4 y+ J( a7 B' {: q# s- L
    (1).编写action,actionform,首先编写actionform,因为action中要用到actionform,在struts-config.xml中注册
      z+ F( [( O, Qactionform,action<form-bean name="addStudentForm"type="cn.itcast.AddStudentForm"></form-bean>  <action path="/addStudentAction"type="cn.itcast.AddStudentAction"name=""></action>
    3 m& ~% |8 Y8 m& S3 y% |! u,还有跳转页面:, j7 @  f* v3 ?/ R6 n' d
    <forwardname="addStudentSuccess"path="/AddStudentSuccess.jsp"/><forwardname="addStudentFailure" path="/AddStudent.jsp"/>
    $ y5 ?( U& O3 n- b; s( e% N & L% W# Q. L! n0 b# j0 G3 T3 T
    (2).对action的编写要控制线程安全! H4 P+ w# J6 j! Z6 t- l% E
    " h% q+ b& p( {
    (3).ActionServlet将请求参数封装成FormBean的属性时进行转换而导致的结果,ActionServlet内部调用BeanUtil这个工具包来将字符串类型的请求参数转换成FormBean中对应的属性的类型,然后再将转换结果装配到FormBean中,这时候可以打印出FormBean中各个属性的结果看看,就大概清楚BeanUtil转换后的结果了,我们在以后的项目实战课程中有对此问题的详细分析和巧妙的解决方案
    ! _; n- F: b3 i6 Q. j" t$ u5 ~
    : o- f! n! q" t7 z0 j/ Z0 M% c8 f19.搭建struts开发环境
    - s) @8 d; a) k导入jar包,使用相关类,建立一个配置文件:struts-config.xml 放在web-inf下,web.xml注册struts中心控制器---ActionServlet,注意事项:struts配置文件的位置,预先加载控制器,可以到struts自带的例子程序,从这些例子程序中的web-inf项目下拷贝struts-config.xml文件9 k2 A0 K# ~1 c" J+ u- y: a' R# a
    / X; _0 y9 t2 h
    20.动态FormBean的讲解- ^% z% B1 i6 c! U+ q- t- m
    (1).动态form不需要用户自己写代码ActionForm7 u/ I6 n9 \% t5 z

    3 S$ Y; ]' m  j$ c+ o# E(2).    <form-bean name="dynaFormForm"type="org.apache.struts.action.DynaActionForm">
    ; `# b, n6 X( X/ @) a<form-property name="userName"type="java.lang.String"/>
    / z6 G0 s; l3 ?! l# z8 G9 M; `<form-property name="password"type="java.lang.String"/>5 G: _3 T  ?; C* B- e8 a1 R
    <form-property name="age"type="java.lang.Integer"/>
    7 ?2 `$ l+ p8 k</form-bean>
    9 u' Y% U9 o4 H需要告诉form-bean的名称,属性名,属性的类型等信息,其中type必须为包装类型.! m; y: H  [1 l
    " a- A8 g; y" E0 x% w$ x9 w' Y. J
    (3).提交一个表单数据,用数据填充动态表单,在Action中的execute方法中,
    4 ]. S4 i8 P1 x% L2 hDynaActionForm addStudentForm =(DynaActionForm)form;/ z5 X% I+ _9 q4 [
    String sname =(String)addStudentForm.get("sname");, |; D# k# _& p
    java.sql.Date birth =(java.sql.Date)addStudentForm.get("birth");" T6 K  F% |+ ^

    6 H7 s* |0 C4 W% H0 J21.分析struts程序的执行流程1 z$ e: T( t; _' T
    (1).ie->服务器->控制器ActionServlet(前端控制器)(送ActionForm给后端控制器)->LoginAction(后端控制器,第一句要进行转型LoginForm loginForm = (LoginForm)form;)->前端控制器->显示页面
    7 ]1 p5 R* g: C- O4 V+ S
    3 ^  k, [% Z5 e3 ]2 w(2).if(loginForm.getUsername().equals("itcast")){returnURLKeyWord= "loginSuccess";}else{returnURLKeyWord="loginFailure";}- J  i. X: W# @! A1 S. ~6 |. r
    % d4 x2 X2 M, s1 S; {5 o
    (3).前端控制器是根据struts-config.xml配置文件找到后端控制器,跳转到成功或者失败的页面也是根据struts-config.xml配置文件的.后端控制器才是真正执行代码.
    " o0 \/ a' e7 Q) G) J * M# @7 e* b5 c0 t1 M
    (4).问题:谁来填充From?,什么时候填充?,根据什么内容来填?! E7 _7 C& H* F# X
          ActionServlet怎样把请求派发给Action?
    ' N' C8 u3 {0 s# F# \      Action运行完后怎样跳转?* i) ]$ a, r! q: E; C

    5 G. |6 e8 k$ _# D) n22.分析自己写struts框架的思路
    $ o  K& A/ C3 Y, ~" D  H开发以下类:: m( j$ j1 X: h. P) K9 |( O
    ActionServlet:读取配置文件dom4j,填充form,派发请求:调用对应的action的execute方法,查找响应,跳转
    / c+ Z( C  m0 KActionForm{reset(),validate()}, C5 J( z8 r4 s- M% o
    Action{execute(ActionMapping,ActionForm,HttpServletRequest,HttpServletResponse}
    5 F4 ?3 f5 w$ _6 Z3 N& z+ g2 K+ n8 RActionMapping{path,name,type,validate}使用HashMap存储
    . V0 V) j) f& m. T2 s4 e- T: X; eActionForward{name,path}使用HashMap存储
    & G1 D+ S$ ~  @" L配置文件struts-config.xml% {* P8 l2 z2 k- C4 D
      x3 e; t4 h  L3 `
    23.配置全局跳转
    3 x# {* K) r% N0 u( m3 zActionA----->Error.jsp
    % x. y, r+ v1 EActionB----->Error.jsp
    : P; Y9 w' H* F$ QActionC----->Error.jsp5 N% F3 [9 Y0 r2 }* X! q
    以前的方法是在三个Action中都写上forward
    0 y6 B4 U+ Z+ C& C* `, D, B
    ; \' n. M" f2 _全局跳转:Action A,B,C---->Error.jsp,在struts-config.xml文件中添加标签:<global-forwards><forward name="error"path="Error.jsp"/></global-forwards>
    - K. F. C1 K7 \' F2 n$ c3 u0 d' i3 x $ K4 H- j* N! S- x8 n. n5 j0 p
    24.通过bean_define标签入门struts标签库* F0 @% O! X5 R: x& @
    (1).JSP视窗组件所使用的struts标记库由四类标记组成:Bean标记:用来在JSP页面中管理bean,Struts-bean.tld;逻辑标记:用来在JSP页面中控制流程,Struts-logic.tld;HTML标记:用来生成HTML标记,在表单中显示数据,使用会话ID对URL进行编程Struts-html.tld;tiles标记:使用动态模板构造普通格式的页struts-tiles.tld; B( s* p: |; {  R  s, Z9 @

    + j6 r0 U6 h$ f: o3 ]9 y(2).Bean标记,这个标记库中包含用于定义新bean、访问bean及其属性的标记,Bean标记库将标记定义在四个子类别中:创建和复制bean的标记,脚本变量定义标记,bean翻译标记,消息国际化标记.4 R' x2 l7 H. M3 I9 c
    & c+ M& V2 o! T: @! B0 V7 v
    (3).Bean:define:从已经有的变量或者变量的属性定义一个新的变量:id,name,property,scope(老bean),toScope(新bean),
    - M# y& {0 d2 E5 t2 T: N3 w3 W查找:到scope中查找name的变量获取它的property,
    3 E( p7 V4 p# f0 u6 G8 d  O) F定义:定义一个新变量的名字(id),同时这个属性是必须的  _0 d. b3 X1 h. R: q
    保存:将新变量保存到toScope
    + _8 f2 g1 ]& Q) L% V% Z 1 {, h. b: d# _! B3 K9 m0 @
    25.通过代码了解ActionForm的基本工作流程; r6 f$ ]7 E# r3 x6 F. D- @
    (1).通过ActionForm的构造函数来观察是否被实例化了,通过reset()方法来观察是否执行了复位方法.set()方法,观察他们三者的运行顺序,1 R- h: T/ Q4 S4 s
    7 u0 {; h0 G3 j, _: _2 p. h0 l
    (2).首先调用构造方法,然后调用reset方法,然后在调用set方法,复位只需要只对请求复位.
    . p+ g/ O- t- v4 n" @. ?
    7 D, a" `; z3 S# l- M+ y/ C26.用struts开发简单的登陆示例程序
    0 m) S/ c; U/ X) R; [) b# q( w(1).建一个类,继承ActionForm,注册,修改配置文件struts-config.xml: a) ^9 w4 ]+ G8 e# {+ Z

    ) k" o$ [7 j7 ~% g, P8 Z(2).public classLoginForm extends ActionForm{7 ~4 \5 ^! h: D
               private String username = null;
    ; C; L% Y2 N2 I/ ^, P% Y           private String password = null;' k& {/ e0 F2 s' A1 P, x* ~
    }
    ! Z" m' I% Q0 D4 s! t8 E
    # q4 i2 d, i$ L" g(3).在struts-config-xml文件中添加一个标签<form-beans></form-beans>
    ; ]: Y. s3 U' v$ _5 Q( s( T/ G $ [! ~: a8 P) b  U9 t: x  R# k. }
    (4).开发Action,建立一个类,继承Action,覆盖execute方法,需要强制转型,调用其他模块,跳转(根据关键字,关键字参照action中forward标签中的name属性),注册:修改配置文件struts-config.xml,Path:指明调用者(jsp)能通过中心控制器ActionServlet找到Action,Type:指明该action类全名,Name:该action引用的form的名称
    , S- l' p  D2 Z) n
    5 U3 ^  y( p2 j. c! K( P( i(5).public classLoginAction extends Action{覆盖execute方法}6 F; P- h' D8 H8 W- d" p2 s
    & L+ S5 U5 |6 l; f$ Y, Y0 X$ l; }5 f
    27.用监听器探索ActionForm如何被存储4 e/ I0 z9 h) T! I9 F- c
    (1).怎样检查ActionForm被存储了,有两种方法:一种是从过程去查看,另一种从结果去查看.
    1 D! I: Z; H, A1 F- F  q! h
    1 A' C$ F% I: T. P(2).从过程查看,通过监听器,从结果查看execute方法3 A0 M5 @7 E( I3 Z  D
    6 D, e0 v8 T+ s; _# d2 X4 m- X. z
    (3).制作监听器:定义一个类,AttributeListener,然后实现两个接口,HttpSessionAttributeListener,HttpRequestAttributeListener,在web.xml中写入:<listener><listener-class>cn.itcast.AttributeListener</listener-class></listener>监听session,request域中属性的变化./ C4 c; E9 V4 _: ?

    , }) T- K. A- c6 }+ G28.在execute方法中分析ActionForm的问题: h7 y; B" p0 \+ r7 J. j+ Z9 n$ P
    (1).从结果查看,
    1 J; \. }: L7 Z1 |' f2 ~AddStudentFormaddStudentFormInSession9 w! C7 }. L0 U  z4 E6 h
    =(ActionForm)request.getSession().getAttribute("addStudentForm");) c  h7 s; b7 a  r. @" u
    或者% b+ v0 G/ }$ g5 ]
    AddStudentFormaddStudentFormInScope
    ' s  @: r# x# Y$ s=null;if(mapping.getScope().equals("request")){addStudentFormInScope2 T% Z; T* x5 j& z
    =(AddStudentForm)request.getAttribute("addStudentForm");}else{addStudentFormInScope
    9 Y% H( A3 D) |4 t9 q=(AddStudentForm)request.getSession().getAttribute("addStudentForm");}从不同的域中获取ActionForm
    & Q. z. x7 i: f$ U$ L. H
    6 H5 |7 O5 v# V" V% N: i; ]& k(2).参数form是和存储在域中的form是相同的(ActionForm);: D  j4 R9 Z# v- n  f+ @" M: \
    ! a$ E- T) @! E) z1 J
    (3).没有sname成员变量,但是有setName()方法,所以准确来说是看是否有setName();页面中的控件<input type="text"name="name">中的name是否和ActionForm中成员变量的名称一样,其实不是看成员变量,而只是看标准的set方法,通过反射技术执行.
    红红火火恍恍惚惚
    回复

    使用道具 举报

    admin    

    1244

    主题

    544

    听众

    1万

    金钱

    管理员

  • TA的每日心情

    2021-2-2 11:21
  • 签到天数: 36 天

    [LV.5]常住居民I

    管理员

    9#
    发表于 2015-06-02 12:45:13 |只看该作者
    三、       Struts23 X6 }$ J1 w/ L5 v0 @7 b
    1.  Action名称的搜索顺序
    ( u( W. a  s6 l(1).获取请求路径的URL,例如URL是:http://server/struts2/path1/path2/path3/test.action. R+ L, S1 T& O0 ?

    - L. G9 {2 p: c) k5 m( I* C(2).首先寻找namespace为/path1/path2/path3的package,如果不存在这个package则执行步棸3,如果存在这个package,则在这个package中寻找名字为test的action,当在该package下寻找不到action时就会直接跑到默认namespace的package里面中寻找action(默认的命名空间为空字符串),如果在默认namespace的package里面还寻找不到该action,页面提示找不到action6 v$ b2 Q' j3 @% r  d
    ) S6 O  }* Q: T% \- g
    (3).寻找namespace为/path1/path2的package,如果不存在这个package,则转至步棸4,如果存在这个package,则在这个package中寻找名字为test的action,当在该package中寻找不到action时就会直接跑到默认namespace的package里面去找名字为test的action,在默认的namespace的package里面还寻找不到该action,页面提示找不到action
    " _& @$ Q/ r2 D3 `0 V" Y) R
    9 |+ S& [8 |* r; ~3 j9 X9 ](4).寻找namespace为/path1的package,如果不存在这个package则执行步棸5,如果存在这个package,则在这个package中寻找名字为test的action,当在该package中寻找不到action时就会直接跑到默认namespace的package里面去找名字为test的action,在默认namespace的package里面还寻找不到该action,页面提示找不到action
    2 s- n' c" {- i& G9 R( \$ S
    6 ~# S/ C8 ^6 h2 i(5).寻找namespace为"/"的package,如果存在这个package,则在这个package中寻找名字为test的action,当在package中寻找不到action或者不存在这个package时,都会去默认namespace的package里面寻找action,如果还是找不到,页面提示找不到action+ ^* i* V: T4 w3 c+ C3 e8 K

    ' T/ B/ O, f& \+ H- i(6).在struts2中默认的处理后缀是.action,不加后缀也可以
    $ l1 I* Q2 h1 m. E& E . ~7 `" p1 m8 N( b5 ^
    2.  Action配置的各项默认值
    / G1 t! X5 d0 @' Z, B) g(1).struts1中:<actionpath="/control/employee/addUI"forward="/WEB-INF/page/employeeAdd.jsp"/>实现请求转发,action将请求转发给视图jsp. N  S  J( L+ {' r

    ) w0 C' w1 k8 D7 E3 k# a: m6 K: ]! v(2).在struts2中,<actionname="addUI"><result>/WEB-INF/page/employeeAdd.jsp</result></action>,不需要设置addUI的类路径class属性了
    % L* r/ g/ f1 h2 Z. `5 J% A 1 O6 b" q. h& A& ]& r/ B+ n
    (3).Action配置中的各项默认值:( n$ ^6 X  X9 {+ ^) k
    <package name="itcastnamespace="/test" extends="struts-default">6 x( v& n" V: [6 S+ Y
    <action name="hellowrold"class="cn.itcast.action.HelloWorldAction"method="execute">' I" A. \, z: V: D) ]# v
    <result name="success">/WEB-INF/page/hello.jsp</result>
    9 h3 u. C& h+ L7 Q; S. F# e2 u, e</action>
    / ~) U$ b% m" R1 Z4 V</package>
    8 |$ @% O" [) b  Z1 y0 R如果没有为action指定class,默认是ActionSupport,可以查看ActionSupport的源代码,首先交给ActionSupport类处理.
    * e- C  \2 k  b9 G, J* G如果没有为action指定method,默认执行action中的execute()方法,这个方法的返回值为"success";
    , s$ a6 j$ N: j2 \( h
    2 Z) J* }4 }! O$ c+ B' O(4).如果没有指定result的name属性,默认值为success,正好和execute方法的返回值相同,所以可以实现视图的转发
    : R4 P, ^3 r' L6 `! n/ ~
    % e! S- B* N( C  `" Y% P% Y3.OGNL表达式
    8 U* F/ O/ ~9 R( m% S(1).OGNL表达式:OGNL是Object Graphic Navigation Language(对象图导航语言)的缩写,它是一个开源项目,struts2框架使用OGNL作为默认的表达式语言
    & U) R8 B$ C5 S  R- D第一:相对EL表达式,它提供了平时我们需要的一些功能,如:
    * O% K7 F9 X7 \支持对象方法的调用,如:xxx.sayHello();
    ; O% G  T; \, \, I# h6 q第二:支持类静态方法调用和值访问,表达式的格式为@[类全名 (包括包路径) ]@[方法名|值名],例如:@java.lang.String@format('foo %s,'bar')或@cn.itcast.Constant@APP_NAME$ |  r0 c. u$ ~% B
    第三:操作集合对象3 h/ D' l0 N2 F+ d% q! |- b

    % R5 q1 C: Q5 K+ S9 D5 c& E- O(2).OGNL有一个上下文(Context)概念,说白了上下文就是一个MAP结构,他实现了java.utils.Map接口,在struts2中上下文(context)的实现为ActionContext$ M- a: L1 [) `5 a; L( _5 ^( ^

    ) P9 p6 w0 L' [8 s; o(3).struts2中的OGNLContext是实现者为ActionContext,它的结构为:OGNL Context:ValueStack(值栈,他是根对象),parameters,request,session,application,attr,当struts2接受一个请求时,会迅速创建ActionContext,ValueStack,action,然后把action存放进ValueStack,所以action的实例变量可以被OGNL访问
    8 Y1 R5 B# {3 E/ Y" F; \
    3 V3 p% M( G9 [(4).当要访问某个对象只需在其前面加上一个'#',例如:#request,当然有一个特殊的例子,就是根对象,会省略'#',OGNL会设定一个根对象(root对象),在struts2中根对象就是ValueStack(值栈),如果要访问根对象(即ValueStack)中的对象的属性,则可以省略#命名空间,直接访问该对象的属性即可.( h  Y6 y2 _" s. X0 l3 R6 a

    # j0 H, E" x3 E/ B! b) F(5).在struts2中,根对象ValueStack的实现类为OgnlValueStack,该对象不是我们想象的只存放单个值,而是存放一组对象,在OgnlValueStack类里有一个List类型的root变量,就是使用它存在一组对象,在root变量中处于第一位的对象叫栈定对象(存放action),通常我们在OGNL表达式里直接写上属性的名称即可访问root变量对象的属性,搜索顺序是从栈定对象开始寻找,如果栈定对象不存在该属性,就会从第二个对象寻找,如果没有找到就从第三个对象寻找,依次往下访问,直到找到为止。大家注意:struts2中,OGNL表达式需要配合struts标签才可以使用,如:<s:property value="name"/>,value属性接受的是OGNL表达式,搜索是否含有name属性
    ; ]3 s9 X1 |4 F1 }- z4 k+ I
    # `; x) x5 `5 {(6).由于ValueStack(值栈)是struts2中OGNL的根对象,如果用户需要访问值栈中的对象,在JSP页面可以直接通过下面的EL表达式访问ValueStack(值栈)中对象的属性:& i0 m/ [+ j6 o+ C) s; T
    ${foo}获得值栈中某个对象的foo属性
    ! O5 H0 a& D3 z如果访问其他Context中的对象,由于他们不是根对象,所以在访问时,需要添加#前缀, P& Q8 P& r+ `+ x  G
    第一:application对象:用于访问ServletContext,例如#application.userName或者#application['userName'],相当于调用ServletContext的getAttribute("username");( ?$ r2 B( A7 s* r
    第二:session对象:用来访问HttpSession,例如#session.userName或者#session['userName'],相当于调用session.getAttribute("userName");3 J' k8 {+ \# m1 ^& s3 r) P2 i
    第三:request对象:用来访问HttpServletRequest属性(attribute)的Map,例如#request.userName或者#request['userName'],相当于调用request.getAttribut("userName");
    & S& @" L( a/ F  m' j. h第四:parameters对象:用于访问HTTP的请求参数,例如#parameters.userName或者#parameters['userName'],相当于调用request.getParameter("username");
    9 Y3 N+ Y7 p5 l9 p$ n' _第五:attr对象:用于按page->request->session->application顺序访问其属性.1 |+ ]) }3 d4 N6 |% l
    % f. o/ i. ?5 ]5 }. n
    (7).为何使用EL表达式能够访问valueStack中对象的属性:原因是struts2对HttpServletRequest做了进一步的封装,简单代码如下
    6 l* T. B& B& ?$ q* _% d! epublic class StrutsRequestWrapper extends HttpServletRequestWrapper{
    , T+ X/ j. d1 L+ e* I. S4 f2 @publicStrutsRequestWrapper(HttpServletRequest req){" G5 W+ v) ]& m2 ?0 ]1 ~, {" H
    super(req);
    / a; O2 C: v/ A: [# K}8 ~; C  B/ K' Z0 T, A3 w4 N: s
    public Object getAttribut(String s){
    6 L8 W0 w; f! H" T4 B' C3 I ActionContext ctx = ActionContext.getContext();6 C; E" e+ S5 h* X* A8 g0 s
    Object attribute = super.getAttribute(s)//先从request范围获取属性值' _1 i! L. k4 T# {
    if(ctx !=null){
    0 c8 p" u8 {/ u) x4 U! l3 d* B if(attribute==null){
    0 w2 v8 p! b0 u5 A9 w5 h% h....% X( o% E) c& X, [
    ValueStack stack =ctx.getValueStack();
    ! l* K3 |% G" U' J8 {* k$ Fattribute=stack.findValue(s);//寻找规则就是前面的寻找规则
    * j' y9 }1 V! g, Z& T7 v2 _....& c+ c  Z, @4 V
    }
    ; D6 }. J' ]  N0 H; Q2 d}
    : `& V% m" X  o8 ?9 rreturn attribute/ Q3 h6 u: X2 D- t* N1 S1 c5 t2 M
    }
    : [, ~! F  `. h4 ~1 v8 R! T}1 A/ ~4 z/ H9 K, k! q! r( g
    EL表达式只能访问ValueStack对象中的内容- G0 P( q$ M) n4 d& N( I
    + d. y9 H$ n/ J4 c* I9 r! C. n
    (8).采用OGNL表达式创建List/Map集合对象,如果需要一个集合元素的时候(例如List对象或者Map对象),可以使用OGNL中同集合相关的表达式,使用如下代码直接生成一个List对象:
    0 q- h# b+ i; w1 B) j. h<s:set name="list"value="{'zhangming','xiaoi','liming'}"/>把当前迭代的对象放在值栈的栈顶
    1 R) Q' y' \% @2 x8 W" f; k<s:iterator value="#list">9 ^  s: l- n. J6 _* E3 r/ l( Y. W
    <s:property/><br>
    / B8 d1 w6 j! P/ V, q6 A# w1 C! R</s:iterator>  p  m$ e& l2 h1 b) U; C8 Y7 f
    Set标签用于将某个值放入指定范围/ h. t7 f) Y% E$ \' G& W( @
    scope:指定变量被放置的范围,该属性可以接受application,session,request,page或action.如果没有设置该属性,则默认放置在OGNL Context中.
    ; m, m! [) u' Xvalue:赋给变量的值,如果没有设置该属性,则将ValueStack栈顶的值赋给变量& K3 h. z4 I2 w" B7 P1 N
    生成一个Map对象:
    6 c# s/ {6 S0 W0 v<s:set name="foobar"value="#{'foo1':'bar1','foo2':'bar2'}"/>; {' g" T+ K8 _% o1 G) f( m% X+ [
    <s:iteratorvalue="#foobar">//迭代标签,把当前迭代的对象放在值栈的栈顶(entry对象),foobar是Map对象和request等对象是同等地位,访问时需要使用'#'
    " o( H/ `% }6 w<s:property value="key"/>=<s:propertyvalue="value"/><br>+ M- l5 u  e$ h/ ?* ~( A5 k
    </s:iterator>
    / u5 H" x; w( ]) S$ v4 C4 T! H0 P数字不用任何符号,字符串使用单引号('),对于Map采用的是maps.entrySet()这个方式进行迭代的.
    0 ^: a9 O( Y* R* b
    ' o6 n; A- V6 {1 G1 `7 M9 q1 J% d7 N(9).property标签用于输出指定值:0 v' I3 T% K! q3 |! R2 t) h
    <s:set name="name"value="kk"/>
    ) k$ \2 h! |3 R) B( F# f0 m) l- t2 o<s:property value="#name"/>) d1 R. v# ?& z* s  c
    default:可选属性,如果需要输出的属性值为null,则显示该属性指定的值
    8 Z6 M# z2 {- _+ m1 [; V  Descape:可选属性,指定是否格式化HTML代码
    6 _" R4 |9 X2 G- U. x$ p6 G( g! Jvalue:可选属性,指定需要输出的属性值,如果没有指定该属性,则默认输出ValueStack栈顶的值8 i. c. S, x/ j5 Q
    id:可选属性,指定该元素的标识5 g8 E6 X9 A9 h. M# m

    0 X; p# k! Z- V, t* r$ l7 O# |2 K! Q(1)0.对于集合类型,OGNL表达式可以使用in和not in两个元素符号,其中in表达式用来判断某个元素是否在指定的集合对象中,not in判断某个元素是否不在指定集合对象中,如下所示:6 T6 H8 v' K0 c+ g1 W1 D
    in表达式:6 V" W, ?6 C0 h& d7 g$ n$ F
    <s:if test="'foo' in {'foo','bar'}">
    5 L6 W/ W4 n6 w  _) c2 N
    , y& _# R/ P% T) w</s:if># K0 ~1 A" D' t) d2 \6 k
    <s:else>+ i8 m. x" T& r
    不在9 n" h0 n5 L6 g& h
    </s:else>- }4 |, @$ p$ N2 M+ V
    no in 表达式. C6 n! M& D& h
    <s:if test="'foo' not in {'foo','bar'}">, b" u$ S4 T$ ]# l
    不在4 M) C+ o3 x8 R7 K+ m
    </s:if>
    & t4 u( B0 W0 a. X, t: Z<s:else>
    2 _+ y2 R+ }$ F4 U. C* d  j6 Z! u, J+ h5 O+ l- t. Y! X/ O
    </s:else>
    9 K% l! Z3 ^  s1 q# j$ c* a( O ; e2 k+ I6 e0 X4 J" l0 m2 L
    (11).OGNL表达式的投影功能,除了in和not in之外,OGNL还允许使用某个规则获得集合对象的子集,常用的有以下3个相关操作符:
    " a( Z7 [0 b5 e! o9 g/ \3 h1 f4 U第一:?:获得所有符号逻辑的元素- q% Y0 I9 U% f  Z3 |/ T3 n
    第二:^:获得符合逻辑的第一个元素9 Y. N- x8 p4 r1 d5 U/ x; s
    第三:$:获得符合逻辑的最后一个元素
    * h9 X) a2 X- \4 o" |) R例如代码:
    1 E; n0 m+ N3 H9 o8 [. y5 G- h<s:iterator value="books.{?#this.price>35}">" X% e9 W) f  Y
    <s:propertyvalue="title"/>-$<s:propertyvalue="price"/><br>7 P7 W4 o; q6 {  y
    </s:iterator>
    ; ?( u" t) v8 L- q; ~在上面代码中,直接在集合后紧跟{}运算符表明用于取出该集合的子集,{}内的表达式用于获取符合条件的元素,this指的是为了从大集合books筛选数据到小集合,需要对大集合books进行迭代,this代表当前迭代的元素,本例的表达式用于获取集合中价格大于35的书集合.
    5 ]' s4 U' }, g  R- g& }public class BookAction extends ActionSupport{
    3 P1 W, V; c: }' l2 `; Wprivate List<Book> books;
    8 N: k4 _" V3 ]) |1 }2 v : z( S: C/ M9 s# p
    @Override
    9 q) T4 [$ I5 h$ f2 epublic String execute(){
    5 @9 E1 R8 S$ C# e) [! Z* Nbooks = new LinkedList<Book>();4 Z( S4 F8 E6 `; z3 K3 B/ i: V7 s+ v4 r. U
    books.add(new Book("aadfsd","spring",23));
    * h9 S1 a& _5 @' t( nbooks.add(newBook("basdfd","ejb3.0",15));; k- E' J4 s3 R8 j. [( N+ F
    }
    4 v" ~  j" U3 J8 A3 d& u3 J}# t# F- b3 }1 @
    9 L8 {% Z. }3 p! S: c3 `) ~
    4.result配置的各种视图转发类型2 z& e. B5 j" n7 ~
    (1).在struts1中有两种视图转发类型:容器内转发,容器外转发(重定向);* q% N/ k" |% I$ `4 j" h! P
    <actionpath="/control/employee/manage".../>4 l' V0 E% |+ z% d8 D' V9 r: z
    <forwardname="add">/index.jsp</forward>
    1 y+ L- P' D5 E! S) _<forward name="add"redirect=""/index.jsp</forward>
    + m1 H- p1 t7 r7 W7 y</action># x. b! G# y+ [
    7 S! _$ Q9 r9 G+ _) h
    (2).struts2中的视图转发类型:result配置类似于struts1中的froward,但是struts2中提供了多种结果类型,常用的类型有:dispatcher(默认值)、rdierect、redirectAction、plainText;dispatcher对应于struts1中的容器内部请求转发,redirect对应于struts1中的容器外部转发(重定向)
    : _, F' X% s9 O4 d* \
    4 z1 [. h6 P& B+ \+ j) H(3).<actionname="helloworld"class="cn.itcast.action.HelloWorldAction">
    1 C( q9 t" ?2 ^6 V; c* S, d/ _0 y<resultname="success">/WEB-INF/page/hello.jsp</result>5 e; I) v+ b. y4 z! b, q! f
    </action>
    : q! i3 C. l+ y! g, ^: a$ c在result中还可以使用${属性名}表达式访问action中的属性,表达式里的属性名对应的action中的属性,如下:<resulttype="redirect">/View.jsp?id=${id}</result>,使用重定向可能需要将Action中的数据属性代入视图页面,这种方式太重要了,很实用.这种表达式叫做ognl表达式,struts1中是没有的,只能将属性值在代码中写死了,不像struts2中的这个表达式,很灵活.当传递的属性是中文时,需要进行URLEncoder.encode("传智播客","UTF-8")编码.8 |) R% \$ d3 C$ Z( V" E3 ]
    6 i& l2 f' ]! K9 Y7 ~- L
    (4).下面是redirectAction结果类型的例子,如果重定向的action中同一个包下:
    / f7 s, b; d/ @. x" s/ G4 F$ G! I<resulttype="redircetAction">helloworld</result>
    ( ^0 ?% R9 f3 X; \如果重定向的action在别的命名空间下:
    : `. f1 {' v6 K7 z) b/ X% }* ~<resulttype="redirectAction">
    3 a5 s6 M7 w" }5 G<paramname="actionName">helloworld</param>
    / N: X# x+ I" ?& F7 m<paramname="namespace">/test</param>* K  J& z' e; X4 \
    </result># q( |1 ~( }1 _$ i
    当添加用户完后,可以回到一个用户列表,此时可以重定向到action! x1 ^3 g3 O. s

      R& {5 j( [  N$ y; t  ]  W# l(5).plaintext显示原始文件内容,例如:当我们需要原样显示JSP文件源代码的时候,我们可以使用此类型
    3 F# v" F9 C: r* a1 ?4 o  c' L2 N% k<result name="source"type="plainText">2 b8 [6 m- K; t8 G
    <paramname="location">/xxx.jsp</param>7 W9 X& J; K6 n7 ^9 G
    <paramname="charSet">UTF-8</param><!--指定读取文件的编码-->/ Y# d% K1 o9 h, i0 V% V
    </result>, D2 f5 i+ k6 o& ~$ @
    在Eclipse中jsp是用UTF-8编码存放的,当读取jsp的内容时,是用本地字符编码的,可能出现乱码,所以要设置读取文件的编码集.4 O: C- e) M% F  ?( G& L
    + b7 X2 V! h  N: R* v" Q) X/ N$ f
    (6).浏览器重定向的JSP不能放在web-inf目录中,而请求转发的JSP可以放在web-inf目录中
    4 Y$ C$ H# ^) Q 5 O9 r/ O! t- Y- [- M
    (7).struts2中的全局视图:
    / B. _/ v; _4 K! M8 h1 E<global-results>1 {6 Y2 c( M5 t/ n# {
    <resultname="index.jsp"></result>' ~$ H) m) l/ t, _9 X
    <global-results>( G5 f% X! N1 _2 V5 O, K
    和struts1中的全局视图是很相似的
    8 f: {2 K0 X! h- O# ?& |我们可以定义一个包,然后将全局视图的配置放到这个包中
    - q1 T5 U. |1 |/ M0 {: }5 J' l, _ ! @0 K9 T1 X# j. |
    5.struts2常用标签7 P, @5 N: [7 P6 q( }3 b
    (1).property标签用于输出指定值:9 f' ?* s; L8 i# \  M
    <s:setname="name" value="kk"/>
    8 [* H4 h; a/ k7 @- Y5 B* @<s:propertyvalue ="#name"/>
    ! p6 w* T3 q  q3 s6 o8 P) ?: Xdefault:可选属性,如果需要输出的属性值为null,则显示该属性指定的值) K2 P: `7 _# B- i+ r
    escape:可选属性,指定是否格式化HTML代码- n- d  k8 P5 I, P4 z
    value:可选属性,指定需要输出的属性值,如果没有指定该属性,则默认输出ValueStack栈顶的值/ ~0 A' g. i. _. h( T0 [  o7 J
    id:可选属性,指定该元素的标识5 g8 M# y) p/ b, y. D- J' g

    ( s) ~8 J; }5 w8 E! g# x9 m(2).iterator标签用于对集合进行迭代,这里的集合包含List、Set和数组
    + L/ f: E" O9 E# [1 D- @<s:set name="list"value="{'aa','bb','cc'}"/>
    / ~' {7 r, Y: n  S) y6 E<s:iterator value="#list"status="st">
    8 \8 s/ |5 Z: _/ D) {4 b* j<font color=<s:iftest="#st.odd">red</s:if><s:else>blue</s:else>>
    # W1 q" F  }0 h+ \( G<s:property/></font><br>. ^! z1 r% _/ h+ J0 I
    </s:iterator>
    % d1 J5 |# |. U) i9 g* N2 c! uvalue:可选属性,指定被迭代的集合,如果没有设置该属性,则使用ValueStack栈顶的集合,
    - _- w! ^" d2 B) z" mid:可选属性,指定集合里元素的id(已被标注为过时)
      m& P" o% L+ ustatus:可选属性,该属性指定迭代时的iteratorStatus实例,该实例包含如下几个方法:
    6 J5 P2 h* K% f6 q. M# {. ]int getCount(),返回当前迭代了几个元素& |0 _  Q  s) K* x) t& H
    int getIndex(),返回当前迭代元素的索引% W/ I$ v' Q( o3 ^
    boolean isEven(),返回当前被迭代元素的索引是否为偶数+ M; r) X" l% M/ G2 d7 b
    boolean isOdd(),返回当前被迭代元素的索引是否是奇数
    " S5 l1 h( W, w( u. Z) mboolean isFirst(),返回当前被迭代元素是否是第一个元素
    ) R  Q" d7 X2 [, h! Uboolean isLast(),返回当前被迭代元素是否是最后一个元素% l3 |2 j# h3 n* |8 q

    9 H+ w: X, G. [/ r; [$ I6 e$ K* p(3).<s:setname="age" value="21" scope="request"/>
    1 g% H0 \4 V5 }- p- Q<s:iftest="#request.age==23">
    1 d1 ]7 n2 h/ |7 \0 Q, c% \6 }, c23
    ; Y0 N2 U" A. d% @8 q# {9 e" W8 F+ g</s:if>- ?. F: m3 r& X# R) h
    <s:elseif test ="#age==21">
    , L* a: I+ r! k2 c( K  ~21" C9 `1 s! f0 s: j# r
    </s:elseif>
    ! B' j: P; n7 H  X9 b+ e<s:else>. d6 ?" p2 {( h  I) L: H
    都不等% t4 B0 R( O+ I$ A* C
    </s:else>! [  R5 R& ]- D* E2 u1 n+ u
      Z4 w/ w' X  f& p+ w' q+ U$ [
    (4).url标签:' X  C5 p6 m8 x0 a3 D% D4 K
    <s:url action="helloworld_add"namespace="/test">
    ) L. p) [3 k6 |$ e<s:param name="personid"value="3"/>
    # a& B# ^) W, A# e5 l& V0 a</s:url>
      t% ?% n9 {0 W0 p生成类似如下路径:; r9 t5 R6 ?8 Y9 ~
    /struts/test/helloworld_add.action?persionid=3
    0 o: M# G8 Y- V7 z1 u4 i2 H, J: U
    1 |; O  `6 |3 j' O$ p当标签的属性值作为字符串类型处理时,"%"符号的用途是计算OGNL表达式的值! p8 |. C6 v; j+ l2 F6 D& O
    <s:set name="myurl"value='"http://www.foshanshop.net"'/>5 I0 S! e; L& q( [/ J2 c% b
    <s:url value="#myurl"/>& F4 W; N" v9 ~  P! r$ I3 M4 Q, W
    <s:url value="%{#myurl}"/>1 R/ H- K" K  }! S! y; ?$ }
    输出结果:9 q( m0 }' Q$ {7 ~+ p- G
    #myurl
      j  y% L7 p; s& Whttp://www.foshanshop.net
      ?5 ^6 {2 Q, t$ W1 r$ t
    * W$ Y1 m1 a: D4 G+ t3 S1 V(5).表单标签checkboxlist复选框  t9 m. N) b+ ]
    如果集合为list
    $ f1 U( s/ D- B( y<s:checkboxlist name="list"list={'Java','Net','RoR','PHP'}" value="{'Java','Net'}"/>
    5 R. Z& o6 W2 }4 F/ J, @<input type="checkbox"name="list" value="Java"checked="checked"/><label>Java</label>  u4 V5 d6 f, u+ c
    <input type="checkbox"name="list" value="Net" checked="checked"/><label>Net</label>8 n6 }% b$ A& o& z5 E" u
    <input type="checkbox"name="list" value="RoR" ><label>Java</label>2 H2 @* A4 O, s2 p$ e$ D
    <input type="checkbox"name="list" value="Java"/><label>Java</label>1 A6 a7 t" Y: A* B
    如果集合为MAP3 @" V# k: }/ U' u% C
    <s:checkboxlist name="map"list="#{1:'aa',2:'bb'}" listKey="key"listValue="value" value="{1}"/>
    3 M9 R- j. E  y* r5 J1 N生成如下html代码:
    + P, U6 @) g$ F<input type="checkbox"name="map" value="1"checked="checked"/><label>aa</label>$ \& a# _7 [3 J$ t, J: A
    <input type="checkbox"name="map" value="2" /><label>bb</label>5 T0 i* u$ Q0 M& @; }- U6 R+ J( h
    当然集合里面存放的也可以是对象类型
    - G: y- v: Y& T: D1 v  u" T+ g, | 5 r+ r9 I9 y* Z2 r% g2 ~- ]+ P
    (6).单选框
    ( V; L7 N; o8 v4 ]5 ^$ F<s:radio name="beans"list="#request_person" listKey="personid"listValue="name"/>5 w2 E0 @6 W2 `6 O
    4 c1 _6 W: n8 X8 n
    6.struts2的处理流程与Action的管理方式, C( E$ M: f5 I  D
    (1).用户请求->(查看web.xml文件)StrutsPrepareAndExecuteFilter->Interceptor(struts2内置的一些拦截器或用户自定义拦截器)->Action(用户编写的Action类,类似Struts1中的action,针对每一次请求,都创建一个Action)->Result(类似struts1中的forward)->Jsp/html(响应)7 ^2 B' Q  T7 c1 ~+ ^" `
    4 z/ g% z5 D3 @
    (2).StrutsPrepareAndExecuteFilter是struts2框架的核心控制器,它负责拦截由<url-pattern>/"</url-pattern>指定的所有用户请求,当用户请求到达时,该fileter会过滤用户的请求,默认情况下,如果用户请求的路径不带后缀或者后缀以.action结尾,这时请求将被转入到struts2框架处理,否则struts2框架将略过该请求的处理,当请求转入struts2框架处理时会先经过一系列的拦截器,然后再到Action,与struts1不同,struts2对用户的每一次请求都会创建一个Action,所以struts2中的action是线程安全的.1 w4 u0 v8 u4 s* P5 V. _
    2 ^3 S( I( d" s2 H9 D+ T9 }2 d) E
    7.XML配置方式实现对action的所有方法进行校验
      K+ u5 J' Y+ R(1).基于XML配置方式实现对action的所有方法进行输入校验:3 C! v4 @& @  Z8 @
    使用基于XML配置方式实现输入校验时,Action也需要继承ActionSupport,并且提供校验文件,校验文件和action类放在同一个包下,文件的取名格式为:ActionClassName-validation.xml,其中,ActionClassName为action的简单类名,-validation为固定写法,如果Action类为cn.itcast.UserAction,那么该文件的取名应为:UserAction-validation.xml,下面是校验文件的模板:5 F- k5 E% t8 d9 ~
    <validators>% d( ~8 e  e* Q- F4 \0 d! x- C' L
    <field name="username">7 j0 Z7 y3 [5 L$ {
    <field-validatortype="requiredstring">6 |, A/ ~$ |/ l/ L2 f  D6 s- _" E; q
    <paramname="trim">true</param>7 M! e0 ~& r8 ]1 C& w  `7 N
    <message>用户名不能为空</message>
    $ ]: A9 V( a6 s8 R* K</field-validator>
    + t# i1 l' y  h- k</field>" c2 k* A( K: c) z) r
    </validators>4 M- b. Q, C; l* _3 x
    <field>指定action中要校验的属性,<field-validator>指定校验器,上面指定的校验器requirestring是由系统提供的,系统提供了能满足大部分验证需求的校验器,这些校验器的定义可以在xwork-2.x.jar中的com.opensymphony.xwork2.validator.validators下的default.xml中找到,<message>为校验失败后的提示信息,如果需要国际化,可以为message指定key属性,key的值为资源文件中的key,在这个校验文件中,对action中字符串类型的username属性进行验证,首先要求调用trim()方法去掉空格,然后判断用户名是否为空.( c* }! k+ Q$ U% j5 V

    ' b+ `) N- \. x9 I+ o. [. t' d& I2 ?(2).struts2提供的校验器列表:1 l0 [* P. Y( M: e4 M! L* v9 g
    required(必填校验器,要求field的值不能为Null)
    + f) A, f6 [6 |; ?* Srequiredstring(必填字符串校验器,要求field的值不能为null,并且长度大于0,默认情况下会对字符串取钱后空格)5 T! m. {1 j$ D2 k
    stringlength(字符串长度校验器,要求field的值必须在指定的范围内,否则校验失败,minLength参数指定最小长度,maxLength参数指定最大长度,trim参数指定校验field之前是否去除字符串前后的空格)2 `3 P7 {/ o% _7 I% b9 ]/ R
    regex(正则表达式校验器,检查被校验的field是否匹配一个正则表达式,expression参数指定正则表达式,caseSensitive参数指定进行正则表达式匹配时,是否区分大小写,默认值为true)
    " w! r* A6 y  r6 }# p) nint(整数校验器,要求field的整数值必须在指定范围内,min指定最小值,max指定最大值)
    ! d1 r1 D# t# ~' ^  R( w! Ndouble(双精度浮点数校验器,要求field的双精度浮点数必须在指定范围内,min指定最小值,max指定最大值)
    - \$ V; x  t% e) U0 r, H* \5 ^& ofieldexpression(字段OGNL表达式校验器,要求field满足一个OGNL表达式,expression参数指定OGNL表达式,该逻辑表达式基于ValueStack进行求值,返回true时校验通过,否则不通过)8 F) w7 B: L/ O" i
    email(邮件地址校验器,要求如果field的值非空,则必须是合法的邮件地址)0 J% V! |! r& h: T- Q
    URL(网址校验器,要求如果field的值非空,则必须是合法的URL地址): O% S3 }, z0 M% l! q; ~6 Z( Q
    date(日期校验器,要求field的日期值必须在指定范围内,min指定最小值,max指定最大值)$ C* }8 |! |$ _. v
    conversion(转换校验器,指定在类型转换失败时,提示的错误信息)
      _2 }( I% O* X6 Kvisitor(用于校验action中的符合属性,它指定一个校验文件用于校验符合属性中的属性)0 `& V( i$ ]5 w
    expression(OGNL表达式校验器,expression参数指定ognl表达式,该逻辑表达式基于ValueStack进行求值,返回true时校验通过,否则不通过,该校验器不可用在字段校验器风格的配置中)
    1 y  S6 f; d- X' `- n) k9 I$ O % F/ ?* _3 F2 J. N
    (3).![CDATA[文本内容]]:文本内容不会被解析,只会原封不动的当做文本处理
    : r  ?) x3 C" G
    : h7 s- s( _9 C6 H9 {& l(4).编写校验文件时,不能出现帮助信息:9 l' B. E1 K0 G0 _2 C& d
    在编写ActionClassName-validation.xml校验文件时,如果出现不了帮助信息,可以按照下面方式解决:
    - Y% I. O: |3 j8 e7 U: Ywindows->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.dtd4 O9 G6 i) D' O" s+ O) j" p% y

    $ b, c& b$ O  ?* P# C- V8.XML配置方式实现对action的指定方法校验* ]0 b& Z2 r* n5 D
    (1).基于XML配置方式对指定action方法实现输入校验:! \! E5 _: a- ]- u3 c1 v* L# J( L
    当校验文件的取名为ActionClassName-validation.xml时,会对action中的所有处理方法实施输入校验,如果你只需要对action中的某个action方法实施校验,那么校验文件的取名应为:ActionClassName-ActionName-validation.xml,其中ActionName为struts.xml中的action的名称,例如:在实际应用中,常有以下配置:, @+ O' R$ q2 O) [$ b) G1 S
    <action name="user_*"class="cn.itcast.action.UserAction" method="{1}">
    7 [4 c9 O+ _  [5 Y* X  m<resultname="success">/WEB-INF/page/message.jsp</result>
    : u* W7 L0 H; w5 a2 Q<resultname="input">/WEB-INF/page/addUser.jsp</result>
    " J7 I; a- M. s- ^1 s7 Y, X; u</action>" t8 F4 l! E3 e4 {& o
    UserAction中有以下两个处理方法:
    4 n: S& i- Q! b) w4 }6 Dpublic String add() throws Exception{- R0 ^9 L  E, e$ V4 E0 @* P* e
    7 B0 J3 z& F' u' W+ w
    }# r& x2 c5 l+ z2 q/ u( u
    public String update() throws Exception{
    ( s4 L8 W3 D& G. j  ~% E8 e, y5 ?
    & O- b7 v  H# R" ^1 p}
    ; P9 s2 T. S: Q6 F8 j要对add()方法实施验证,校验文件的取名为:UserAction-user_add-validation.xml
    6 |3 u. z0 K% `. T. d$ V. C+ m要对update()方法实施验证,校验文件的取名为:UserAction-user_update-validation.xml: C0 [8 {: R% c2 Z' [
    5 o+ J6 z& l- Q6 \9 C
    (2).基于XML校验的一些特点:
    1 R/ W+ o' ]4 B/ J# d: M7 j% V当为某个action提供了ActionClassName-validation.xml和ActionClassName-ActionName-validation.xml两种规则的校验文件时,系统按下面顺序寻找校验文件:& ?% f& `3 y* _% y
    ActionClassName-validation.xml" a  m( Z5 W- @: s. o
    ActionClassName-ActionName-validation.xml
    $ L# d& h1 Q* D# N" i$ u" f) G% [系统寻找到第一个校验文件时还会继续搜索后面的校验文件,当搜索到所有校验文件时,会把校验文件里的所有校验规则汇总,然后全部应用于action方法的校验,如果两个校验文件中指定的校验规则冲突,则只使用后面文件中的规则。* B) U. U5 n& c1 w4 b
    当action继承了另一个action,父类action的校验文件会先被搜索到
    7 G1 ]: |( H! l2 {6 N8 v# _, J& k! g% ]假设UserAction继承BaseAction
    8 y3 M0 T4 V; U' ^3 u0 n<actionname="user" class="cn.itcast.action.UserAction"method="{1}">
    $ w1 Q6 Z# X9 b</action>
    / G( G! @0 b! j4 n, I访问上面的action,系统先搜索父类的校验文件:BaseAction-validation.xml,BaseAction-user-validation.xml,接着搜索子类的校验文件:UserAction-validation.xml,UserAction-user-validation.xml,应用于上面action的校验规则为这四个文件的总和+ E+ b- H! ?8 t9 h9 v
    * C4 L' \# X5 a, Y- `' [( V
    9.动态方法调用和使用通配符定义action9 y* I4 T& t* w9 v& X& U. ^
    (1).在struts1中实现方法的动态调用:
    3 W, J4 _* j" {, b7 N$ |<actionpath="/control/employee/manage" type="....DispatchAction"parameter="method"/>
    9 d" o5 C0 i+ S6 \/ c& N% a- W: h</action>& C! e; y! F0 p' G
    /control/employee/manage?method=addUI+ f" t- p2 V6 r" Z. o, ?
    但是Action必须继承DispatchAction
    " E% ?2 }! I4 _! M 7 M$ J# o7 Z# W* O8 s: P
    (2).在struts2中有两种方式:
    7 S+ n; ]% {% N, z第一种:(struts2.1版本后就不建议使用了)是动态方法调用:如果Action中存在多个方法时,我们可以使用!+方法名调用指定方法,如下:3 V4 a2 w5 D3 {) b' T7 x/ @
    public class HelloWorldAction{
    / ]" K3 r8 c" t& N- J' Pprivate String message;$ u5 V9 B# `3 r
    ....7 m/ u- [/ w% m1 G
    public String execute()throws Exception{0 {  i4 L: o' j1 F2 h  [
    this.message="我的第一个struts2应用";$ {% K3 ?, Z4 B; J: U- g2 I
    }
    ' V+ K, Z$ D  w. l0 v# [5 P4 Qpublic String other() throws Exception{
    : N# \+ o# h8 b% O& e% }5 Wthis.message="第二个方法";9 [3 y1 O/ \9 J
    return "success";/ s' i- \' y& G9 `& V- k% \) x2 M
    }! S% p/ i( ~& D; @. n/ Z  ?
    }* n# g+ D0 e' q& L
    假设访问上面的action的URL路径为:/struts/test/helloworld.action,要访问action的other方法,我们就可以这样调用:/struts/test/helloworld!other.action,如果不想使用动态方法调用,我们可以通过常量struts.enable.DynamicMethodInvocation关闭动态方法调用:% e: h6 H- h( `6 C; ~" o
    <constantname="struts.enable.DynamicMethodInvocation"value="false"/>& H/ t1 G& ~( I6 y. K* ~
    第二种:使用通配符定义action(推荐使用的)
    3 I6 U9 D4 M( \<package name="itcast"namespace="/test" extends="struts-default">& Q' p: M* Q  \  H8 @
    <action name="helloworld_*"class="cn.itcast.action.HelloWorldAction" method="{1}>( F, m, `( P7 d4 f) K. @
    <resultname="success">/WEB-INF/page/hello.jsp</result>
    7 i" T! a" z+ t/ F9 f* m0 c7 a</action>- g9 t3 k" v6 W
    </package>
    : h6 R0 l) [+ B- j$ L! N0 }public class HelloWorldAction{+ f8 y$ w) i! s: g, q
    private String message;& }3 l# ]$ ]/ N9 i) ^
    ....  k) n2 Z- }7 r5 q+ e! \/ C
    public String execute()throws Exception{' o7 u, ?- Y$ G* V
    this.message="我的第一个struts2应用";
    : m+ V" ~; O# i8 F8 y. S' w}% Z& m( A' [  `- s3 v8 h5 h
    public String other() throws Exception{
    , `7 w; l8 e7 S7 b4 D5 Xthis.message="第二个方法";6 Y/ _7 U" W8 o# M5 `2 @3 {
    return "success";
    0 ~. S1 N  J; H9 C/ O/ v7 w3 L}
    2 a6 Q2 u# x; h8 {}
    " e. S. x- j# d0 \( T9 ^' t/ O& d要访问other()方法,可以通过这样的URL访问:/test/helloworld_other.action
    6 [+ d- ^8 J9 M# n) s" l1 Kname="helloworld_*"后可根据多个*,method={1},'1'表示匹配*的位置
    0 c6 g$ v3 c' i+ \) \+ H6 Hname="helloworld_*_*",method={2}:要访问other()方法,可以通过这样的URL访问:/test/helloworld_xxx_other.action8 H9 \* m& u9 Z4 `, b& q

    5 K. \" `5 ~9 X9 F9 n+ H10.对action指定的方法进行校验# ^  ~. a0 o8 w$ G( s2 c
    手工编写代码实现对action指定方法输入校验:# w/ }) D5 ~) \7 Z5 f. e$ D
    通过validateXxx()方法实现,validateXxx()只会校验action中方法名为Xxx的方法,其中Xxx的第一个字母要大写,当某个数据校验失败时,我们应该调用addFieldError()方法往系统的fieldErrors添加校验失败信息,(为了使用addFieldError()方法,action可以继承ActionSupport(),如果系统的fieldErrors包含失败信息,struts2会将请求转发到名为input的result,在input视图中可以通过<s:fielderror/>显示失败信息.
    ; ~* g; T* ~, l# a9 FvalidateXxx()方法使用例子:
    ; j  Z* L+ {% T7 v! Zpublic Stringadd() throws Exception{return "success";}4 E. q, l# E( l5 Q  H3 g& D
    public voidvalidateAdd() {
    & x2 A9 t# D& nif(username==null&&"".equals(username.trim()))this.addFieldError("username","用户名不能为空");" S6 a: G% U' b) N
    }
    , e2 ~$ m! D, b5 N" P验证失败后,请求转发至input视图:<resultname="input">/WEB-INF/page/addUser.jsp</result># t& S- ^2 Q) v4 h
    在addUser.jsp页面中使用<s:fielderror/>显示失败信息
    % v) i, J  N* s5 t " K) U  ?$ q. J% {; ^% f
    11.对Action中所有方法进行输入校验
    4 n1 M  _, Z  W* s3 ~( |(1).在struts2中,我们可以实现对action的所有方法进行校验或者对action的指定方法进行校验: L8 t" {3 h1 z/ A. V0 ^

    ( o/ b- K, D8 \5 x(2).对于输入校验struts2提供了两种实现方法:一种是采用手工编写代码实现,另一种是基于XML配置方式实现
    8 m  m/ v! p' I& j ; X5 o) a+ o  ?) n, y
    (3).手工编写代码实现对action中所有方法输入校验:通过重写validate()方法实现,validate()方法会校验action中所有与execute方法签名相同的方法,当某个数据校验失败时,我们应该调用addFieldError()方法往系统的fieldErrors添加校验失败信息(为了使用addFieldError()方法,action可以继承ActionSupport(),如果系统的fieldErrors包含失败信息,struts2会将请求转发到名为input的result,在input视图中可以通过<s:fielderror/>显示失败信息.
    1 ~, ?+ W, `4 Z" \" ~6 fvalidate()使用例子:
    ' f  R6 e0 u- R1 i( Ipublic voidvalidate(){
    9 H7 w) E, x/ v3 a6 z/ I# M0 @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","手机号的格式不正确");}}. U4 b! R& h$ K7 P
    }
    ( f0 p2 L5 }# N) }7 z9 ]5 v9 p验证失败后,请求转发至input视图:
    + E% `( A. F: ~* A<resultname="input">/WEB-INF/page/addUser.jsp</result>
    2 \: K9 O1 D8 f* A2 L在addUser.jsp页面中使用<s:fielderror/>显示失败信息+ b: r/ u( E" U: Z7 F- S

    : s8 A+ _" ~7 S* U* Q0 o6 A12.多文件上传7 n) |- i! ]7 A
    (1).多文件上传,就是在一个文件上传的基础上,将属性File变成数组类型File[]类型即可,同时该字段的名称必须要和上传页面中属性name的名称一样.
    ( x$ }- \: H, q0 C. U4 O7 q然后进行一次迭代,就可以得到所有的文件. g) j! R8 g8 N6 S* Z4 E6 \

    / ]2 I5 z( R  e13.防止表单重复提交
    & d* v# b, Z: [% n8 g& D1 D$ c<s:token>标签防止表单重复提交
    ! x; [( k& x* z! _, k9 d第一步:在表单中加入<s:token/>6 j1 M2 f3 t! T. s+ X% K  R
    <s:form action="helloworld_other"method="post" namespace="/test">( o7 H% e# C- d# \1 C! ?2 Z
    <s:textfieldname="person.name"/><s:token/><s:submit/>9 v3 X) w1 b1 Y; {1 u/ J2 U
    </s:form>
    8 H5 t! o! t4 z: @+ H第二步:<action name="helloworld_*"class="cn.itcast.action.HelloWorldAction" method="{1}">
    0 k8 k5 X6 n1 x$ `<interceptor-refname="defaultStack"/>
    / t& P1 {  @" o6 X1 c" {; F<interceptor-refname="token"/>
    7 t0 B1 k7 ~' E. u3 K<resultname="invalid.token">/WEB-INF/page/message.jsp</result>7 `% L; H2 M5 [
    <result>/WEB-INF/page/result.jsp</result>0 p# Y* g0 \- }0 f* A5 X+ {; y8 V
    </action>
    ( [- y4 @5 ^- u! {以上配置加入了"token"拦截器和"invalid.token"结果,因为"token"拦截器在会话的tlent与请求的token不一致时,将会直接返回"invalid.token"结果+ H8 l, }& b. [4 s( ?$ |$ [) o
    在debug状态,控制台出现下面信息,是因为Action中并没有struts.token和struts.token.name属性,我们不用关心这个错误4 k1 B7 m4 E  U3 ^. d
    使用了<s:form/>标签可以不指定action的上下文标签路径,可以通过命名空间实现.和前面的原理是一样的,在路径后面添加上sessionid号,只是这步操作不需要我们自己得到sessionid号,struts帮我们操作.
    ) I6 P6 T; G+ K# v8 {  L在值栈中的对象,访问无需添加'#'
    + A0 i7 V) O8 r# i; N8 s4 O$ T
    7 _0 H- {2 Z' y' P14.访问或添加几个属性" B3 Q. u; b  ^, f. T. B- K7 E
    (1).访问或添加request/session/application属性,在struts2中的Action中的execute方法中没有Servlet api(没有响应的参数);% D; X/ |- ^# n. y& ]
    public String scope() throws Exception{
    / V9 y) Z/ p3 _+ ^ActionContextctx=ActionContext.getContext();+ d5 [, E' ?- @
    ctx.getApplication().put("app","应用范围");
    7 O& B% A. _7 ?- H- w  v- Mctx.getSession().put("ses","session范围");
    0 u6 M3 l4 T4 u1 u# Uctx.put("request","request范围");5 `; `7 Y: f6 S
    return "scope";% o. t% S) Y. d% U7 b
    }
    7 _5 K" F& B1 b2 ]
    , X! ?; N" z7 O! m<body>
    ' t% x2 B& b5 p5 ]% J${applicationScope.app}% W& u4 c! i) z
    ${sessionScope.ses}* h$ A$ V# ^  B/ S. F3 j& I
    ${requestScope.req}) N+ }7 m& k' a8 J' Z
    </body>
    1 d# b- n. O  \& n' r: o. M
    8 u) F4 s( g2 o) {% t+ l4 `(2).获取HttpServletRequest/HttpSession/ServletContext/HttpServletResponse对象:
    / o' x% T& I) C$ q2 {1 T% I$ W方法一:通过ServletActionContext类直接获取
    1 j; ?2 A& Z' R! u! c4 H) M8 spublic String rsa() throws Exception{+ i0 O4 v* }+ Q/ Y: P
    HttpServletRequest request =ServletActionContext.getRequest();
    4 v& |7 c; y& s# hServletContextservletContext=ServletContext.getServletContext();! ?* g9 M4 D9 o) E& M- o. h: B
    request.getSession();3 r$ {9 L8 f% Q1 q' U6 o7 u
    HttpServletResponseresponse=ServletActionContext.getResponse();
    7 a) E% [* v% h) _return "scope";
    9 g: d$ J% Z! o" v* S) N" ]5 J}
    0 T: p: f; q5 K* J方法二:实现指定接口,由struts框架运行时注入:# K9 j0 R9 o1 g% Y8 {
    public class HelloWorldAction implementsServletRequestAware,ServletResponseAware,ServletContextAware{
    # k" T2 Q8 |2 Z# {( a' }private HttpServletRequest request;3 z% O8 u9 U6 T! a
    private ServletContext servletContext;
    ' u- a, g; h6 h' s- T% z) @private HttpServletResponse response;
    % _6 u" |7 s9 }9 g" ]. hpublic voidsetServletRequest(HttpServletRequest req){
    , ^) {- ]& u1 @/ D5 a$ l4 p2 Vthis.request.req;
    3 L' y% J3 w$ o4 N' E) [}
    4 O/ ^1 _& l- Z. gpublic voidsetServletResponse(HttpServletResponse res){8 m3 J9 L) X4 ~& L5 M/ q5 i
    this.response=res;4 W4 W# l; f. r3 ?
    }
    7 m. M) t. x2 K" o- b: npublic voidsetServletContext(ServletContext ser){
    6 n. N6 d: d6 D) n+ Nthis.servletContext=ser;
    2 r* t  \. t  t5 u6 w8 A1 K- t8 [& t0 ~}
    0 f# C( C. c( e+ p6 t6 B0 e注意1和2的不同,一个不需要得到对象,一个需要得到对象,所以要区分两个的应用场景& j" U3 l) Q! T+ Q9 `' \0 _# ^
    }# R: }( t' V- x$ r
    3 L, Y! x  \3 L4 j, ~
    15.解决struts配置文件无提示问题
    4 c3 g8 r6 y9 f# c找到struts2.0.dtd文件即可,windows->preferences->MyEclipse->XML->XMLCatalog,点击添加strut2.dtd
    ) I+ ?2 m# Q2 Q, U) z9 Q, H5 K
    9 s4 ]9 v6 S  j7 w' h16.介绍struts2及struts2开发环境的搭建
    0 d1 K2 m0 p6 W! y; ](1).struts2是在webwork2基础发展而来的,和struts一样,struts2也属于MVC框架,不过有一点大家需要注意的是:尽管struts2和struts1在名字上的差别不是很大,但是struts2和struts1在代码编写分割上几乎是不一样的,那么既然有了struts1,为何还要推出struts2,主要是因为有一下有点:( u2 c- B+ Z) [  I& x+ T
    第一:在软件设计上struts2没有像struts1那样跟Servlet api和struts api有着紧密的耦合,struts2的应用可以不依赖于servlet api 和struts api,struts2的这种设计属于无侵入式的设计,而struts1却属于侵入式设计,因为其的
    ) I% _  g! A1 j) T- a& p# dexecute()方法中的参数为ActionMapping,ActionForm,HttpServletRequest,HttpServletResponse
    / @/ F% Q+ }) i) u; G& d& z第二:struts2提供了拦截器,利用拦截器可以进行AOP编程,实现如权限拦截等功能
    % d1 k* V: U3 ]第三:struts2提供了类型转换器,我们可以把特殊的请求参数转换成需要的类型,在struts1中,如果我们要实现同样的功能,就必须向struts1的底层实现BeanUtils注册类型转换器才行
    ' F9 ~; d' v% ?2 o$ {第四:struts2提供支持多种表现层技术,如:JSP,freeMarker,Velocity等/ V( m8 A" f6 i9 E- ^: c8 q. P. a
    第五:struts2的输入校验可以对指定方法进行校验,解决了struts1长久之痛,struts1中的validate方法对所有的方法进行校验
    4 I4 S* c6 c/ d9 }第六:提供了全局范围、包范围、和Action范围的国际化资源文件管理实现.
    & U! W. c6 r5 y! E. F7 y! h' F& M * t! d8 M; f5 z7 J; |
    (2).搭建struts2的环境和struts1是相同的,第一步导入相关包,第二步建立struts2的配置文件,第三步在web.xml中注册struts2框架的配置) T7 g5 d: s- u, x7 N

    + y0 S2 e# l( c4 b0 T" @9 G(3).所需的包:struts2-core-2.x.x.jar,xwork-2.x.x.jar(webwork的核心架包),ognl-2.6.x.jar4 F  ~  o) u  q, L

    : N; R- J, Z$ C8 J(4).struts2默认的配置文件为struts.xml,该文件需要放在/web-inf/classes目录下
    : M! ~1 [6 u0 L- _, O, D! r2 n  y
    . @* u) ^+ Y& D# q* E2 X4 d/ r. T(5).在struts1中,struts框架是通过servlet启动的,在struts2中,struts框架是通过Filter启动的,它在web.xml中的配置如下所示:可以参照struts文件夹下的例子中拷贝,在strutsperpareExecuteFilter的init()方法中将会读取类路径下默认的配置文件struts.xml完成初始化操作,注意:struts2读取到struts.xml的内容后,以javabean形式存放在内存中,以后struts2对用户的每次请求处理将使用内存中的数据,而不是每次都读取struts.xml文件
    0 `  x& b: S" D6 o. z  g * Q) d. m/ z2 f( ?2 Z3 g
    (6).自从struts2.1.3以后,下面的FilterDispatcher已经标注为过时了,struts2.1.3后期版本为StrutsPrepareAndExecuteFilter类" m* \- E" ]( v# s) P; ~6 b

    ) f- D$ f- j* w, E# \17.开发第一个应用" F( e6 l, J/ z6 r
    (1).在struts.xml中的配置:
    + t3 p! U+ a$ Y8 b: _& E6 d: j<package name="itcast"namespace="/test" extends="struts-default">
    6 I- ~6 K  B: a <actionname="helloworld" class="cn.itcast.action.HelloWorldAction"method="execute">
    + v6 ]" u0 @& B1 f3 B* K7 |) N$ W <resultname="success">/WEB-INF/page/hello.jsp</result>
    * u9 A& O) g9 J2 \) h2 T</action>
    - A# Q/ t4 n: s2 p2 M</package>
    * k+ B1 w* ^& w在struts2框架中使用包来管理Action,包的作用和Java中的类包是非常类似的,它主要用于管理一组业务功能相关的action,在实际应用中,我们应该吧一组业务功能相关的Action放在同一个包下, [" r- b7 {: j% ^
    配置包时必须指定name属性,该name属性可以任意取名,但必须唯一,它不对应java的类包,如果其他包要继承该包,必须通过该属性进行引用,包的namespace属性用于定义该报的命名空间,命名空间作为访问该包下Action的路径的一部分,如访问上面例子的Action,访问路径为:/test/helloworld.action,namespace属性可以不配置,对本例而言,如果不指定该属性,默认的命名空间为" "(空字符串).当然配置可以减少重复的代码,struts1中的重复代码就可以使用命名空间来解决
    0 A$ J+ B+ X/ n1 o' C5 `* Q; b2 G通常每个包都应该继承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文件中,就可以看到定义了很多拦截器# h. V# T. d; T  u
    <result></result>和struts1中的forward很相似,定义视图* h$ W9 {$ n0 ~; a/ B$ Y

    ; Q; M' ?! H* m8 J% T8 T(2).public Stringexecute(){return 视图的名称;}注意到这个方法和struts1不同,没有参数,返回类型也不同,这就降低了耦合性,非侵入式的编程了.! {  h' h' s, U& ~: W

    ; y, ~: l. r* {6 d(3).在jsp中使用el表达式即可${message},message是Action中的一个方法getMessage()方法,而不是根据Action中的成员变量message
    & ?- g4 L; z/ ~8 r# a
    ) F* E; m' I6 K18.配置Action范围国际化资源文件5 V9 U' q& G  O2 E
    (1).我们也可以为某个action单独制定资源文件,方法如下:在Action类所在的路径,放置ActionClassName_language_country.properties资源文件,ActionClassName为Action类的简单名称当查找指定key的消息时,系统会先从ActionClassName_language_country.properties资源文件查找,如果没有找到对应的key,然后沿着当前包往上查找基本名为package的资源文件,一直找到最顶层包,乳沟还没有找到对应的key,最后会从常量struts.custom.i18n.resources指定的资源文件中查找
    8 }; T9 @7 O/ F& `, u) O
    / X' A4 P/ Q1 M) g& B$ Q7 G(2).JSP中直接访问某个资源文件% R: r/ C' R5 N+ w/ i
    struts2为我们提供了<s:i18n>标签,使用<s:i18n>标签我们可以在类路径下直接从某个资源文件中获取国际化数据,而无需任何配置:  A. @3 v& t- [& d8 T' e$ b8 n$ s
    <s:i18n name="itcast">6 C$ {+ ^: F! n2 B7 u
    <s:text name="welcome"/>
      ]4 N' w+ @+ r' p' A* Y# w+ M7 p) `</s:i18n>. }% @5 L. d6 ]  d) U
    itcast为类路径下资源文件的基本名. F* p4 }2 v4 [4 e  C& L$ B- h
    如果要访问的资源文件在类路径的某个包下,可以这样访问:; A% W+ o8 R) i% X) `$ C7 z
    <s:i18nname="cn/itcast/action/package">; j; g: a8 \8 F* M
    <s:text name="welcome">/ }& i! T" L& t1 m& f
    <s:param>小张</s:param>& H+ y; N1 T2 B! ?- D# y9 z
    </s:text>8 ]" U7 G/ X. Z9 K# A8 s/ H* d
    </s:i18n>) T% _% Y8 j; o$ k+ y
    上面访问cn.itcast.action包下基本名为package的资源文件9 o1 M* X4 B5 f( P0 n/ c
    % b8 t3 o  A0 u8 ^' E5 P
    19.配置包范围的国际化资源文件
    & d0 r% t- Z3 ~5 I& s0 b(1).在一个大型应用中,整个应用有大量的内容需要实现国际化,如果我们把国际化的内容都放置在全局资源属性文件中,显然会导致资源文件变得过于庞大、臃肿,不便于维护,这个时候我们可以针对不同模块,使用包范围来组织国际化文件
    2 G! \6 N( M+ t$ T+ Z方法如下:在java的包下放置package_language_country.properties资源文件,package为固定写法,处于该包及子包下的action都可以访问该资源,当查找指定key的消息时,系统会先从package资源文件中查找,当找不到对应的key时,才会从常量struts.custom.i18n.resources指定的资源文件中寻找.
    , L: F5 Q) G% r
    / F( g+ F' w% N* Z& L20.配置国际化全局资源文件、输出国际化信息
    ! n5 o- A3 R# Z4 O9 e- C(1).准备资源文件,资源文件的命名格式如下:
    + l' [  J) I% ibaseName_language_country.properties9 u/ I' E9 K3 s; W1 Y
    baseName_language.properties
    * y. q+ z2 p) p* c! jbaseName.properties: h' b( u0 X4 g  j; Y
    其中baseName是资源文件的基本名,我们可以自定义,但是language和country必须是java支持的语言和国家。如:
    ; g/ Z6 m- ]: X5 _3 o5 x中国大陆:baseName_zh_CN.properties/ u  x  K7 [1 |; A! H: h
    美国:baseName_en_US.properties7 F' {0 D: a' u" I
    3 {+ s5 Z0 R! n6 ~2 I( j; v
    (2).现在为应用添加两个资源文件:* ]3 q" T, v) ^* S4 J3 g- n
    第一个存放中文:itcast_zh_CN.properties
    " L4 }6 |/ L1 Z0 G8 I: {* l, |内容为:welcom=欢迎来到传智播客
    5 ^/ F) A: c4 `2 t& z第二个存放英语(美国):itcast_en_US.properties
    $ X9 d+ J: c8 _+ K# e6 w8 `3 k内容为:welcome=welcom to itcast7 ^$ G8 [, i* s8 x& t2 K3 U
    5 I: r# N0 A! h; C: r
    (3).对于中文的属性文件,我们编写好后,应该使用JDK提供的native2ascii命令把文件转换为unicode编码的文件,命令的使用方式如下:7 w) f4 G. x: {6 D
    native2ascii 源文件.properties 目标文件.properties,在MyEclipse6.6版本以及后面的版本会自动转换.
    ; t6 o$ X3 q* ]- _  h( \: z* i
    $ V6 t8 _7 A4 }+ p(4).struts2有:全局范围,包范围,action范围的资源文件
    0 V, P! w- [( u  S7 e6 k + C8 T4 f( P/ @
    (5).配置全局资源与输出国际化信息:
    * t) K  _, V7 U3 w$ d) K当准备号资源文件之后,我们可以在struts.xml中通过:struts.custom.i18n.resources常量把资源文件定义为全局资源文件,如下:
    4 U7 i4 d- }( `( G7 H% a! {- n; b<constantname="struts.custom.i18n.resources" vlaue="itcast"/>
    - C4 R- q# q* Y5 N, V  q/ Xitcast为资源文件的基本名" U. U. s# l  L: L
    后面我们就可以在页面或在action中访问国际化信息:- f9 m$ A0 q& ?: s& A2 }
    在JSP页面中使用<s:text name=""/>标签输出国际化信息:2 F$ i6 m" s4 w3 s$ d3 Z
    <s:textname="user"/>,name为资源文件中的key
    & S" h* W3 ^# n2 U; b在Action类中,可以继承ActionSupport,使用getText()方法得到国际化信息,该该方法的第一个参数用于指定资源文件中的key,' {& e8 A3 Q1 Q; Z9 z$ V
    在表单标签中,通过key属性指定资源文件中的key,如:4 h( O, ]/ V7 x0 n! {" R9 i
    <s:textfieldname="realname" key="user"/>. B2 Q) w% f2 |1 m4 v) d
    8 N4 G( q9 L5 x- X5 u, L2 E- N
    21.请求参数接受
    " c$ U) S% o) h) W0 W(1).struts1中是使用ActionForm接受用户的请求参数7 [/ Q% K  n% H/ N) B) H/ F* N

    % c3 ]5 Z5 s* w(2).采用基本类型接受请求参数(get/post):6 S. u  h+ g- E5 L6 f8 B' E9 O
    在Action类中定义与请求参数同名的属性,struts2便能自动接受请求参数并赋予给同名属性:请求路径:http://localhost:8080/test/view.action?id=782 O% g* V9 D# u( m7 ]9 M/ t
    public classProductAction{2 X. M' [+ q; Z9 @" y1 L8 u. f  K* r
    private Integerid;
    ; X3 v: B2 v' B5 E% ]public voidsetId(Integer id){//struts2通过反射技术调用与请求参数同名的属性的setter方法获取请求参数值
    + \* e( |8 y2 x+ Lthis.id=id;. V. g( n3 x& l, _& T7 H
    }. `9 J% Z% n1 x! y
    public IntegergetId(){return id;}
    8 `- M  P3 o: ~3 S- T) K+ j7 I}
    + D- S- v7 B$ |. \( |* d5 k 8 ?* E; [3 E' J# Z% Q1 h' W
    (3).采用复合类型接受请求参数
    + B' }1 Z: D0 l6 N2 Q! B8 z( ]请求路径:http://localhost:8080/test/view.action?product_id=78
    5 I0 u* l2 Q, C2 b9 n; Rpublic class ProductAction{
    / ?: l' J6 U( _1 @* ]/ q' iprivate Product product;
    + H7 o# k! Y: e7 s& lpublic void setProduct(Product product){htis.product=product;}9 W4 k8 _2 n. j" H2 n- @' X* R- z# ^
    public Product getProduct(){returnproduct;}9 W9 ^% D  P9 Y& M
    }1 a# J9 l: @; l% H; c! \* n
    struts2首先通过反射技术调用Product的默认构造器创建product对象,然后再通过反射技术调用product中与请求参数同名的属性的setter方法来获取请求参数值% n! G: O" ?6 K" W6 P* b! M2 W) e
    2 A5 |3 o, L4 E
    (4).关于struts2.1.6版本中存在一个Bug,及接受到的中文请求参数为乱码(以post方式提交),原因是struts2.1.6在获取并使用了请求参数后才调用HttpServletRequest的setCharacterEncoding()方法进行编码设置,导致应用使用的就是乱码请求参数,这个Bug在struts2.1.8中已经解决,如果你使用的是struts2.1.6,要解决这个问题,你可以这样做:新建一个Filter,把这个Filter放置在Struts2的Filter之前,然后再doFilter()方法中添加以下代码:. n2 v) `& g6 F, B. @! ~- ^3 X
    public void doFilter(..){# ?: d1 H  O- z% H9 r
    HttpServletRequest req=(HttpServletRequest)request;
    8 `2 J) @' I* v- m/ G' Jreq.setCharacterEncoding("UTF-8");
    0 p( d% i  p  J) a' g/ `5 s6 Z% jfilterchain.doFilter(request,response);
    " x% K6 O/ y" K5 }8 [0 r}
    ; A6 b6 l' U+ R% ]! H$ T0 u; { 6 B0 R7 r5 l7 [5 x
    22.全局类型转换器
    % b: x6 R# B+ g' q4 K3 U$ W自定义全局类型转换器:将上面的类型转换器注册为全局类型转换器:在WEB-INF/classes下放置xword-conversion.properties文件,在properties文件中的内容为:待转换的类型=类型转换器的全类名
    9 J0 u' q8 b! K8 s1 d8 D; Z对于本例而言,xwork-conversion.properties文件中的内容为:
      d9 E8 A8 |6 B/ W$ Ujava.util.Date=cn.itcast.conversion.DateConverter
    3 a9 R: a% V! A) B8 ^# J- M8 l# j' I 6 q0 E- {. t/ R  K: C* f0 z/ P
    23.输出带有占位符的国际化信息
    2 \- W, y  i5 {" P(1).资源文件中的内容如下:1 F' r& h( U: d, M
    welcom={0}欢迎来到传智播客{1}7 {3 x* I# Q- ]' z. z2 B
    在jsp页面中输出带占位符的国际化信息
    # q% \, x. W+ w) q/ t) E<s:text name="welcom">! [- h- _4 X% q: R+ ^6 h8 b6 p; \
    <s:param><s:propertyvalue="realname"/></s:param>
    " Q8 T5 b0 Y. x: }+ V' D/ O% j<s:param>学习</s:param>
    / ?% @$ S* w9 L4 `6 E2 n</s:text>7 u+ @3 r- L. H' }
    在Action类中获取带占位符的国际化信息,可以使用getText(String key,String[] args)或getText(StringaTextName,List args)方法.; P+ _; P% \. x

    1 ?/ W* z$ p; n4 s1 x1 v(2).占位符就当是一个变量参数,可以传递给定的参数值.
    # O/ E& G1 D, y' \' P! g( C% s
    0 `4 X( e5 Q; Q# w: c24.为Action属性注入值& r+ D- @* Q, O1 U* `
    struts2为Action中的属性提供了依赖注入功能,在struts2的配置文件中,我们可以很方便的为Action中的属性注入值,注意:属性必须提供setter方法,  q9 n* m. ^* U0 U% ^; [" m1 I
    public class HelloWorldAction{' o, o  s4 z- f  c& J, A
        private String savePah;) ^; B" s; i" W7 _  O! t9 n
        public String getSavePath(){
    & w: @9 z  g# T) }) U, ~                return savePath;* s! ?3 m: i# A' q: [# |
              }
    8 q+ g& f) d2 |! B$ O& Z    public void setSavePath(String savePath){# T6 _6 [, v" P% O
              this.savePath=savePath;
    ( h8 k& O1 v3 V% M          }6 m! w7 a: j0 D7 b* q# H
    }
    + h+ `1 K* \( Y. A# `<package name="itcast"namespace="/test" extends="struts-default">. B+ T" v" n* {8 ?& T! o! j
    <action name="helloworld"class="cn.itcast.action.helloWorldAction">' Y1 l2 u3 S& x1 D( n3 L) {0 p4 H
    <paramname="savePath">/images</param>
    8 S0 V% w  _$ T+ U! V  @<resultname="success">/WEB-INF/page/hello.jsp</result>  N1 _9 ]; F* n, ^4 ^+ P
    </action>
    + b% C& c& U" i7 o</package>
    % C/ E; d- O! Q* u上面通过<param>节点为action的savePath属性注入"/images";Action的变量的值,不能写死,经常变换,需要通过配置来设置参数
    + E9 n$ o/ m; R: p- `' M% Z+ x : f! E6 N0 `+ o' I/ A
    25.为应用指定多个配置文件
    1 F; q+ n3 X& T  r' y(1).在大部分应用中,随着应用规模的增加,系统中的Action的数量也会大量增加,导致struts.xml配置文件变得非常臃肿,为了避免struts.xml文件过于庞大、臃肿,提高struts.xml文件的可读性,我们可以讲一个struts.xml配置文件分解成多个配置文件,然后再struts.xml文件中包含其他配置文件,下面的struts.xml通过<include>元素指定多个配置文件:
    9 f' ~: M) t- j8 x<struts>5 T" Q1 f- A4 P% \" `/ L
    <includefile="struts-user.xml"/>
    ' r. c0 |- ~; i- }4 Y( g" {5 ]/ A$ S5 F<includefile="struts-order.xml"/># Q9 Z* L: O( u- ~* c: O
    </struts>
    . u- `+ D. N) Z. k# v通过这种方式,我们就可以将struts2的Action按模块添加在多个配置文件中
    # R/ S' p' c2 U) E' J6 O- A * m) t8 f* c$ |" g) s
    26.文件上传/ V, I& u* m0 n# e; T  B. O3 @* }
    第一步:在WEB-INF/lib下加入commons-fileupload-1.2.1.jar、commons-io-1.3.2.jar,这两个文件可以从http://commons.apache.org下载,在struts2.1以前的版本需要添加,以后的版本就不需要添加  L, B  v) M$ K7 e
    : s& v2 m9 c& f! k+ D' t
    第二步:把form表的enctype设置为:"multipart/form.data",如下:/ A6 C  b" @# E" {
    <form
    " w& \( O, K& F" genctype="multipart/form-data"action="${pageContext.request.contextPath}/xxx.action"method="post">
    * x2 U+ ?7 Q" Z" _+ ?8 B: ?<inputtype="file" name="uploadImage">这个属性的name必须要和类中File名称一样5 _: q8 f2 w9 {. I
    </form>
    7 o9 {" X  {) `) B0 ^  D0 q9 J第三步:在Action类中添加以下属性- X9 R8 T; P1 w1 c% ?5 B8 U2 f& Y9 O$ P
    public class HelloWorldAction{$ T: u  d( `- @# W1 i# Y' ^
    private File uploadImage;//得到上传文件;
    2 B6 z  K, D$ [7 d$ Z- c: Zprivate String uploadImageContentType;//得到文件的类型
    ! N# d( e$ m7 O8 g9 f; u9 Dprivate String uploadImageFileName;//得到文件的名称
    8 A1 V) L' K# X8 @& b! U//这里省略了属性的get/set方法(但是要注意get/set方法是必须的)& c) W0 V' k) U4 y
    public String upload() throws Exception{$ G3 g1 g, P; }$ i- N: q
    String realpath =ServletActionContext.getServletContext().getRealPath("/images");, n/ Z: Q( r2 @
    File file=new File(realpath);' m9 g3 N) o, \5 j& r( k
    if(file.getParentFile().exists())file.getParentFile().mkdirs();//目录是否存在,不存在就创建
    * Q" N# U0 {* X" V# |- m8 a6 qFileUtils.copyFile(uploadImage,newFile(file,uploadImageFileName));
    2 w" a/ Q4 c* ?3 Z2 \return "success";, U3 [2 h. e- x* \) X- i' l
    }& C7 ^9 P% {) _+ ?& c* ~5 n
    }; A1 M: i% n/ X! R5 y; ]5 V% o( M" A

    ! \8 R4 r' v, a) I4 a. N. r5 O(1).如果文件不保存,struts2会把文件保存到自己的目录中,但是当这个Action执行完后,该文件就会被删除,所以我们要将上传的文件保存到硬盘上. m0 v! _+ M" B# W; K
      ~$ y) E( F0 B, {4 l! n
    (2).最好还要判断以下,文件uploadImage是否为空$ _9 e5 v% C8 R3 ~1 {

    - }1 e0 }+ J/ z5 w(3).如果上传大的文件,web都会失败,像一些门户网站上传视频,都是通过一个插件,可以把这个插件看成一个程序,只是这个程序是通过Socket变成的,针对一个端口进行传输数据3 Q3 \& |( ?! I0 L$ ?7 ]' l/ _/ _

      w" f2 d& J$ m2 A; w, m8 Z27.指定struts2处理的请求后缀
    * x& Z# Q" \. E6 A( \* f; D(1).前面我们都是默认使用.action后缀访问Action,其实默认后缀是可以通过常量"struts.action.extension"进行修改的,例如:我们可以配置struts2只处理以.do为后缀的请求路径5 U, s( S( B6 x6 }
    <struts>+ T( z2 `8 ?& R3 U/ \
    <constantname="struts.action.extendsion" value="do"/>  Z3 g8 j3 A$ d* `; u) p6 Y* s+ l
    </struts>, w2 G8 ^: r1 m  e+ e
    如果用户需要制定多个请求后缀,则多个后缀之间以英文逗号","隔开,如:
    5 F, W* t& F3 [5 [8 @<constantname="struts.action.extendsion" value="do,go"/>, Q( Y/ R7 W6 d, V6 A
    + x; X+ U/ o, ^  Z
    (2).常量可以在struts.xml或struts.properties中配置,建议在struts.xml中配置,两种配置方式如下:
    9 `& m# X# j6 M0 }& T0 f在struts.xml文件中配置常量:
    : \3 P( s: d! y. _( `<struts>$ b- L% A& Z( W8 k3 e" ~- x
    <constantname="struts.action.extendsion" value="do"/>
    2 w' o6 i: ~& d# i5 R1 b4 K# F</struts>
    : e( F& S- c5 H在struts.properties中配置常量:
    , t9 s) n0 I3 t- a+ @struts.action.extension=do
    ( I- I2 v8 N5 l1 I因为常量可以在下面多个配置文件中进行定义,所以我们需要了解struts2加载常量的搜索顺序:+ n  W1 B- P/ s) q
    struts-default.xml
    * q. g/ q( p( F1 l; Cstruts-plugin.xml+ a( ?, ~. n& l+ ^. S
    struts.xml
    ( z# \& D9 L7 O9 ^) Tstruts.properties
    0 J. j* D9 `$ q: w8 Qweb.xml& w7 @! e& O' u6 n9 M+ j7 W
    如果在多个文件中配置了同一个常量,则后一个文件中配置的常量值会覆盖前面文件中配置的常量值9 N" U  H, z1 X0 I  N

    - I" u1 v0 p8 Q4 t(3).$ P, B/ U( D" o3 \/ V0 [
    第一:默认编码集,作用于HttpServletRequest的setCharacterEncoding方法和freemarker、velocity的输出:
    2 S- z6 \4 O" }1 L& `<constantname="struts.i18n.encoding" value="UTF-8"/>
    2 f8 M. n: n: a6 }7 k第二:该属性指定需要struts2处理的请求后缀,该属性的默认值是action,即所有匹配*.action的请求都由struts2处理,如果用户需要指定多个请求后缀,则多个后缀之间以英文逗号(,)隔开1 T2 i" \+ ~( R7 N7 X6 X
    <constant name="strtus.action.extension" value=do"/>7 [, P& R, b. K) I9 z
    第三:设置浏览器是否缓存静态内容默认值为true(生产环境下使用)开发阶段最好关闭,不然看不到修改后的数据
    ; n6 a$ u- d) B% ~. t" o! q<constantname="struts.serve.static.browserCache" value="false"/>
    ) P0 [3 X% E- A* ?, ^第四:当struts的配置文件修改后系统是否自动重新加载该文件默认值为false(生产环境下使用),开发阶段最好打开
    . b" E' G/ v+ I) p<constantname="struts.configuration.xml.reload" value="true"/>
    & [, Y- w5 Z  b- q0 X第五:开发模式下使用,这样可以打印出更详细的错误信息
    0 p( l7 }- _8 P( y, J5 A! |; V& K<constant name="struts.devMode"value="true"/>1 c6 W6 A4 K& R: g- Y
    第六:默认的视图主题) }0 h2 o3 S7 I. j2 q. r  y
    <constantname="struts.ui.theme" value="simple"/>0 D8 C6 Q! X1 k- V
    第七:与spring集成时,指定由spring负责action对象的创建2 A/ g) S/ e2 H, F
    <constantname="struts.objectFactory" value="spring"/>
    ) Y! L7 P' {/ g4 [3 @7 v* g1 f, m第七:该属性设置struts2是否支持动态方法调用,该属性的默认值是true,如果需要关闭动态方法调用,则可设置该属性为false
    " i/ X, x7 J( K<constantname="struts.enable.DynamicMethodInvocation"value="false"/>
    . D0 ^: A& a0 w( O8 J+ U  f1 Q) ]第八:上传所有文件的总大小限制5 e' ~+ C( G' s; p( z9 _$ s9 e+ m
    constantname="struts.mulitipart.maxSize" value="10701096"/>
    4 l9 I9 u* F/ k
    2 D8 k9 j1 ]1 u28.自定义拦截器) E0 w7 Z! ~+ E: ~  \$ D! }
    (1).如果用户登录后可以访问action中的所有方法,如果用户没有登录不允许访问action中的方法,并且提示"你没有权限执行该操作"
    2 z2 T/ s9 C* Z4 Q' T1 } 3 K% S. b1 m* {0 C; G+ \$ @5 F
    (2).& h  e4 j; `, I4 d
    <interceptors>
      a4 p! h# O! _9 i/ C6 d  p<interceptorname="permission"class="cn.itcast.interceptor.PermissionInterceptor"/>2 c6 U0 |0 m" v
    </interceptors>1 |/ @! K& w4 q; o( ^
    <actionname="list_*" class="cn.itcast.action.HelloWorldAction"method="{1}">! p/ a* v5 J' {6 g6 @
    <interceptor-refname="permission"/>9 B$ Q9 h2 @8 O8 b7 o: Y4 u
    如果为某一个Action定义一个拦截器,struts2中对Action的默认的很多拦截器都失去功能,所以要想做到两全其美,需要定义一个拦截器栈:
    ( d( X0 F; R" b* ^+ C- i2 Y5 m1 u<interceptors>8 W& W: h' f# c: |/ z
    <interceptorname="permission"class="cn.itcast.interceptor.PermissionInterceptor"/># Y3 J5 h: Y+ |, e/ [. i
    <interceptor-stackname="permissionStack">7 O+ c: H: w2 c: P
    <interceptor-refname="defaultStack"/>
    2 D: C" @; x! R( F- a3 L/ F<interceptor-refname="permission"/>
    8 ^1 @! P. ~' C+ v7 t; }4 q8 k0 B</interceptor-stack>) [6 k; o& t, `/ J( E# C
    </interceptors>
    / N5 w1 h2 U) N9 C因为struts2中如文件上传,数据验证,封装请求参数到action等功能都是由系统默认的defaultStack中的拦截器实现的,所以我们定义的拦截器需要引用系统默认的defaultStack,这样应用才可以使用struts2框架提供的众多功能,如果希望包下的所有action都使用自定义的拦截器,可以通过<default-interceptor-refname="permissionStack"/>把拦截器定义为默认拦截器,注意:每个包只能指定一个默认拦截器,另外,一旦我们为该包中的某个action显示指定了某个拦截器,则默认拦截器不会起作用.2 M+ R8 ^# _7 {: R" X; i
    (3).系统默认的拦截器可以到struts-default.xml中查看,很多功能.系统拦截器放在最前面,自定义的拦截器放在后面.. G$ H5 P8 Z' \+ A

    $ g$ j. N. B7 d. W/ }2 `: U' {0 v29.自定义类型转换器  u- }/ p! P9 T5 D7 v0 ~  N
    (1).struts2中提供了两种类型转换器:局部类型转换器(只对某一个action起作用),全局类型转换器(所有的action起作用)
    6 q# H- ?* c  D - O+ s9 ~8 \: w
    (2).类型转换器必须继承DefaultTypeConverter最好用xwork2.jar中的,重写converValue(Map<String,Object>context,Objectvalue,Class toType){
    $ O4 w. G7 s: N9 nreturnsuper.convertValue(context,value,toType);$ A# p8 d4 k' v4 y
    }7 N9 Q$ V* i' C+ {# z' H
    其中第一个参数和ognl表达式,第二个参数是需要转换类型的内容(是String数组,因为可能有多个值),第三个参数是需要转换成什么类型,要实现双向转换7 n+ ]( `$ l3 j
    % f; R* y& ]% f, J6 X0 u  i  `
    (3).将上面的类型转换器注册为局部类型转换器:
    " J* h2 w: I# ^! s$ k  m在Action类所在的包下放置ActionClassName-conversion.properties文件,ActionClassName是Action的类名,后面的-conversion.properties是固定写法,对于本例而言,文件的名称应为HelloWorldAction-conversion.properties.在properties文件中的内容为:
    4 W* y0 S# A3 P+ g% J$ r" c8 ^+ [需要转换的属性名称=类型转换器的全类名# B0 @; J, c/ G2 k$ N# {# ~
    对于本例而言,HelloWorldAction-conversion.properties文件中的内容为:
    4 @( b' {; ^) _. l, o; e& u. ucreatetime=cn.itcast.conversion.DateConverter
    红红火火恍恍惚惚
    回复

    使用道具 举报

    admin    

    1244

    主题

    544

    听众

    1万

    金钱

    管理员

  • TA的每日心情

    2021-2-2 11:21
  • 签到天数: 36 天

    [LV.5]常住居民I

    管理员

    10#
    发表于 2015-06-02 12:45:41 |只看该作者
    四、       Spring8 R6 H! i/ y- A' s% L- P$ p
    1.     @Autowired注解与自动装配
    3 p3 S2 q; X! q! K, g9 A: C@Autowired2 N- n; h: ]& A( U7 z
    private PersonDaopersonDao;
      B, H* U- u. a6 d4 Z; I+ x0 G拿PersonDao与<bean id=""..../>中的id值进行比较,相同就找到了,即进行类型注解,当然也可以过@Qualifier注解进行名称进行注解.. }! g% S# M8 b3 g5 y: G6 |
    自动装配:8 l" f( a- l/ `9 K2 C6 w
    对于自动装配,大家了解一下就可以了,实在不推荐大家使用,例子:<bean id="" class=""autowire="byType"/>
    ) E1 a0 M3 y% R3 X$ O$ Wautowire属性取值如下:  A$ K- ]- T  f, S. h: ]# b0 U* z
    byType:按类型装配,可以根据属性的类型,在容器中寻找根该类型匹配的bean,如果发现多个,那么将会抛出异常,如果没有找到,即属性值为null8 t# G& l6 N3 x" F
    byName:按名称装配,可以根据属性的名称,在容器中寻找跟该属性名相同的bean,如果没有找到,即属性值为null
    5 r* s' r' l  V/ M3 K- L& pconstrutor与byType的方式类似,不同之处在于它应用于构造器参数,如果在容器中没有找到与构造器参数类型一致的bean,那么将会抛出异常.5 p+ `; _: R* Z5 Q! [8 C
    autodetect:通过bean类的自省机制,来决定是使用constructor还是byType方式进行自动装配,如果发现默认的构造器,那么将使用byType方式.
    $ {8 l/ n9 D1 ~4 Z: w! H
    * a. K% V9 R, F$ j1 @( B8 ]9 d2.     @Resource注解完成属性装配
    ) p' L8 @$ I% `* N' G6 ~4 V; a(1).前面讲到了使用构造器注入,属性的setter方法注入,这里还可以使用注解的方式对Field进行注入
    0 W) m% u$ e% W7 s' O% t , U& t( Y! t  O
    (2).注入依赖对象可以采用手工装配或自动装配,在实际应用中建议使用手工装配,因为自动转配会产生未知情况,开发人员无法预见最终的装配结果
    9 |% S8 j# X( j# p! I. J9 e, r
      k8 ]7 x6 h( |+ r(3).手工装配依赖对象,在这种方式中又有两种编程方式
    1 U# R) f1 c9 ~' j: H方式一:在XML配置文件中,通过在Bean节点下配置,如:- i6 T' p" m/ D  Q( G. U
    <bean id="orderService"class="cn.itcast.service.OrderServiceBean">1 K& b& L) `* o9 [9 z! {( V
    <construtor-arg index="0"type="java.lang.String" value="xxx"/>构造器注入7 H7 _  t# M/ p! Y) f2 W/ j
    <property name="name"value="zhao"/>属性的setter方法注入  k$ ~9 I5 H& e& N
    </bean>
    $ V0 r- {' |# r9 d" F7 V在XML中注入属性,会给XML文件变得很臃肿.特别是对集合类型进行注入时,变得很臃肿.+ o5 I6 I$ U5 x
    方式二:1 S$ T4 i' C9 @  W* |
    在java代码中使用@Autowire或@Resoruce注解方式进行装配,但我们需要在XML配置文件中配置以下信息:( J7 r& Y1 s4 ?
    <beansxmlns="http://www.springframe.....( B; b; o$ E3 l) U( k8 g
    ....
    - h1 c; `$ y: G- Q9 A....
    0 t6 A2 Y+ a7 X</beans>
    ' U, [: `, A3 n" J2 Q3 A这些配置项隐式注册了多个对注释进行解析处理的处理器:AutowiredAnnotationBeanPostProcessor,CommonAnnotationBeanPostProcessor,PersistenceAnnotationBeanPostProcessor,RequireAnnotationBeanPostProcessor,每个注解都有一个注解处理器,注解本身不干活,是相对应的注解处理器在干活
    : o. p& ~, @* k注:@Resource注解在spring安装目录的lib\j2ee\common-annotations.jar
    : a& V0 F4 S) X$ Z/ V4 e( \ : J- Z! {8 Z3 y, J
    (4).在java代码中使用@Autowired或@Resource注解方式进行装配,这两个注解的区别是:8 O5 Q: m, |1 q: c" d0 X; e) }4 C
    @Autowired默认按类型装配,@Resource默认按名称装配,当找不到与名称匹配的bean才会按类型装配.
    , U6 _  E4 [+ I@Autowired$ ~& |4 A0 \. A, N1 d# y  ^- Y, n+ I
    private PersonDaopersonDao;//用于字段上, ~+ d! f- @; X$ Q; M: K0 E% u
    @Autowired  I7 T' L7 z( o+ `5 o/ m
    public voidsetOrderDao(OrderDao orderDao){//用于setter方法上- G3 Q( j6 a% H) Y3 z& F
    this.orderDao=orerDao;
    * |* X: G% D+ e$ }1 k1 Z}! e; E4 j7 U% c. C- S2 m$ t3 X
    @Autowired注解是按类型装配依赖对象,默认情况下,它要求依赖对象必须存在,如果允许null值,可以设置它required的属性为false,如果我们想使用按名称装配,可以结合@Qualifier注解一起使用,如下:/ a1 p. @' ^/ g
    @Autowired@Qualifier("personDaoBean")
    ) t. K9 E4 R) Q0 M# t7 Aprivate PersonDaopersonDao;% R* O, p, h3 b, S$ w
    @Resource注解和@Autowired一样,也可以标注在字段或属性的setter方法上,但他默认按名称装配,名称可以通过@Resource的name属性执行,如果没有指定name属性,当注解标注在字段上,即默认取字段的名称作为bean名称寻找依赖对象,当注解标注在属性的setter方法上,即默认取属性名作为bean名称寻找依赖对象- p9 ^( m; S5 X& K5 @
    @Resource(name="personDaoBean")
    & H* }- A" a  [private PersonDaopersonDao;//用于字段上
    / o" |) S0 C; O. Y1 @注意:如果没有指定name属性,并且按照默认的名称仍然找不到依赖对象时,@Resoruce注解会回退到按类型装配,但一旦指定了name属性,就只能按名称装配了.拿personDao与<bean id=" ".../>中的id是否相同,相同就找到,属性的setter方法也不用写,既方便,又优雅.4 O* S* x* g' _% @1 e$ t- N
    同时@Resource是j2ee提供的注解(建议使用),@Autowired是spring框架提供的.
    " X; N) k# Z  ]5 X3 r' F
    ( ^4 W: A1 J! J0 I) X- l3.     Spring的三种实例化Bean的方式# N( o# ]! T1 [0 d1 }4 c- s
    (1).三种实例化bean的方式:
    3 Y' w/ g6 y( p# u, ~1 z2 _第一种:<bean id="orderService"class="cn.itcast.OrderServiceBean/>4 p4 J7 k' C# i1 _: e
    第二种:使用静态工厂方法实例化:
    " p3 c5 K; M9 }) o% E" R<bean id="personService" class="cn.itcast.service.OrderFactory"factory-method="createOrder"/>
    ( k& C" r7 I. t: {7 z$ ypublic class OrderFactory{! v: W5 q7 F2 b6 I* k( g0 V
    public static OrderServiceBeancreateOrder(){( |1 c$ J7 K! f- ^2 h# E
    return new OrderServiceBean();
    ) A/ v' ^$ G4 k- ^0 r}
    7 W( @# @% e2 W: j5 K}& [0 T( y/ e" N" ~- o
    第三种:使用实例工厂方法实例化:
    ( d; [; t- n: u8 V. h+ n# ~<beanid="personServiceFactory" class="cn.itcast.service.OrderFactory"/>% n" F; V2 O' j& M, d# n
    <bean id="personService"factory-bean="personServiceFactory"factory-method="createOrder"/>- D8 ]7 t8 f" y, E3 b: `1 |+ |
    public class OrderFactory{
    ( a( t8 u" p& H! W+ G) A6 Bpublic OrderServiceBean createOrder(){
    , i. C1 M& `8 |1 ]3 {3 @6 N4 Lreturn new OrderServiceBean();
    7 I, K8 ~7 h! g1 E( [7 Y}9 m! S: g3 M; C: C% _
    }
    / Z8 Y% ~) ?; b/ ]2 r4 o/ [
    ; b; k' t* e/ `$ c5 ^& e4.     Spring管理的Bean的生命周期
    , H7 H) c2 E, _1 \4 x(1).Bean实例化是在Spring容器实例化时进行的,但是这是Singleton作用域中,实例化的时机是可以更改的,lazy-init="true"延迟初始化,即更改为调用getBean()方法时进行初始化.同时也可以在配置文件中设置所有的bean延迟初始化,其实这个标签是不建议使用的.
    % ]7 w0 u3 M& V$ d" L " o9 R3 D# G: Q- G5 f
    (2).当把作用域改成Prototype时,Bean实例化是在调用getBean()方法进行的
    0 ^! i4 J/ o6 Y( L  }( Y ' K; [4 `0 B+ \  z/ w
    (3).可以指定一个初始化方法:init-method="";即在bean实例化后执行的初始化方法.如数据库的连接.容器通过反射技术调用的,同时还需要进行资源的释放.destory-method="";即在bean被销毁时执行的方法.
    2 M9 Q) x* \/ ]/ q& C5 W 5 G) i! `0 g8 Y6 v2 f( `. S1 l
    (4).关闭Spring容器:ctx.close()方法,bean此时被销毁了4 l; r% }, D; a9 I
    5 Y" J* z  Y$ s  r. I6 [
    5.Spring自动扫描和管理bean
    8 @8 E. S) C$ ~4 P9 U5 Y通过在classpath自动扫描方式把组件纳入spring容器中管理,前面的例子我们都是使用XML的bean定义来配置组件,在一个稍大的项目中,通常会有上百个组件,如果这些组件采用xml的bean定义来配置,显然会增加配置文件的体积,查找及维护起来也不太方便,spring2.5为我们引入了组件自动扫描机制,它可以在类路径底下寻找标注了@Componet、@Service、@Controller、@Reponsitory注解的类,并把这些类纳入进spring容器中管理,它的作用和在XML文件中使用bean节点配置组件是一样的,要使用自动扫描机制,我们需要打开以下配置信息:
    3 Q# F, ]" @, w: B5 B! t<beansxmln="http://www.springframework.org/schema/beans") y$ E' s+ O4 l0 u: M+ @
    ..... X6 g# m6 H4 ^. q! {1 v- Z
    ....
    1 i# X3 c7 w2 i& ]</beans>: c# V2 g: |6 r, b3 {) v
    其中base-package为需要扫描的包(含子包)
    * O& T/ _; L( p# ]/ X7 l& u@Service用于标注业务层组件、@Controller用于标注控制层组件(如struts中的action)、@Repository用于标注数据访问组件,即Dao组件,而@Component泛指组件,当组件不好归类的时候,我们可以使用这个注解进行标注.同时也可以通过注解@Scope("prototype")修改bean的作用域.@Service("personService")中的名称必须和bean的名称相同,只是开头字母变成小写了.当然这是默认设置,可以修改的.' u6 X4 X) s/ [7 Q3 _7 q6 k  _
    可以使用注解的方式指定初始化方法,在初始化方法init()上添加注解@PostConstruct,同样可以指定销毁方法destroy(),注解为:@PreDestroy5 g( @% h1 j# I! S+ A
    这种扫描方式是很方便的,很多人都采用9 ~1 \$ |6 I: t& a" s* S- z
    即前面所说的功能都使用注解进行操作: c+ A( `( h3 {

    7 a( }3 @) B  l1 h& \- G6.  SSH整合开发. F+ F" N1 Y) E. w/ U( {/ w
    hibernate核心安装包下的:# [* ]3 A$ |5 s3 G5 e
    hibernate3.jar
    0 e, g% K5 W" i- x6 B* clib\required\*.jar
    9 C( x5 X: B- n" T+ a8 w1 X. Qlib\optional\ehcache-1.2.3.jar5 q: b: n$ G* z3 q3 E; Z
    hibernate注解安装包下的
    ! K% B/ u$ v3 [% `% F; G* mlib\test\slf4j-log4j12.jar5 c6 B4 F/ `$ e* M
    $ b4 Z, K8 C* y: K$ b0 |; R
    Spring安装包下的5 m9 R. ]( E( s
    dist\spring.jar9 e1 Y/ t3 W' q- ?1 N5 L. Y: l
    dist\modules\spring-webmvc-struts.jar1 U/ [. }" j$ q* S0 A6 ]. k
    lib\jakarta-commons\commons-logging.jar、commons-dbcp.jar、commons-pool.jar
    ' [+ G8 S9 @+ X! Rlib\aspectj\aspectjweaver.jar、aspectjrt.jar
    4 h# ~* N# e4 y% V( Alib\cglib\cglib-nodep-2.1.3.jar
    7 z  f0 M# l" T: @# Vlib\j2ee\common-annotations.jar% r  z0 y, L3 ~
    lib\log4j-1.2.14.jar7 s/ a8 y# V$ Y2 U
    - }/ e8 `4 ?5 ~/ q/ d) L4 ^( C
    Struts:下载struts-1.3.8-lib.zip,需要使用到解压目录下的所有jar,建议把jstl-1.0.2.jar和standard-1.0.2.jar更换为1.1版本,Spring中已经存在一个antlr-2.7.6.jar,所以把struts中的antlr-2.7.2.jar删除,避免jar冲突' p# p" ^# a9 z5 P- Z' ]6 D
    数据库驱动jar
    * M5 z& l0 W3 G$ i首先整合struts和hibernate然后再整合spring
    ) j; L9 e3 e3 d" q5 i
    9 T- d4 @" c+ m0 J7.编码解析@Resource注解的实现原理
    % X2 c2 l+ [. m2 W0 K3 n. {9 m(1).新建一个注解
    ) {4 Y1 H, j  R@Retention(RetentionPolicy.RUNTIME)& S; |# F8 c2 s" d# r
    @Target({ElementType.Field,ElementType.METHOD}): x+ E+ D. n- s: k* k
    public @interface ItcastResource{
    ( a  i$ G/ j1 N* V" t9 Y9 `String name() default "";4 |8 Q( @- d5 w/ A* O
    }# f+ t5 F7 O. o/ _  ?9 Y0 i& m+ N
    当然还要编写一个注解处理器.5 r% @8 ~( S' z% M9 j
    private voidannotationInject(){
    ) U9 b+ N* ?& b# R5 n) k+ H. O首先循环所有的bean对象,判断其是否使用了注解) c: i4 X* q# L7 x& {5 E
    如果使用了注解,就进行属性注入.
    4 s) X3 `5 O: m# Q& g9 z) x}
    ! Y7 S' o7 n+ [% e: Q
    6 C9 s+ A9 v4 P, L! O8.编码解析Spring依赖注入的原理
    . }' d! x9 _7 U5 U/ {8 y(1).基本类型对象注入:
    ( }, K8 y+ [5 ]. s<bean id="orderService"class="cn.itcast.service.OrderServiceBean">
    , P  G8 e& Z& h+ n/ M# B5 K- @1 M# R<constructor-arg index="0"type="java.lang.String" value="xxx"/>构造器注入
    % h7 M& q; A% q/ p0 [' r/ h<property name="name"value="zhao"/>//属性setter方法注入( L. T( f% [. \; a- G
    </bean># _5 @; o, T; z
    1 `! M5 G, w- h: W  h' G% r
    (2).注入其他bean:: v' j3 [& h. v. L+ _- G  f& n
    方式一:
    , s* K  k/ I  E/ h<bean id="orderDao"class="cn.itcast.service.OrderDaoBean"/>5 Y' ?4 C- f! v9 T, p' B
    <bean id="orderService"class="cn.itcast.service.OrderServiceBean">
    8 q0 k6 r0 ?3 d6 j# E; `<property name="orderDao"ref="orderDao"/>ref是指被注入的对象personDao7 J) A- X2 @0 ?! j% O
    </bean>
    ' {) f9 p' @* w方式二:(使用内部bean,但该bean不能被其他bean使用)- \3 d. A) U; A0 X
    <bean id="orderService"class="cn.itcast.service.OrderServiceBean">  _9 d8 I3 u9 o; U4 a& ~0 f- H% X% L. h
    <property name="orderDao">
    3 ]) M- G1 u. P) J' [<beanclass="cn.itcast.service.OrderDaoBean"/>2 x3 g0 ^, M, R- t6 q3 C9 A6 K( i* d
    </property>- j8 t; e. w3 r5 m1 Q: {2 x8 A
    </bean># Q1 T+ ]6 e( U! ?
    9 k) S4 {% C. d
    (3).依赖注入的内部原理( d  c3 h4 U, t4 X

    ( }% b% W9 ]; e0 S2 Z9.编码解析Spring装配基本属性原理
    8 m1 l. x. M! Z2 n" ]# [也可以对基本类型进行注入,类型转换.- A+ l( n. m4 I! b$ G: H

    / [" U  _& ?! ]) }; h3 n6 @% g10.编码剖析Spring管理bean的原理
    / A9 z" O4 G8 n+ v6 U# ]6 d- }使用dom4j读取spring配置文件,使用反射技术即可
    4 T! W( J) Z; N2 S: R8 {
    " D) W% U( ~- ^. t11.搭建和配置Spring与JDBC整合的环境
      b' N( z" b7 Q(1).配置数据源:
    - y8 ?/ O* F+ Q<bean* X; E3 H3 Y: y2 [
    id="dataSource"/ i* e4 \) o( c7 W
    class="org.apache.commons.dbcp.BasicDataSource"destroy-method="close">$ W" k, K  j3 T: p0 _- l
    apache的数据源:BasicDataSource
    ! }; @. Y/ S5 M: q1 n, _<propertyname="driverClassName" value="org.gjt.mm.mysql.Driver"/>8 D/ M# G' [6 x( V" p
    <property3 y& J$ Q% v3 [3 ^0 F5 ^
    name="url" value="jdbc:mysql://localhost:3306/itcast?useUnicode=true&amp:characterEncoding=UTF-8"/>8 S4 ?* ?  I. g& ~/ r/ Y
    <property name="username"value="root"/># ^3 }6 Q" A2 k
    <property name="password"value="123456"/>
    ' r* g9 ?" \8 W/ q- p+ b2 {$ a<property name="initialSIze"value="1"/>连接池启动时的初始值
    " }1 ]: t2 b7 h- T7 I" S7 Q<property name="maxActive"value="500"/>连接池的最大值) l5 K$ y$ X8 n, d5 S( A
    <property name="maxIdle"value="2"/>最大空闲值6 B2 r8 r& o  a) m8 Q5 P4 z3 B
    <property name="minIdle"value="1"/>最小空闲值3 w0 E1 t  a/ R& z4 z# B
    </bean>3 f8 Z& r% n# Y8 R  n
    配置事务:/ y9 @6 A5 S- e
    采用注解的方式:
    # T0 |$ {6 L7 x7 C4 o<bean
    5 Z5 |- M6 y: r" J0 m" F8 R8 [id="txManager"class="org.springframework.jdbc.datasoruce.DataSourceTransactionManager">
    ( A" M' C( x" t' C<property name="dataSource"ref="dataSource"/>9 Z& d6 Z) I  ]* c4 {
    </bean>
    & A* S% U8 ?- H! ~$ V6 ~
    4 [0 X+ r1 q- w7 f<ts:annotation-driventransaction-manager="txManage"/>
    ' K0 S. ^2 S4 e0 W6 G3 c6 m) r7 R
    + K" j% v- k5 W' @12.搭建与测试spring的开发环境% _  `& r; z) z9 g. ]9 I1 X
    (1).dist\spring.jar;+ ^* d, B! z9 X( O
    lib\jakarta-commons\commons-logging.jar: b& S' T3 s' R
    以上这两个包是必须的
    . A$ H/ w" ?1 @2 B/ d. i  M3 G4 rlib\aspectj\aspectjweaver.jar和aspectjrt.jar8 y3 Z4 T: A7 [
    lib\cglib\cglib-nodep-2.1.3.jar
    " R% a/ ~! o1 E" l0 c以上这两个包是用于切面编程(AOP)
    1 {1 v3 r5 A9 n; a, m- y& n/ h% ilib\j2ee\common-annotations.jar
    , s- T1 s$ U  q' ]以上的这个包是JSR-250中的注解: D: G8 q8 [# a* [; B; n
    % _/ _8 a2 s# @2 r7 D+ D
    (2).Spring项目既可以在j2se中也可以在j2ee中
    1 p6 Y3 i) g9 @" }/ |( d0 y1 D / L5 b4 Q2 a9 G
    (3).spring的配置文件模板可以从spring的参考手册或spring的例子中得到,配置文件的取名可以任意,文件可以存放在任何目录下,但考虑到通用性,一般放在类路径下4 a- P# t1 f$ n2 D' n0 y0 @

    2 ~+ A/ l) P8 p/ _0 ]' b(4).实例化spring容器常用方式:8 [! D% a6 J1 \6 V1 h
    第一种方式:在类路径下寻找配置文件来实例化容器
    7 D# f- c5 ~+ v6 a1 mApplicationContextctx=new ClassPathXmlApplicationContext(new String[]{"beans.xml"});
    1 `2 q  h! y8 g' E4 U; {+ y第二种方式:在文件系统路径下寻找配置文件来实例化容器
    6 H$ u% c; G" ?7 CApplicationContextctx=new FileSystemXmlApplicationContext(newString[]{"d:\\beans.xml"});//将路径写死了,通用性不好,不建议使用.1 X7 F: B, M, M" {% d" z
    spring的配置文件可以指定多个,可以通过String数组传入# a6 h. ~% o9 S3 S) _( J9 d3 ~

    - X( G2 w. {0 u7 [(5).IOC:控制反转:7 ?( Q. k7 [+ z1 N+ e
    public class PersonServiceBean{
    * G3 i- O6 P6 m3 F0 l$ ?private PersonDao personDao = newPersonDaoBean();
    6 C! A, r0 P% b! [- [public void save(Person person){
    * I- Q- E. e& C& ypersonDao.save(person);0 P) d) j4 U5 q2 w) u' Q) K
    }
    . j/ T4 R# ~+ E3 ?0 A/ e}/ J: V- M6 I% [: ^, H$ J
    PersonDaoBean是在应用内部创建及维护的,所谓控制反转就是应用本身不负责依赖对象的创建及维护,依赖对象的创建及维护是由外部容器负责的,这样控制权就由应用转移到外部容器,控制权的转移就是所谓反转.& M0 T) W1 [6 O+ F3 Q& K0 e9 B: z
    ; q. [! V5 G" ]* Y
    (6).建立一个业务bean为PersonServiceBean,放在cn.itcast.service.impl包中.面向接口编程,进行解耦,怎么将业务bean交给spring管理,需要在beans.xml配置文件中:<bean id="" name=""class=""></bean>,其中id,name都是bean的名称,但是id不能使用特殊字符,id本身就属于XML属性的,如:"/sfs/"就会出错,但是name不会出错,class属性是bean类的路径,需要操纵bean的时候,只需到spring容器中取出bean进行操作,而不需要实例化一个bean了:
    5 N. @9 S, k7 I9 u2 T7 UApplicationContext ctx=newClassPathXmlApplicationContext(new String[]{"beans.xml"});) P" J+ r& L; j% C" I
    PersonServiceBeanpersonService=(PersonServiceBean)ctx.getBean("personService");
    : F% ~2 H  a6 a9 ^" ipersonService.save();! Y' ~: g" E3 B0 z. F0 r! \

    4 S/ g+ _( B. c2 S# a& X% V8 z(7).当在配置文件中没有标签的提示信息,需要手动添加schema文件,方法如下:windows->preferences->myeclipse->filesand editors->xml->xmlcatalog,点击添加,在出现的窗口中的Key Type中选择URL,在location中选"File System",然后在spring解压目录的dist/resources目录中选择spring-beans-2.5.xsd,回到设置窗口的时候不要急着关闭窗口,应该把窗口的Key Type改为Schema location,Key改为http://www.springframework.org/schema/beans/spring-beans-2.5.xsd' v+ o. V6 ?* D5 \, d3 [

    0 w5 d! f1 F' U, ]13.配置Spring管理的bean的作用域+ j3 p# X. s( f! b
    (1).Singleton:默认情况下,bean是单例模式的,在每个springIoc容器中一个bean定义只有一个对象实例,默认情况下会在容器启动时初始化bean,但是我们可以指定Bean节点的lazy-init="true"来延迟初始化bean,这时候,只有第一次获取bean会才初始化bean,如:<bean id="xxx"class="cn.itcast.OrderServiceBean" lazy-init="true"/>$ ?! s7 S) o9 P- q, T$ |
    如果想对所有bean都应用延迟初始化,可以在根节点beans设置default-lazy-init="true",如下:<beans default-lazy-init="true">
    . F4 ]% o# G% k: j* U& FPrototype:每次从容器获取bean都是新的对象& {2 E8 I3 O) Z( a5 K9 d5 e% L0 ^
    Request:在request的域中4 P* ?, d7 m5 L- x
    Session:在Session的域中$ @6 H- k8 m4 U6 T4 C
    Global Session:在全局的Session的域中
    + B( B  e& ?8 U0 P4 p% N8 p% J 1 Q: a; @( i4 f- T% i6 N, A( X
    14.全面阐释Spring及其各项功能
    ) M# C9 q5 w  @& z. y0 `0 g% z(1).Spring是一个开源的控制反转(Inversion of control,IoC)和面向切面(AOP)的容器框架,它的主要目的是简化企业开发6 n$ H1 d* G3 L' t
    ) a1 ]' ~' r5 T5 {% N9 n* ]+ {5 a
    (2).IOC:控制反转:& |/ k8 G; Q6 O  P$ |) q& e- ~
    public class PersonServiceBean{
    7 o4 i& i% m  I: t) a' Y# `+ Gprivate PersonDao personDao = newPersonDaoBean();% |" `4 _( r; F8 V( r$ x8 [
    public void save(Person person){( p' n9 T- E' r( W" F( n+ `6 [
    personDao.save(person);
    7 J! O+ x7 O3 G: S& \$ B& b% H9 x}
    + K) W- [* D! E6 T}% m5 v; T6 M- T- p" }7 Z) s9 |7 O2 l! X
    PersonDaoBean是在应用内部创建及维护的,所谓控制反转就是应用本身不负责依赖对象的创建及维护,依赖对象的创建及维护是由外部容器负责的,这样控制权就由应用转移到外部容器,控制权的转移就是所谓反转.PersonServiceBean是业务逻辑层类,PersonDaoBean是Dao层类,在业务逻辑层类中控制和创建PersonDaoBean,这就是应用本身创建和维护了,但是有了spring容器后,PersonDaoBean的创建和维护是在容器中进行的,不需要PersonServiceBean进行管理了,控制权进行的反转
    ( k1 t# L. b1 B4 H* U% D
    - P, O# L) \! o& {- U/ |6 `(3).依赖注入:当我们把依赖对象交给外部容器负责创建,那么PersonServiceBean类可以改成如下:
    1 M$ i$ e2 |' s4 b9 {! w* r4 |3 }  f; C. S3 cpublic class PersonServiceBean{
    8 }& v0 M) e  P! s7 U( Sprivate PersonDao personDao;2 b3 R. u. r" a5 D& t1 \! |3 i
    //通过构造器参数,让容器把创建好的依赖对象注入进PersonServiceBean,当然也可以使用setter方法进行注入
    ; F% u" N9 h2 e8 o1 W% H/ mpublic PersonServiceBean(PersonDaopersonDao){8 D, m' |( r: }
    this.personDao=personDao;* n7 j  x: O% g: j. w
    }4 q1 t( E! |9 Q0 M  U$ {2 P
    public void save(Person person){1 H0 W" S, h" W
    personDao.save(person);0 A+ H  p9 Q+ [: J' m. \+ V
    }$ R0 V- Y6 ?5 h6 ?7 N" r1 N4 q
    }8 p  G/ f/ U3 P! [! w) A( i
    所谓依赖注入就是指:在运行期,由外部容器动态的将依赖对象注入到组件中." C1 N; v7 y3 j8 ~
    ; w+ [7 ?; s; t) S  I
    (4).为何要使用spring:
    / v6 Y! Y2 Y$ _第一:降低组件之间的耦合度,实现软件各层之间的解耦:$ |8 V8 e6 C. M
    Controller->Service->Dao, O7 Y2 z2 p6 R* j% ~3 f
    第二:可以使用容器提供的众多服务,如:事务管理服务,消息服务等,当我们使用容器管理事务时,开发人员就不再需要手工控制事务,也不需要处理复杂的事务传播: a) \( G. [0 F0 a3 a7 V/ A
    第三:容器提供单例模式支持,开发人员不在需要自己编写实现代码
    9 y7 b+ a. W8 L4 R& T. I第四:容器提供了AOP技术,利用它很容易实现如权限拦截、运行期监控等功能
    9 {( o/ k4 T/ |/ i$ V第五:容器提供的众多辅助类,使用这些类能够加快应用的开发,如:JdbcTemplate,HibernateTemplate
    , g/ L; R! G* I3 G& O+ u第六:Spring对于主流的应用框架提供了集成技术,如集成Hibernate、JPA、Struts等,这样更便于应用的开发." ^+ N7 n' I* U& z# k% [7 j! x3 r
    % C5 ?+ M$ D6 T* _$ ^/ {6 z: X
    (5).轻量级和重量级概念的划分,其实划分一个应用是否属于轻量级还是重量级,主要看他使用了多少服务,使用的服务越多,容器要为普通的java对象的工作就越多,必然会影响到应用的发布时间或者是运行性能,对于spring容器,它提供了很多服务,但是这些服务并不是默认为应用打开的,应用需要某种服务,还需要指明使用该服务,如果应用使用的服务很少,如:只是用了spring的核心服务,那么我们就可以认为此时应用属于轻量级的,如果应用使用了spring提供的大部分服务,这时应用就属于重量级,目前EJB容器就因为他默认为应用提供了EJB规范中所有的功能,所以他属于重量级+ i% u9 f8 A1 ~) R/ m. Z9 [: ?+ Q
    ( A1 D" z( y/ v5 h! E( c
    15.使用CGLIB实现AOP功能与AOP概念详解
    5 L5 y! W' w+ T; m: a(1).使用cglib架包,构建代理,不需要被代理的对象需要实现接口$ s! n) l" b8 O. t, t  k! R! O
    public class CGlibProxyFactory implementsMethodInterceptor{
    $ a/ v7 q% z) P7 N. n% h; I private Object targetObject;
    - W$ `& k6 _/ _: vpublic Object createProxyIntance(ObjecttargetObject){
    + ~! c: H% E$ A9 W/ `7 dthis.targetObject=targetObject;* H0 d* U( Z8 f% U7 q. y2 G0 u
    Enhancer enhancer = new Enhancer();
    # F6 v  W" d( U! l/ venhancer.setSuperclass(this.targetObject.getClass());2 I7 ?1 o( k9 G1 t7 r6 ?4 e8 k' U2 R
    enhancer.setCallback(this);
    5 S6 W7 ?& x* K0 b$ kreturn enhancer.create();
    9 s' }, a1 Q: x# l: F  L! \}+ N) m! ^( }2 v
    public Object intercep(Object proxy,Method method,Object[]args,MethodProxy methodProxy)throws Throwable{: j+ e* c; `# g% N
    returnmethodProxy.invoke(this.targetObject,args);' ?9 Y! U" S, T- Z0 \
    }
    8 Y3 _" }0 ]4 R! Q}
    , u6 A. X! l7 Y& zCGLIB可以生成目标类的子类,并重写父类非final修饰符的方法
    # p8 `  q9 ?: x& ]8 A ' Y# _* B" T$ ^. D$ K, W
    16.使用JDK中的Proxy技术实现AOP功能
    , G* z& a& J4 y; i: L1 p(1).使用在权限拦截上.7 A: X. u( Y. @4 a( K

    , u9 G$ D$ q9 u% t3 M6 Y(2).被代理对象必须实现接口
    1 A; E# F' s6 M2 d& Z6 z& n( Gpublic class JDKProxyFactory implementsInvocationHandler{3 b3 e% F( a2 e
    private Object targetObject;+ S4 y7 J+ T( B- p5 ~* U) O
    public Object createProxyIntance(ObjecttargetObject){
    0 d  F4 g5 @- zreturnProxy.newProxyInstance(this.targetObject.getClass().getClassLoader(),this.targetObject.getClass().getInterfaces(),this);7 l4 {& M8 Y/ e. Y+ W
    }
    2 x+ R  H  w9 v0 ^8 d4 u  ?" epublic Object invoke(Object proxy,Methodmethod,Object[]args) throws Throwable{//环绕通知
    , q! |5 j0 {+ c0 R4 n9 fPersonServiceBean bean=(PersonServiceBean)this.targetObject;
    0 ^" @0 _1 K" o, JObject result=null;3 e9 R& \0 x8 e4 T* f8 y- T
    if(bean.getUser()!=null){0 m0 L( D6 c+ U) S" |6 T# f( x% i
    //....advice(),调用方法前处理,叫做前置通知
    5 j$ G8 |. s+ i$ O. O' o/ D8 ]try{( q$ |  J$ h  e* \! t1 Q# ~- N
    result=method.invoke(targetObject,args);- ]6 P2 N5 `0 e
    //....afteradvice();调用方法后处理,叫做后置通知9 y% D1 l. {* a" u. t0 V, k
    }catch(RuntimeException e){
    ) q" d0 x9 r% e- V//....exceptionadvice();调用方法出现例外后处理,叫做例外通知* R; U/ J0 Y! P3 C
    }finally{
      E9 \: p2 K5 |) G$ Z) `0 t& s//....finallyadvice();调用方法最终都会处理,叫做最终通知  H+ r  q3 Y. s# y3 @1 j3 X
    }
    ( q0 \( k4 |- i0 M! @}  O  {  N6 b" x' U& F
    return result;$ L; G) u# X0 E# u+ z+ t, o
    }# m" |( v; E6 d7 @7 V# o2 Y' u
    }& y( G5 k1 Q" V
    整个invoke方法叫做环绕通知.
    0 c# k$ c# W1 { 9 i8 h$ J9 Q: {+ |
    (3).AOP中的概念:" n5 Q' [( `1 K* B
    Aspect(切面):指横切性关注点的抽象即为切面,它与类相似,只是两者的关注点不一样,类是对物体特征的抽象,而切面是横切性关注点的抽象。* ^7 R8 N" P! O* u7 C
    JoinPoint(连接点):所谓连接点是指那些被拦截到的点,在spring中,这些点指的是方法,因为spring只支持方法类型的连接点,实际上joinpoint还可以是field或类构造器
    6 z. r7 h0 }0 P* BPointcut:切入点:所谓切入点是指我们要对那些joinpoint进行拦截的定义. P: `* u, c1 e3 d) c
    Advice(通知):所谓通知是指拦截到joinpoint之后所要做的事情就是通知,通知分为前置通知,后置通知,异常通知,最终通知,环绕通知, E# u- `8 ~9 M  V$ x: q
    Target(目标对象):代理的目标对象
    ( l. u8 \5 {4 \; \% h5 R8 fWeave(织入):指将aspect应用到target对象并导致proxy对象创建的过程称之为织入
    + h: t! L4 F- fIntroduction(引入):在不修改类代码的前提下,Introduction可以在运行期为类动态的添加一些方法或Field( `0 p& X9 [' ?% m3 J: }% \& c

    + m+ Z0 I  `5 q. ~$ ^9 D4 G9 ?& p17.使用Spring的注解方式实现AOP
    7 G  h1 @' n) u2 D1 t& _. p(1).要进行AOP编程,首先我们要在Spring的配置文件中引入AOP命名空间:# l6 o, N! a& f3 @
    <beansxmlns="http://www.springframework.org/schema/beans"5 H+ c0 @. N# |( \, y
    </beans>" W& ?) M" S' H, Q
    Spring提供了两种切面使用方式,实际工作中我们可以选用其中一种:一种是基于XML配置方式进行AOP开发,另外一种是基于注解方式进行AOP开发
    9 C6 `/ I% [! k* Q5 P$ B. W ' w; w; U; _4 s+ \" ~( ~! A
    (2).基于注解方式声明切面:首先启动对@AspectJ注解的支持:9 H' }9 W0 x$ ~* g  Z# Y
    <aop:aspectj-autoproxy/>
    ! Q! m; S7 V+ k& z5 Z; K* V: M<bean id="orderservice"class="cn.itcast.service.OrderServiceBean"/>
    5 t% j+ v: D, }' b" [' ?) U* w<bean id="log"class="cn.itcast.service.LogPrint"/>9 K2 u( S% i) w+ C7 _' V* b' V& q4 `
    </beans>
    5 d7 }/ m6 K/ U定义一个切面类(MyInterceptor):
    & |  ]* v  `  y3 R. X3 Z@Aspect/ ]8 ?: w# f! I5 u$ n+ p
    public class MyInterceptor{
    ! c5 l$ J' W- V* B@Pointcut("execution(*cn.itcast.service..*.*(..))* A5 S2 x8 G5 o1 y6 V/ F& I
    //定义切入点,通配符:*指的是任何返回类型,..是指子包下也进行拦截,*:指的是拦截所有类,*:指定的是所有方法,..是指任意参数
    ' w9 u7 l0 Q9 D4 d- w, ~  sprivate void anyMethod(){}//声明一个切入点4 y" _8 E$ |; T# D8 ]4 a
    @Before("anyMethod()")//定义一个前置通知,名称为切入点名称- w% u% N+ k3 x# }5 W$ [9 J7 X2 i& R
    public void doAccessCheck(){5 ^# t6 `( K3 e7 K* Y# K3 l& p
    System.out.println("前置通知");/ _, x/ _% ]& a, @# ^3 k# R* k/ l
    }
    + h  p9 g4 m( g1 L3 h+ b@AfterReturning("anyMethod()")//后置通知
    9 c: v6 y' \! apublic void doAfterReturning(){* V* d0 h1 b0 f, u' i
    System.out.println("后置通知");
    . P( W9 g( ]4 [5 @5 i}
    , F+ ^# [& @1 h; z" a& p@After("anyMethod()")//最终通知
    7 r8 S$ ?8 J5 H, W5 w0 [! e. cpublic void doAfter(){
    " D, T) K& E4 t" X/ [1 S  y. _System.out.println("最终通知");$ y5 ]! f1 ~( Z( \% Q: e
    }
    3 d+ n& S' o. a; _1 R0 g@AfterThrowing("anyMethod()");//例外通知
    7 O1 S0 `. T+ p" W, ^) Hpublic void doAfterThrowing(){
    6 W" o$ p* f  G6 @; j3 x. A2 p0 kSystem.out.println("例外通知");4 J. H7 u' l5 @2 t
    }% E8 v/ |$ D/ V+ u( W0 P
    @Around("anyMethod()")//环绕通知* Q; H6 J4 n+ K3 C1 r/ {% L9 L6 e
    public ObjectdoBasicProfiling(ProceedingJoinPoint pjp)throws Throwable{: C) S: p! @# [
    System.out.println("进入方法");% [' \! k, p- ]+ i" u3 I4 o0 {$ i
    if(){//判断用户是否有权限,有权限就执行该方法.. A  N7 J& n/ B: U. I9 I& d& I
    Object result = pjp.proceed();' y* ^9 N+ \9 \. t: T4 \8 l8 w  ^
    System.out.println("退出方法");( n& {1 ^6 q2 P- W; R
    }else{; n0 }4 l- z5 q: X* I
    }* y+ ^2 j0 j) ]/ \% {- }) X
    return result;/ {* N7 e) \7 h' e6 t; O" a' L
    }) ]3 [( h: @7 z/ W$ `5 B6 h- S
    . y) x9 @' O) q+ v+ I$ n
    }! U) J: M+ B+ x0 F
    所以当出现例外通知时,后置通知是不执行的,即它们两者肯定有一个不执行,一个执行.0 d9 ]: a& r% v  }
    需要将切面类交给spring管理:基于XML配置管理或者自动扫描管理
    0 l$ T9 [! n0 p/ a& H% d/ y
    8 h1 Q# G) B0 M% ~1 c5 ^(3).想得到参数:
    $ ]( t. y- A3 H0 U/ Y@Before(anyMethod() &&args(userName)")//添加参数,只会拦截到对应的方法,即一个参数的方法.
    * {5 R$ ~$ s4 I" M+ D0 f9 spublic void doAccessCheck(String name){
    7 b' y; J3 B8 y4 M( u6 L- kSystem.out.println("前置通知");
    ) e- J0 Y5 K: g, o% R- w  [' |}8 |: Z) {6 E- X& R& r

    4 w2 b% ^7 ?- k9 o(4).想得到返回结果:
    ' G, H' z4 A" P( s@AfterReturning("anyMethod()",returning="result")//后置通知: d: @$ y+ }6 f; ^# F. d
    public void doAfterReturning(String result){
    / ~' F1 M$ A: W1 RSystem.out.println("后置通知");
      {8 a- S2 g+ [' h: G, kSystem.out.println(result);//打印返回结果' j8 Y8 D2 n, V' W
    }
    , R: i$ w" W( O0 Q* L+ t' A; u
    * |6 z  Y. S- t(5).得到异常(例外):# y! U2 L7 u- y8 e; U! P$ e8 @8 O
    @AfterThrowing("anyMethod()",throwing="e");//例外通知
    1 s7 b+ G* s( y' f8 spublic void doAfterThrowing(Exception e){
    . E; B/ s7 a3 Z# ]8 oSystem.out.println("例外通知");! Y3 c  i' k3 }1 \* c  u/ D
    System.out.println(e);//打印例外
    * ]7 o8 U% T6 ^. J. l2 `}
    ' M: x2 ~, F! L! o - V0 L, r. O2 ~* X5 t+ Q& R" x2 d3 a
    18.使用Spring配置文件实现AOP, w4 |3 T  h7 G
    (1).使用配置文件实现AOP,切面类只是个普通的类,其内部没有任何注解6 Z- Q6 P" s, |7 _
    public class MyInteceptor{" Y5 s6 l8 t9 `, j5 z" U
    public void doAccessCheck(){
    5 ?8 H# c1 h5 A9 j  mSystem.out.println("前置通知");
    - a( D2 v$ i' a: X}
    4 r" g. h' |  T0 q$ P. o7 spublic void doAfterReturning(){
    & B+ l) m2 X5 W% b4 R, X5 Q  fSystem.out.println("后置通知");, {7 x7 R% C9 `. S7 C* G9 r
    }& Q! Q3 K/ {' Q9 f% p
    public void doAfter(){/ d* M. a: a# o1 h
    System.out.println("最终通知");: B+ u% @& P/ a9 N1 J/ f
    }
    , U* D% q) B& Z5 dpublic void doAfterThrowing(){
    + D4 {( ?! z0 X* e; s. _System.out.println("例外通知");
    9 M( A+ [. Z" G}
    : L8 r' w: e1 K* k9 g, q( qpublic ObjectdoBasicProfiling(ProceedingJoinPoint pjp)throws Throwable{
    1 I" X& W& c* c: X% r5 U  e2 wSystem.out.println("进入方法");3 M7 y3 M, m7 N2 s4 M
    Object result=pjp.proceed();3 j; p  J( I: N7 y+ v* N8 p
    System.out.println("退出方法");
    6 ?% ^3 N+ I) v3 R/ c9 e* Ereturn result;- w9 V6 n- N5 l; x4 j: f8 X7 }6 O
    }1 w' b) _& [3 b  }8 F( q$ n, k& D
    }" S8 V% C7 X' T3 k: d

    5 e1 [% j, b8 Q% T. ?(2).基于XML配置方式声明切面
    3 |& T- n8 f1 }9 |6 @; n5 x<bean id="orderservice"class="cn.itcast.service.OrderServiceBean"/>
    / t- _4 b' X! n0 X' [" s8 I" j<bean id="log"class="cn.itcast.service.LogPrint"/>  S" z8 ^! g$ Q
    <aop:config>
    0 I# ^+ Z# @) J7 P1 L8 l' v# T<aop:aspect id="myaop"ref="log">$ K% R1 o6 `9 {( S
    <aop:pointcut id="mycut"expression="execution(* cn.itcast.service..*.*(..))"/>
    ! {9 S2 t, U  }% ~; O+ J<aop:beforepointcut-ref="mycut" method="doAccessCheck"/>! R% W+ Y4 T" K, C, o8 m9 z+ @
    <aop:after-returningpointcut-ref="mycut" method="doReturnCheck"/>, k$ H& s2 w% t9 C
    <aop:after-throwingpointcut-ref="mycut" method="doExceptionAction"/>6 O2 F5 [" U, u& I1 M
    <aop:afterpointcut-ref="mycut" method="doReleaseAction"/>
    % U9 m5 o: T, Q3 j# L; ^<aop:aroundpointcut-ref="mycut" method="doBasicProfiling"/>
    $ Q" y0 i- O" }. F( Y; H</aop:aspect>/ M$ f3 }4 `; G) Q' K' r
    </aop:config>' F1 k$ ?  _& Q( v) M" n3 _
    ' R6 h' Z  y. [$ c# b, g2 r) k
    (3).对于表达式expression的细节:# d) k, e  D6 l  U+ P. S
    返回值类型为String:execution(java.lang.Stringcn.itcast.service..*.*(..))8 \3 o, Y1 U0 a, c, k( O
    第一个参数为String:execution(java.lang.Stringcn.itcast.service..*.*(java.lang.String..))! a! r. U6 t+ b8 t1 F
    返回值类型为非void:execution(!void cn.itcast.service..*.*(..))" a6 b" _# W( G6 y+ e" @, M

    . V6 b; L5 J1 d: I* B1 C$ `19.使用Spring配置文件实现事务管理
    - x& H; c# w: p+ P/ s6 u(1).* P. k; e. a' f* T, `
    <aop:config>
    8 S) D3 T3 h6 J  b+ N- s" [* T& z<aop:pointcutid="transactionPointcut" expression="execution(*cn.itcast.service.*.*(..))"/>对指定的方法进行拦截
    $ Q6 R. _! `+ f, r<aop:advisoradvice-ref="txAdvice"pointcut-ref="transactionPointcut"/>
    2 C- C: Q: R# q1 N7 L( `7 t% `</aop:config>+ s* [6 w$ Y7 M
    <tx:adviceid="txAdvice" transaction-manager="txManager">
    7 V& i8 C/ O& ^  u<tx:attributes>
    " c/ v7 G. T% n% a, Y5 c7 {; S& P<tx:method
    . _4 e0 O+ _  Y/ l6 Pname="get*"read-only="true" 所有已get方法都不开启事务propagation="NOT_SUPPORTED"/>
    9 n  b3 Z# p  i% B5 p! }+ W<tx:methodname="*"/>
    3 M6 V8 G) R4 x  J/ O% Q</tx:attributes>
    , {4 R! @; i- p' x</tx:advice>: Y7 E7 Q! u/ J$ K# i

    : k: d" p" a) P# f$ e! X20.使用Spring注解方式管理事务与传播行为详解
    3 X4 U+ F9 d' }. F, ^(1).只有当遇到RuntimeException时,事务进行回滚.Spring开启的事务管理,当遇到运行期例外(unchecked),而(checked异常)是不进行事务的回滚的.
    * T2 H) g" a% ]6 A% x- O1 X 3 R0 U4 D) j, L
    (2).当然也可修改这种情况.把unckecked异常改成不会进行回滚了:@Transactional(noRollbackFor=RuntimeException.class),7 [( n# Z( Y5 M/ |% G

    : u/ y. g7 q3 _8 n! Z4 \(3).@Transactional(propagation=Propagation.NOT_SUPPORTED);关闭事务,不开启事务的.spring容器默认是打开事务的.当然还有其他一些值:(事务的传播行为)* n' b+ f7 W, h' @
    REQUIRED:业务方法需要在一个事务中运行,如果方法运行时,已经处在一个事务中,那么加入到该事务,否则为自己创建一个新的事务(容器的默认值)
    3 ?$ L$ W% c) _7 MNOT_SUPPORTED:声明方法不需要事务,如果方法没有关联到一个事务,容器不会为他开启事务,如果方法再一个事务中被调用,该事务会被挂起,在方法调用结束后,原先的事务便会恢复执行
    " A: X" z: a! C/ P  ~- V2 pREQUIRESNEW:属性声明不管是否存在事务,业务方法总会为自己发起一个新的事务,如果方法已经运行在一个事务中,则原有事务会被挂起,新的事务会被创建,直到方法执行结束,新事务才算结束,原先的事务才会恢复执行( M: [" q0 l) E# e
    MANDATORY:该属性指定业务方法只能在一个已经存在的事务中执行,业务方法不能发起自己的事务,如果业务方法再没有事务的环境下调用,容器就会抛出异常
    / j6 y0 t, a( C9 j# r# QSUPPORTS:这一事务属性声明,如果业务方法再某个事务范围内被调用,则方法成为该事务的一部分,如果业务方法再事务范围外被调用,则方法再没有事务的环境下执行$ a6 P6 `( C8 y% b7 `8 S
    Never:指定业务方法绝对不能在事务范围内执行,如果业务方法再某个事务中执行,容器会抛出异常,只有业务方法没有关联到任何事务,才能正常执行.
    1 t& N0 n, Z9 }NESTED:如果一个活动的事务存在,则运行在一个嵌套的事务中,如果没有活动事务,则按REQUIRED属性执行,它使用了一个单独的事务,这个事务拥有多个可以回滚的保存点,内部事务的回滚不会对外部事务造成影响,它只对DataSourceTransactionManager事务管理器起效.Savepoint savepoint=conn.setSavepoint();conn.rollback(savepoint);8 f( F( i: S( P3 ]' D' a. G
    # ^. R0 p% b/ ?) e! \# u
    (4).readOnly值为事务不能修改了.timeout是事务的超时时间,isolation数据库中的隔离级别.
    + q1 E- C5 q3 ], C. T/ K4 | . G! O; ~/ q6 x0 l7 X3 C- \8 M0 }, ~4 n
    (5).数据库系统提供了四种事务隔离级别供用户选择,不同的隔离级别采用不同的锁类型来实现,在四种隔离级别中,Serializable的隔离级别最高,Read Uncommited的隔离级别最低,大多数据库默认的隔离级别为Read Commited,如SQLServer,当然也有少部分数据库默认的隔离级别为Repeatable Read,如MySql
    - Q4 R" f5 i; Z0 f) u+ W) iRead Uncommited:读未提交数据(会出现脏读,不可重复读和幻读)$ H( C8 E3 Z( s: y' c/ `
    Read Commited:读已提交数据(会出现不可重复读和幻读)
    8 N0 O$ M5 a* M0 O( p9 @7 SRepeatable Read:可重复读(会出现幻读), t' @: N: X- L0 X- x, z% p
    Serializable:串行化
    * g' D. P  l. E" D; M( S; v# \+ y- X脏读:一个事务读取到另一个事务未提交的更新数据
    6 Z5 O" |0 T' G! [$ m  _* z不可重复读:在同一事务中,多次读取同一数据返回的结果有所不同,换句话说就是,后续读取可以读到另一事务已经提交的更新数据,相反,“可重复读”在同一事务中多次读取数据时,能够保证所读数据一样,也就是,后续读取不能读到另一事务已提交的更新数据.$ g. Y% Q2 @1 o% I; X( R
    幻读:一个事务读取到另一个事务已提交的insert数据.
    红红火火恍恍惚惚
    回复

    使用道具 举报

    快速回复
    您需要登录后才可以回帖 登录 | 立即注册

       

    关闭

    站长推荐上一条 /1 下一条

    发布主题 快速回复 返回列表 联系我们 官方QQ群 科帮网手机客户端
    快速回复 返回顶部 返回列表