我的日常

登录/注册
您现在的位置:论坛 盖世程序员(我猜到了开头 却没有猜到结局) 盖世程序员 > Java安全实现自动登陆功能
总共48086条微博

动态微博

查看: 7062|回复: 0

Java安全实现自动登陆功能

[复制链接]

326

主题

72

听众

999

金钱

实习版主

该用户从未签到

优秀版主

跳转到指定楼层
楼主
发表于 2015-06-02 10:44:46 |只看该作者 |倒序浏览
现在很多网站都有为用户保存登陆信息(即保存Cookie)的功能,当用户下一次进入网站时,可以帮助用户自动登陆,使网站显得更加友好。笔者通过研究ACEGI项目的自动登陆源码,编写了一个安全有效的实现两星期自动登陆功能的java工具类,下面是具体的实现流程和实现代码。  J0 |1 S) s  X5 n, s7 |

# a4 e9 f/ U( J' J) y5 r; @- c  先说一下流程:
0 B+ ]/ S) u) B& i5 |1 D
1 X3 h1 d0 b, _$ T  1.保存用户信息阶段:
) Q0 M9 n% W; ^+ G2 |' T+ t9 l" S% j/ C! a' p( e
  当用户登陆网站时,在登陆页面填写完用户名和密码后,如果用户在提交时还选择了“两星期内自动登陆”复选框,那么在后台程序中验证用户名和密码全都正确后,还要为用户保存这些信息,以便用户下一次可以直接进入网站;如果用户没有勾选“两星期内自动登陆”复选框,则不必为用户保存信息,那么用户在下一次登陆网站时仍需要填写用户名和密码。
3 [1 X$ `" X# ~9 b
7 A/ n" |; z8 N4 r  I; D# d; W; S  在保存用户信息阶段,主要的工作是对用户的信息进行加密并保存到客户端。加密用户的信息是较为繁琐的,大致上可分为以下几个步聚:6 z0 ]8 ?8 `7 M3 m: T9 J: Y+ T
" [! e4 j- Z  |/ G
   ① 得到用户名、经MD5加密后的用户密码、cookie有效时间(本文设置的是两星期,可根据自己需要修改)# L1 X; X# m) o. q9 U% C7 v
   ② 自定义的一个webKey,这个Key是我们为自己的网站定义的一个字符串常量,这个可根据自己需要随意设置; P& `+ n) c$ n. B
   ③ 将上两步得到的四个值得新连接成一个新的字符串,再进行MD5加密,这样就得到了一个MD5明文字符串6 Q! U% l2 L4 `) B, y- _
   ④ 将用户名、cookie有效时间、MD5明文字符串使用“:”间隔连接起来,再对这个连接后的新字符串进行Base64编码
9 J- t1 r, ~$ u6 E/ [( _   ⑤ 设置一个cookieName,将cookieName和上一步产生的Base64编码写入到客户端。/ o0 {! M, V& A

5 y7 a! V5 t8 }( j0 W' T/ A  2.读取用户信息:$ l2 @( s5 o+ G0 b9 o

9 g: M6 F6 i- e+ ]: |) J' J  其实弄明白了保存原理,读取及校验原理就很容易做了。读取和检验可以分为下面几个步骤:6 y' E# t( g4 A, F0 Q

1 _# }( A# @/ w8 M+ w4 R   ① 根据设置的cookieName,得到cookieValue,如果值为空,就不帮用户进行自动登陆;否则执行读取方法1 p7 N, K1 \) y) _. M% D  Y0 W
   ② 将cookieValue进行Base64解码,将取得的字符串以split(“:”)进行拆分,得到一个String数组cookieValues(此操作与保存阶段的第4步正好相反),这一步将得到三个值:
" c; |" m( Y7 _$ P7 ?, k. B, f, W
& Q: w8 _1 }3 \& m  \9 _cookieValues[0] ---- 用户名
( c* v3 i6 a& WcookieValues[1] ---- cookie有效时间
4 H2 W6 K" I. pcookieValues[2] ---- MD5明文字符串
7 t5 c" l* e: S$ q3 K$ l" b
- g: G, f$ h3 X* g; B0 A9 b   ③ 判断cookieValues的长度是否为3,如果不为3则进行错误处理。
, @, [- `& b0 M2 R6 G   ④ 如果长度等于3,取出第二个,即cookieValues[1],此时将会得到有效时间(long型),将有效时间与服务器系统当前时间比较,如果小于当前时间,则说明cookie过期,进行错误处理。/ h- b% j5 L% \9 Z% |, I; F) K
   ⑤ 如果cookie没有过期,就取cookieValues[0],这样就可以得到用户名了,然后去数据库按用户名查找用户。# }4 S* w: `6 @1 s
   ⑥ 如果上一步返回为空,进行错误处理。如果不为空,那么将会得到一个已经封装好用户信息的User实例对象user" w6 F' `+ ^( ^8 C: b* L
   ⑦ 取出实例对象user的用户名、密码、cookie有效时间(即cookieValues[1])、webKey,然后将四个值连接起来,然后进行MD5加密,这样做也会得到一个MD5明文字符串(此操作与保存阶段的第3步类似)
0 {7 }* Y& I% O  V8 w& d   ⑧ 将上一步得到MD5明文与cookieValues[2]进行equals比较,如果是false,进行错误处理;如果是true,则将user对象添加到session中,帮助用户完成自动登陆
' ?/ Z0 Q/ T2 S% k7 [6 M! l. ~' i2 I" o' I$ ?
  完整的代码,用途请参见注释0 M- z$ K$ ]/ Q7 D* [: M9 ?
CookieUtil.java; F: p: `) s$ H% O  B
& I1 ?4 q, b7 S2 B
处理cookie的工具类,包括读取,保存,清除三个主要方法。
5 J8 a# g# J- B7 q+ I, A; f8 ]
! Y' `/ @* H: Q. a0 _: G' D
  1. public class CookieUtil {
    2 s0 q: X/ S7 O! \( M: p; i4 ]. `$ b
  2. //保存cookie时的cookieName1 V0 r7 S" I; b2 c, O1 \
  3. private final static String cookieDomainName = “cn.itcast”;
    ) n" R, [3 W0 q2 M; G4 ^
  4. //加密cookie时的网站自定码
    4 N, A$ j: s+ S
  5. private final static String webKey = “itcast”; 7 G+ |4 M5 i9 y: r
  6. //设置cookie有效期是两个星期,根据需要自定义
    / x$ D5 Q! v" z6 P; [# ?
  7. private final static long cookieMaxAge = 60 * 60 * 24 * 7 * 2; ' w5 k( V/ h; e" H9 J( Y
  8. //保存Cookie到客户端--------------------------------------------------------------------------------------------------------
    8 F% W% M& ~9 U3 f( l1 A3 @  H' o
  9. //在CheckLogonServlet.java中被调用
    5 d! K" K: F5 _1 b% C& K9 X7 [
  10. //传递进来的user对象中封装了在登陆时填写的用户名与密码
    : F  A3 r5 m, t. R2 B5 H
  11. public static void saveCookie(User user, HttpServletResponse response) {/ H3 _7 i" A$ L$ b1 I

  12. 8 I; u2 g' g# X/ Q5 e& E+ `" J
  13. //cookie的有效期3 Q& a4 y) A8 B" h1 G. X; x
  14. long validTime = System.currentTimeMillis() + (cookieMaxAge * 1000); , h& b& A. h0 f' D8 Z6 e. |/ s/ }
  15. //MD5加密用户详细信息* K$ g- r: q2 N2 d" G
  16. String cookieValueWithMd5 =getMD5(user.getUserName() + ":" + user.getPassword()+ ":" + validTime + ":" + webKey); $ g' }% ?* @5 Q! c8 S3 b
  17. //将要被保存的完整的Cookie值
    1 Z+ ?" {4 n4 y& E7 v: s, z& S0 w
  18. String cookieValue = user.getUserName() + ":" + validTime + ":" + cookieValueWithMd5; 0 }/ I6 u* x% P) _4 \9 L
  19. //再一次对Cookie的值进行BASE64编码+ Q$ l$ x) ?8 @. I" K- X
  20. String cookieValueBase64 = new String(Base64.encode(cookieValue.getBytes()));
    5 _7 V1 \7 Z, q/ X8 s
  21. //开始保存Cookie! c# D# K" x' X0 r0 J
  22. Cookie cookie = new Cookie(cookieDomainName, cookieValueBase64);
    7 s. t- c7 d  {8 [  V% }
  23.   O0 k# d9 ?/ u# l& I2 ~* i
  24. //存两年(这个值应该大于或等于validTime)- I5 y: K( \! D9 H+ R+ ^, H9 E9 t
  25. cookie.setMaxAge(60 * 60 * 24 * 365 * 2);
    9 H9 i! J+ ~+ h( k
  26. //cookie有效路径是网站根目录
    ' L+ s) ?# M' Q4 ^! }" a* h
  27. cookie.setPath("/"); 6 p$ `% e; R* U$ a6 ^8 p* K: \5 _
  28. //向客户端写入1 Y% x+ J, W& i! u: l- ~
  29. response.addCookie(cookie);
    0 v4 w" `( ~' T7 r
  30. 2 c7 x+ ^& y0 J( ?; L% K( H6 N. o1 {
  31. }9 s) r+ g. L7 V- t3 O+ \
  32. 5 _5 ~7 e# w, @  B4 ?# B: k6 J% f
  33. //读取Cookie,自动完成登陆操作--------------------------------------------------------------------------------------------
    . p; O) B; k* r, Y: f

  34. ) }% n- g6 @4 K) J$ d6 \
  35. //在Filter程序中调用该方法,见AutoLogonFilter.java
    2 I' l. o7 g$ B) J* g+ o) W
  36. public static void readCookieAndLogon(HttpServletRequest request, HttpServletResponse response, 8 u, s, w" Z1 f% R% m
  37. FilterChain chain) throws IOException, ServletException,UnsupportedEncodingException{
    " _- ~# w4 I# d8 M7 D
  38. //根据cookieName取cookieValue
    : J7 O* R/ C$ G' N7 c
  39. Cookie cookies[] = request.getCookies();
    5 d5 w: A/ P, B' c$ f0 i4 y' ^" C
  40. String cookieValue = null;
    ' m  R- o  S  y8 g; |  o% e- A, Y( ^
  41. if(cookies!=null){/ \# m; s0 R; j0 e# @
  42. for(int i=0; i<cookies.length; i++){& B# ?6 \+ A2 s# a1 N) ~! ?+ [
  43. if (cookieDomainName.equals(cookies[i].getName())) {, ]: N1 `) K* h2 m
  44. cookieValue = cookies[i].getValue(); 3 k3 Z. A1 k7 x! X4 I2 N
  45. break;
    / Q  K8 ~; a, i& [
  46. }
    / i" U% S5 z) X: j( e; x
  47. }
    " b& c: U+ Q6 W( m* V: Z- a) Q) W' q
  48. }
    5 X. Z( F0 S  p3 A# q  q

  49. ; U$ m& F. X% ^  T
  50. + X& F4 l  A/ R8 ^0 t. k
  51. //如果cookieValue为空,返回,
    1 h9 u2 `! }% t& E

  52. - N! S; o  Z7 @3 {8 y% h
  53. if(cookieValue==null){$ l2 ~3 q! l# d' w, b& X: M
  54. - L( c. \: S7 w% k$ X4 @4 Y
  55. return;1 d/ r, d- W8 [: @3 d6 P  ^

  56. ; H1 D* l* w4 J+ H9 P5 \
  57. }
    * ?  i+ {2 `2 D# n
  58. - }7 h+ F8 `! x+ [; E5 ~, S' F
  59. //如果cookieValue不为空,才执行下面的代码
    + `) N  P* B- n2 \, \* K3 |6 _' s

  60. 8 n. {# U' l: a$ T& }6 G$ b
  61. //先得到的CookieValue进行Base64解码$ C- ~- Q9 j9 g* Q8 h' s
  62. + H9 w# y; L$ X. ]  {
  63. String cookieValueAfterDecode = new String (Base64.decode(cookieValue),"utf-8");
    . j3 [9 L' z, I/ _8 u
  64. 5 C! ~  L0 D8 j' v# }3 ?
  65. //对解码后的值进行分拆,得到一个数组,如果数组长度不为3,就是非法登陆
    ; f- w/ |. B0 p4 }# L& l
  66. String cookieValues[] = cookieValueAfterDecode.split(":"); 0 P, T9 ]3 k( z. n1 `" [
  67. if(cookieValues.length!=3){4 M  R8 ~- V, @% a( }& a5 J$ t" _
  68. response.setContentType("text/html; charset=utf-8");
    . N% a  {# r1 n. u  I6 n
  69. PrintWriter out = response.getWriter();
    # s# _/ L. k1 \
  70. out.println("你正在用非正常方式进入本站..."); - b; I+ u9 r7 c
  71. out.close(); 3 Y2 m' d7 y# b
  72. return; 1 c" H% r, Z8 R: z* K! R( B
  73. }
    ' G1 y' Q& {) P& g" g2 c
  74. //判断是否在有效期内,过期就删除Cookie2 w' [+ l7 Y& W: k
  75. long validTimeInCookie = new Long(cookieValues[1]);
    $ \) |7 ]3 C" E- f0 B# B
  76. if(validTimeInCookie < System.currentTimeMillis()){7 T4 G) b  H* P; ]+ ~
  77. //删除Cookie( C2 q5 Y1 s' H
  78. clearCookie(response); 6 t; u7 T0 }5 B" R8 T$ _
  79. response.setContentType("text/html; charset=utf-8");
    & t& L% m6 Z/ L
  80. PrintWriter out = response.getWriter(); # ]; Q. N0 ?7 U! s! ^
  81. out.println("<a href=’logon.jsp’>你的Cookie已经失效,请重新登陆</a>"); % L* u3 \! W8 `9 l" ~# V% y: g
  82. out.close();
    % D0 Y% p4 H/ ]. c
  83. return;
    9 z& y# A0 C: Z/ _$ g
  84. }  d' V$ M) c3 C2 K- I
  85. ! f+ _5 A6 Q8 `8 {# N% F; x3 [9 n
  86. //取出cookie中的用户名,并到数据库中检查这个用户名,! ]8 k/ F$ f# z! c3 C
  87. String username = cookieValues[0]; ) E+ _3 B2 X0 P# O
  88. //根据用户名到数据库中检查用户是否存在/ l4 `; {/ g: F
  89. UserDAO ud = DaoImplFactory.getInstance();
    3 b' e' P( w! M8 y( e0 z- U

  90. ; Q$ z* d4 I0 o
  91. User user = ud.selectUserByUsername(username);' e9 z$ K/ f# N+ l5 V. O
  92. ) E" c$ u* n9 Z8 \8 R7 p
  93. //如果user返回不为空,就取出密码,使用用户名+密码+有效时间+ webSiteKey进行MD5加密
    * J- X1 Y) g/ m( ^+ g) k
  94. if(user!=null){
    9 h# W. I( Y% F$ _3 A
  95. String md5ValueInCookie = cookieValues[2]; , ~) c0 r- Q/ M
  96. String md5ValueFromUser =getMD5(user.getUserName() + ":" + user.getPassword()+ ":" + validTimeInCookie + ":" + webKey);
    , k) C1 ~1 ]! B7 e
  97. //将结果与Cookie中的MD5码相比较,如果相同,写入Session,自动登陆成功,并继续用户请求
    # p2 w/ j* f6 J) r6 \/ t
  98. if(md5ValueFromUser.equals(md5ValueInCookie)){, F" P4 a$ ^. I, D3 z
  99. HttpSession session = request.getSession(true);
    6 l, p4 d9 I) \
  100. session.setAttribute("user", user);
    9 P- u  A1 |) j5 |+ c
  101. chain.doFilter(request, response); % v' d& z5 ^( K4 N6 }. o
  102. }
    6 K" \* _( s5 n( H1 s
  103. }else{
    9 V1 p' n/ V: @% S( C1 y
  104. //返回为空执行
    ) R! D: B  g! [% u
  105. response.setContentType("text/html; charset=utf-8"); % B1 h% B, V- j/ F
  106. PrintWriter out = response.getWriter(); % Z/ S9 \: v/ B( b  i* X$ _) h
  107. out.println("cookie验证错误!"); 9 C4 y+ h! Z) t0 B: {
  108. out.close();
    4 g; F9 j( x+ p" B" v8 T
  109. return;
    5 i. a( P$ X. t& ^: ~; R
  110. }( m! m( d+ r7 f
  111. }8 u0 o8 M5 K& |( C1 M5 d  _: z: H
  112. //用户注销时,清除Cookie,在需要时可随时调用------------------------------------------------------------. A' `+ \) f' w7 H5 ~4 d! ?& b
  113. public static void clearCookie( HttpServletResponse response){
    - E/ |7 `) w  Q9 p$ h" ^  |/ N) z
  114. Cookie cookie = new Cookie(cookieDomainName, null);
      t$ ]! T( R! J: I) [( F
  115. cookie.setMaxAge(0);
    ; f  O+ {: d& d# |7 q: |+ Y- m
  116. cookie.setPath("/");
    ! J  Y# Y; O& O. V/ v+ ]5 s7 O: U
  117. response.addCookie(cookie);
    % g$ P6 A. P. N
  118. }
    3 H8 D; [% [& P. F* s6 y
  119. //获取Cookie组合字符串的MD5码的字符串----------------------------------------------------------------------------
    $ O3 G5 h( t. ?0 ]; D1 n, M) R
  120. public static String getMD5(String value) {
    $ Z; s% W  _$ e& v9 V
  121. String result = null; + a: e: D- Q: ?& X1 R4 e7 r/ U
  122. try{
    + i! ^8 D! b8 R) p4 ~
  123. byte[] valueByte = value.getBytes(); ( N) K' [+ j1 X6 w% L1 i
  124. MessageDigest md = MessageDigest.getInstance("MD5");
    + E. N. I$ P2 {' v$ H
  125. md.update(valueByte); 0 j' ^/ }0 b% e5 J! {
  126. result = toHex(md.digest()); ' J3 d* [5 _% Y$ j/ N6 T. Z& R
  127. } catch (NoSuchAlgorithmException e2){7 G6 c$ `" R. C" V/ A* L" `
  128. e1.printStackTrace(); : i% v7 ~% Y& C' I+ E( K! r8 o
  129. }
    ( K! V, o/ f1 ^) w5 q  B) C
  130. return result; 0 p) o* h( y% y" a4 r4 s
  131. }4 z9 u, H, M& s% h( g5 }

  132. 1 @( a9 l# l$ r: u. a
  133. //将传递进来的字节数组转换成十六进制的字符串形式并返回0 \/ r1 I1 ?* |9 {+ H2 l: a
  134. 5 O, i5 q; h) |3 Y) I
  135. private static String toHex(byte[] buffer){
    - Y( [$ Y- |/ m3 p1 ~) r3 l, v
  136. StringBuffer sb = new StringBuffer(buffer.length * 2); 9 Q* F( u! X8 d5 Z; F
  137. for (int i = 0; i < buffer.length; i++){
    ' f7 ~% x4 D% f* _2 }: [
  138. sb.append(Character.forDigit((buffer[i] & 0xf0) >> 4, 16)); 6 [- a: t; o/ h1 s& i( H, y
  139. sb.append(Character.forDigit(buffer[i] & 0x0f, 16)); ) f  k7 @& U! B( q: \( L) @0 n
  140. }  I0 u! |2 J1 d7 E
  141. return sb.toString();
    6 H) m% W+ V, D, C7 k: U- w' e$ l. q
  142. }' L! [- s7 ?! ^
  143. }
    0 H" y. j# D7 t( Q8 c# o- h: f- X( m
  144. 7 I3 y1 r: a  X% R% T2 r( u6 W5 P
  145. 下面的是对CookieUtil工具类各方法的调用演示:! K  S, x) ^- t5 Q0 J9 s

  146. 7 r+ l7 j5 y, }
  147. User.java5 B& E6 s9 ~5 I0 z
  148. % s. {& N  A7 c- e
  149. 封装用户信息的JavaBean对象模型
    ' @% T# ^9 t8 H) H5 A! O$ H' D+ ^
  150. package com.itcast.bean;
    9 r3 H' }8 `; ?2 Q, [' O
  151. public class User {
    9 H9 K* F! _. Y0 y2 e
  152. private int id;
    / b( |0 X: `8 P2 H1 _( o
  153. private String userName; / h( H0 x4 G3 C' ]
  154. private String password;
    5 @& N0 v7 ^% S# t6 W3 l
  155. public String getPassword() {
    ' Q' J; l: K7 T5 A, u1 h

  156. # g. Y8 B  s6 Z, p2 c7 l
  157. return password;
    0 k9 _) J8 Q- h' ~

  158. 1 F% a% \7 P: |  x' i8 l
  159. }
    % o/ _! D1 N  i$ ^8 f
  160. public void setPassword(String password) {  o7 \! g3 ~% l& M! l

  161. 2 J# E- l( B7 p' p  I
  162. this.password = password;* `( I* B7 A/ ~; A0 r5 @* O/ v
  163. 1 [( j# E/ s! s  R2 C
  164. }
    + J" L/ o1 a. `' ?* c
  165. : z" u5 E3 K% w' ]% J
  166. public String getUserName() {
    * ^/ {5 \3 I* a) t
  167.   v& J, B) K  G5 o. z
  168. return userName;
    ( }& q, s  z9 v* Y0 W

  169. 5 t  i. `* n0 J2 s( i' K3 O  W
  170. }2 d8 b# D8 X8 R# E- Q! y- b5 F

  171. " F& x, m4 L' J" W
  172. public void setUserName(String userName) {
    " @9 [% Z4 u6 T/ j
  173. 1 p$ W, Q# p, O9 `+ ~$ w- E8 R+ V
  174. this.userName = userName;
    + i/ y3 v5 u9 @# t5 j* n
  175. ; k8 p/ ~- S7 X. E/ @
  176. }
    " U( m: L6 n3 ~9 n5 [) l$ k
  177. 9 r1 j; i; C: {# E& e+ e  F
  178. public int getId() {
    ; g' _* B: |- b# b; M
  179. $ t/ q0 _( y& _( \/ M" K
  180. return id;
    , d6 Z5 {+ J1 k5 F

  181. 9 q, }; p: O- ]9 t9 B! ~0 c3 |+ q
  182. }6 h- f1 E1 B9 Q0 {2 G  T. i" L

  183. + N$ k% l6 M! \" k2 o
  184. public void setId(int id) {2 h) g0 e+ ]2 ?0 ^  v
  185. / y8 `7 l. A4 e. [
  186. this.id = id;
    # Q* l' A0 I6 N1 [9 \; a* j
  187. * D- T2 g& B  X% W. g+ P# K% g
  188. }
    % F; |4 x1 D: D; G' d9 t4 {% Y. D8 j
  189. + O3 P$ @) H; H- `- Q1 f  z5 W
  190. }
复制代码

9 [% h: Y0 F- N5 G; \% ?, UAutoLogonFilter.java
5 O. A! T) G/ [/ G. o
9 m0 G5 {! `# H: T' }2 N, }+ S+ ?' Q过滤器程序,可在WEB-INF/web.xml中设置过滤规则,本文对过滤规则不作介绍,此程序主要作用是检查用户在上一次登陆时是否保存了Cookie,如果保存了,就处理Cookie信息,并帮助用户自动登陆
* x) r2 O/ d+ h. g' O# ^8 @8 B7 o5 [6 _- O7 W" B

1 a7 ?; ?* c) c  ?. t& m* u本程序主要调用了CookieUtil.java中的读取与自动登陆方法,即readCookieAndLogon方法4 o, ]1 N' q7 `2 r: h
  1. package cn.itcast.filter;+ c0 r( v: r" v. {- U
  2. $ p3 Y& S. d3 g+ o) o8 \/ m* S
  3. import java.io.IOException;
    : u& I3 x- i4 D6 S8 _" R0 Y
  4. import javax.servlet.Filter; / o! Z/ m& \' s: V8 u
  5. import javax.servlet.FilterChain;
    7 d+ E6 F* H* O+ D* o
  6. import javax.servlet.FilterConfig;
    8 h) Z9 r* S) \! P
  7. import javax.servlet.ServletException; - H  j' D; D5 _5 ^* c
  8. import javax.servlet.ServletRequest; % q6 m1 F* v( g" d
  9. import javax.servlet.ServletResponse;
    / i% [9 t4 A2 H
  10. import javax.servlet.http.Cookie; 0 ^' j% q; p; c( Z1 I
  11. import javax.servlet.http.HttpServletRequest; # l1 m- k" M2 u" t
  12. import javax.servlet.http.HttpServletResponse;
    1 z! b% u* k9 _
  13. import javax.servlet.http.HttpSession; 9 r% L) h8 _+ Z9 V
  14. import cn.itcast bean.User; 5 Y, H7 L$ K3 m  w% j. z' c$ n3 k) f
  15. import cn.itcast.util.CookieUtil;
    $ K9 Y  [3 Z/ t  L+ Q
  16.   s, V" t# x6 V3 r6 i
  17. public class AutoLogonFilter implements Filter {* f1 V3 V" T! r8 k
  18. ' x3 T: `4 \  z' ]3 h# E( d) g
  19. public void destroy() {, F& |6 ^6 r5 c0 x

  20. 8 J2 a: M  [' \9 L) G. h
  21. }" f, g" m0 Q. u! n; h8 R  N+ m

  22. + ?9 X; ~. G6 N# P9 u& f2 x  j
  23. //保存cookie时的cookieName,与CookieUtil.java中的设置相同9 ]! c9 |$ h6 V8 ]( B' \% |. _% C& |) v

  24. ) k7 m& Z( `# L8 y# O- m
  25. private final static String cookieDomainName = “cn.itcast”;
    / b2 s/ }3 M  a4 D. V# `1 y
  26.   g% n$ E& w3 @8 {% O
  27. % m# Y- D; i! v9 U2 |
  28. , {. n% d) K# n3 B& k! A" L& g) A
  29. public void doFilter(ServletRequest req, ServletResponse resp,) [- ~* D* s, ~- C# @% R6 R: D
  30. FilterChain chain) throws IOException, ServletException {9 m- r* {( l( M2 N0 L# L
  31. HttpServletRequest request = (HttpServletRequest)req; 8 X8 e2 I! q; l; d, P  t- P7 y% D0 G
  32. HttpServletResponse response = (HttpServletResponse)resp;
    ; X( [6 Y* N, W! |3 K
  33. HttpSession session = request.getSession(true); # v, v" O/ y: M4 d6 R" E- N7 b
  34. User user = (User)session.getAttribute("user");3 n8 _9 \( n7 O% K: ?  g

  35. . L0 L/ I$ b4 I+ E

  36. ) i+ Q; h1 g4 {) v: Q1 ^; ~- n$ G+ q
  37. //如果封装的user不为空,说明已经登陆,则继续执行用户的请求.下面的就不处理了
    1 i% `8 D7 S. e1 e

  38. # [) e8 {  ?" n( o* m  g
  39. if(user!=null){0 _1 U3 H" H0 D  Y( I3 W% h
  40. chain.doFilter(request,response);
      s2 Z, t1 L# j1 ]2 `
  41. return;
    7 w  G$ {- q* s# O! w& P5 u
  42. }' U5 q/ h- I3 i& j: O* z; f

  43. & ?& B( J) R5 u
  44. //user为空,说明用户还没有登陆,就尝试得到浏览器传送过来的Cookie: g2 n7 Q+ L8 C: y  w* D
  45. Cookie cookies[] = request.getCookies(); * V% O& e8 c7 o$ q# i: }' B3 S
  46. String cookieValue = null; ' N' c6 S$ F' S4 w, j" b* O
  47. if(cookies!=null){
    ) U- \3 w+ @4 r/ _# x
  48. for(int i=0; i<cookies.length; i++){- v8 `- ]7 o" B, ~) Y
  49. if (cookieDomainName.equals(cookies[i].getName())) {
      v3 Y) l# {; T! O- t
  50. cookieValue = cookies[i].getValue(); * N: Z3 x$ j5 F+ F* r" b7 q' n
  51. break; ( K* j2 k8 s, @4 u8 h
  52. }
    * [: G% v) d. _' l7 c$ `
  53. }0 S* m- Z) @. _- }* o0 s
  54. }+ m: c" _  t! T
  55. 7 U2 d4 X1 f, X/ f

  56. ( W5 k7 X/ s) r8 I: O. c
  57. //如果cookieValue为空,也继续执行用户请求
    % ]. N/ D0 ?) e; H
  58. if(cookieValue==null){
    ; m4 y) |4 c* R; w5 U# h
  59. chain.doFilter(request,response);   {/ S# g) [$ V/ W1 ]$ ^6 S
  60. return;   L, ]+ `* z$ R- R3 M: ~! X
  61. }5 ^% e( o* Q$ S) D/ _- l7 v

  62. . ~& J: w, g6 D" N1 @! U6 v
  63. //cookieValue不为空执行下面的方法,调用CookieUtil.java中的readCookieAndLogon方法
    $ r. f' B; Z4 ^& c2 k: M# L
  64. try{, \$ K& ?% W3 H8 V9 Y7 a
  65. CookieUtil.readCookieAndLogon(cookieValue, request, response, chain);
    9 t* K$ g+ Z3 _" W4 M8 j5 }
  66. }catch(Exception e){$ m/ S8 Y4 A/ e$ R( ?) x, x7 u* B
  67. e.printStackTrace();
    % m6 @( Q- u6 d% m! n
  68. }
    7 v) k/ v0 V% m/ o6 ~
  69. }+ p. z2 D( F/ Y6 v2 z* T
  70. 5 W! ~8 x- z+ |/ i$ `. u, H
  71. public void init(FilterConfig arg0) throws ServletException {& e6 W6 U: r" v9 F& w2 V2 e
  72. }
    $ r% f8 n2 w7 s1 L/ C6 o
  73. }
复制代码

" j# {8 j) b; F1 a$ oCheckLogonServlet.java
5 _- F4 {9 p+ O$ k5 `9 n, j5 L' v) E! {% L# {+ q# c- b/ x! ?
验证用户登陆信息的Servlet,此程序调用了CookieUtil.java中的saveCookie方法3 A- n4 s) L5 [- w6 l
* v) F$ n1 Z- \8 P5 W. g+ z
package cn.itcast.servlet;3 k0 o  l2 ]1 W( b/ v
  1. import java.io.IOException; * C# |% n* o! X
  2. import javax.servlet.ServletException;
    ' K" [1 e4 j2 g/ i  E
  3. import javax.servlet.http.HttpServlet;
    4 A0 ~5 o' d% u0 P5 P% H8 ^
  4. import javax.servlet.http.HttpServletRequest;
    ; j. v* v& l3 P! G6 u
  5. import javax.servlet.http.HttpServletResponse; , q& x+ H: F; G1 h, \5 z" M5 M4 @$ K4 x
  6. import javax.servlet.http.HttpSession; 5 d% @- }' |& {* F$ J2 |
  7. import cn.itcast.bean.User; 8 z" e/ f9 x% T1 N( Q! R4 b
  8. import cn.itcast.dao.UserDAO;
    . C  J' C% a/ c8 N+ _" [, j
  9. import cn.itcast.factory.DaoImplFactory; # P) A: }; ^4 o# F
  10. import cn.itcast.util.CookieUtil;
    5 ?. z: q- D0 w. k6 W0 j" ]

  11. # K, S+ Q5 e6 Z) ?! k7 i. e$ W4 V
  12. public class CheckLogonServlet extends HttpServlet {
    , \1 m. {* i3 h& Y

  13. ; r+ P# K+ `4 l1 u' c/ C! R; f4 d
  14. public void doGet(HttpServletRequest request, HttpServletResponse response)
    , t( r# Y& M* {! ]. u0 ]! }
  15. throws ServletException, IOException {
    ' C$ `( W' s3 a6 u8 A
  16. doPost(request, response);
      M0 A2 g+ K0 f9 q+ t1 J6 \
  17. }
    % v4 x  p7 H( V2 h
  18. $ v: U. M* f/ K& e$ `* H' @8 d
  19. public void doPost(HttpServletRequest request, HttpServletResponse response)
    7 k& m: F& j" U$ \' t
  20. throws ServletException, IOException {
    : K/ V, N6 J0 H; j# ?
  21. request.setCharacterEncoding("utf-8"); + j4 q. V* A" ~  O5 D
  22. String username = request.getParameter("username").trim();
    : Q; k) D/ S) Z( _
  23. String password = CookieUtil.getMD5(request.getParameter("password"));
    - f- n; @, B- A8 I* z0 [
  24. String remeberMe = request.getParameter("remeberMe"); + h( i7 o5 I& ]( H' V3 C( i
  25. HttpSession session = request.getSession(false);
    4 Z+ A* @; v/ u5 I/ H5 p. K

  26. 5 W* _# d( w; |9 t  T$ R
  27. // 将接收到的用户名传递到UserDao的checkUser方法中,检查用户+ A. \  b/ Z- E& T
  28. // 返回一个User类型的对象" ~+ @; y& f1 b
  29. UserDAO ud = DaoImplFactory.getInstance();
    # B% v3 }! P& l* F& A4 E+ J
  30. User user = ud.selectUserByUsername(username); # r: T+ ^  b& m- {8 H0 c# o* ]
  31. if (user == null) {( D8 C. B1 j- T. C
  32. request.setAttribute("checkUserError","<a href='register.jsp'><font color=red>用户名不存在,请先注册</font></a>"); / c, v* }  d$ K5 O2 r1 z
  33. request.getRequestDispatcher("index.jsp").forward(request, response);
    $ p( A3 w0 L1 E! f: _# \! Y
  34. return;
    7 a1 H8 j8 q% ^) p
  35. }
    2 u+ C: [7 ]* F
  36.   B# i: H! P" ]& `
  37. if(!password.equals(user.getPassword())){
    ) z% D5 S, D$ f! Z" n
  38. request.setAttribute("checkPasswordError","<font color=red>密码输入错误,请重新输入</font>"); ' R2 P+ g( j* u9 N; M
  39. request.getRequestDispatcher("index.jsp").forward(request, response);
    3 I) |$ q& ]+ I5 u0 B
  40. return;
    / U9 P" o: W) ?  w" _4 ^6 C
  41. }
    + |* e4 ?: b& M9 j
  42. & q( x0 P  O6 I5 C" `. X

  43. 2 q' P& B4 }( U: f: l% ^9 \
  44. //保存Cookie,这里调用了CookieUtil.java中的saveCookie方法,将上面的user对象作为参数传递8 o. [) W- ^5 D7 y7 w
  45. if ("on".equals(remeberMe)) {
    ( [) J# h# P( a* p' x7 b3 m
  46. CookieUtil.saveCookie(user, response);
    7 X" Y( K1 a7 r1 t, h
  47. }
    3 J* z" i9 G; J; h9 ]* \* K
  48. //在Session中保存用户信息,并转向用户的个人信息页面
    & `% s: Z) X! Y( E$ S. f6 R
  49. session.setAttribute("user", user);
    3 ^) x& d/ c2 X+ y$ |- {$ j3 |6 l1 T
  50. request.getRequestDispatcher("User/userInfo.jsp").forward(request,response);
    $ Y2 x8 O8 f: T
  51. }5 C! @% ~2 h$ m: E
  52. }
复制代码
0 n0 k/ j( f" q- p/ E
UserDAO.java与DaoImplFactory.java属于持久层相关的程序,这里就不贴出来了,读者可根据自己需要选择不同的持久层框架,在本程序中只要实现查询用户的功能就可以了8 _- H# b9 O7 S2 d
' p6 }2 |: y0 m* w; m& W# M2 W
, M0 n) L8 u) v  a: g

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


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

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

   

关闭

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

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