<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    隨筆 - 147  文章 - 71  trackbacks - 0
    <2009年3月>
    22232425262728
    1234567
    891011121314
    15161718192021
    22232425262728
    2930311234

    常用鏈接

    留言簿(1)

    隨筆分類(146)

    隨筆檔案(147)

    文章分類(28)

    文章檔案(28)

    喜歡的Blog

    搜索

    •  

    最新評論

    閱讀排行榜

    評論排行榜

    JSP作為后起之秀能夠在服務(wù)器編程環(huán)境中占據(jù)一定地位,是和它良好支持一系列業(yè)界標(biāo)準(zhǔn)密切相關(guān)的。Session就是它提供的基礎(chǔ)設(shè)施之一。作為一個程序員,你可以不介意具體在客戶端是如何實現(xiàn),就方便的實現(xiàn)簡單的基于session的用戶管理。

    現(xiàn)在對于處理在線用戶,有幾種不同的處理方法。一種是頁面刷新由用戶控制,服務(wù)器端控制一個超時時間比如30分鐘,到了時間之后用戶沒有動作就被踢出。這種方法的優(yōu)點是,如果用戶忘了退出,可以防止別人惡意操作。缺點是,如果你在做一件很耗時間的事情,超過了這個時間限制,submit的時候可能要再次面臨登陸。如果原來的頁面又是強制失效的話,就有可能丟失你做的工作。在實現(xiàn)的角度來看,這是最簡單的,Server端默認(rèn)實現(xiàn)的就是這樣的模式。

    另一種方式是,站點采用框架結(jié)構(gòu),有一個Frame或者隱藏的iframe在不斷刷新,這樣你永遠(yuǎn)不會被踢出,但是服務(wù)器端為了判斷你是否在線,需要定一個發(fā)呆時間,如果超過這個發(fā)呆時間你除了這個自動刷新的頁面外沒有刷新其他頁面的話,就認(rèn)為你已經(jīng)不在線了。采取這種方式的典型是xici.net。 他的優(yōu)點是可以可以利用不斷的刷新實現(xiàn)一些類似server-push的功能,比如網(wǎng)友之間發(fā)送消息。

    不管哪一種模式,為了實現(xiàn)瀏覽當(dāng)前所有的在線用戶,還需要做一些額外的工作。servlet API中沒有得到Session列表的API。

    可以利用的是Listener. Servlet 2.2和2.3規(guī)范在這里略微有一些不一樣。2.2中HttpSessionBindingListener可以實現(xiàn)當(dāng)一個HTTPSession中的Attribute變化的時候通知你的類。而2.3中還引入了HttpSessionAttributeListener.鑒于我使用的環(huán)境是Visual age for java 4和JRun server 3.1,他們還不直接支持Servlet 2.3的編程,這里我用的是HttpSessionBindingListener.

        需要做的事情包括做一個新的類來實現(xiàn)HttpSessionBindingListener接口。這個接口有兩個方法:
    public void valueBound(HttpSessionBindingEvent event),和
    public void valueUnbound(HttpSessionBindingEvent event)。
    當(dāng)你執(zhí)行Session.addAttribute(String,Object)的時候,如果你已經(jīng)把一個實現(xiàn)了HttpSessionBindingListener接口的類加入為Attribute,Session會通知你的類,調(diào)用你的valueBound方法。相反,Session.removeAttribute方法對應(yīng)的是valueUndound方法。
     1public class HttpSessionBinding implements javax.servlet.http.HttpSessionBindingListener 
     2{
     3    ServletContext application = null;
     4
     5    public HttpSessionBinding(ServletContext application)
     6    {
     7        super();
     8        if (application ==null)
     9            throw new IllegalArgumentException("Null application is not accept.");
    10        
    11        this.application = application;
    12        
    13    }

    14
    15    public void valueBound(javax.servlet.http.HttpSessionBindingEvent e) 
    16    
    17        Vector activeSessions = (Vector) application.getAttribute("activeSessions");
    18        if (activeSessions == null)
    19        {
    20            activeSessions = new Vector();
    21        }

    22
    23        JDBCUser sessionUser = (JDBCUser)e.getSession().getAttribute("user");
    24        if (sessionUser != null)
    25        {
    26            activeSessions.add(e.getSession());
    27        }

    28        application.setAttribute("activeSessions",activeSessions);
    29    }

    30
    31    public void valueUnbound(javax.servlet.http.HttpSessionBindingEvent e) 
    32    {
    33        JDBCUser sessionUser = (JDBCUser)e.getSession().getAttribute("user");
    34        if (sessionUser == null)
    35        {
    36            Vector activeSessions = (Vector) application.getAttribute("activeSessions");
    37            if (activeSessions != null)
    38            {
    39                activeSessions.remove(e.getSession().getId());
    40                application.setAttribute("activeSessions",activeSessions);
    41            }

    42        }

    43    }

    44}

    假設(shè)其中的JDBCUser類是一個任意User類。在執(zhí)行用戶登錄時,把User類和HttpSessionBinding類都加入到Session中去。這樣,每次用戶登錄后,在application中的attribute "activeSessions"這個vector中都會增加一條記錄。每當(dāng)session超時,valueUnbound被觸發(fā),在這個vector中刪去將要被超時的session。
     1public void login()
     2throws ACLException,SQLException,IOException
     3{
     4    /* get JDBC User Class */
     5    if (user != null)
     6    {
     7        logout();
     8    }

     9    {
    10        // if session time out, or user didn't login, save the target url temporary.
    11
    12        JDBCUserFactory uf = new JDBCUserFactory();
    13
    14        if ( (this.request.getParameter("userID")==null
    15            || (this.request.getParameter("password")==null)  )
    16        {
    17            throw new ACLException("Please input a valid userName and password.");
    18        }

    19        
    20        JDBCUser user = 
    21            (JDBCUser) uf.UserLogin(
    22                this.request.getParameter("userID"),
    23                this.request.getParameter("password") );
    24        user.touchLoginTime();
    25        this.session.setAttribute("user",user);
    26        this.session.setAttribute("BindingNotify",new HttpSessionBinding(application));
    27    }

    28}

    這兩個函數(shù)位于一個HttpSessionManager類中.這個類引用了jsp里面的application全局對象。這個類的其他代碼和本文無關(guān)且相當(dāng)長,我就不貼出來了。

    下面來看看jsp里面怎么用。
    假設(shè)一個登錄用的表單被提交到doLogin.jsp, 表單中包含UserName和password域。

    節(jié)選部分片段:
     1<%
     2    HttpSessionManager hsm = new HttpSessionManager(application,request,response);
     3    try
     4    {
     5        hsm.login();
     6    }

     7    catch ( UserNotFoundException e)
     8    {
     9        response.sendRedirect("InsufficientPrivilege.jsp?detail=User%20does%20not%20exist.");
    10        return;
    11    }

    12    catch (    InvalidPasswordException e2)
    13    {    
    14        response.sendRedirect("InsufficientPrivilege.jsp?detail=Invalid%20Password");
    15        return;
    16    }

    17    catch ( Exception e3)
    18    {
    19    %>    Error:<%=e3.toString() %><br>
    20        Press <a href="login.jsp">Here</a> to relogin.
    21<%        return;
    22    }

    23    response.sendRedirect("index.jsp");
    24%>

    再來看看現(xiàn)在我們怎么得到一個當(dāng)前在線的用戶列表。
     1<body bgcolor="#FFFFFF">
     2<table cellspacing="0" cellpadding="0" width="100%">
     3
     4    <tr >
     5      <td style="width:24px">SessionId
     6      </td>
     7      <td style="width:80px" >User
     8      </td>
     9      <td style="width:80px" >Login Time
    10      </td>
    11      <td style="width:80px" >Last access Time
    12      </td>
    13    </tr>
    14<%
    15    Vector activeSessions = (Vector) application.getAttribute("activeSessions");
    16    if (activeSessions == null)
    17    {
    18        activeSessions = new Vector();
    19        application.setAttribute("activeSessions",activeSessions);
    20    }
    21    
    22    Iterator it = activeSessions.iterator();
    23    while (it.hasNext())
    24    {
    25        HttpSession sess = (HttpSession)it.next();
    26        JDBCUser sessionUser = (JDBCUser)sess.getAttribute("user");
    27        String userId = (sessionUser!=null)?sessionUser.getUserID():"None";
    28
    %>
    29    <tr>
    30          <td nowrap=''><%= sess.getId() %></td>
    31          <td nowrap=''><%= userId %></td>
    32          <td nowrap=''>
    33        <%=  BeaconDate.getInstance( new java.util.Date(sess.getCreationTime())).getDateTimeString()%></td>
    34          <td class="<%= stl %>3" nowrap=''>
    35        <%=  BeaconDate.getInstance( new java.util.Date(sess.getLastAccessedTime())).getDateTimeString()%></td>
    36    </tr>
    37<%
    38    }
    39
    %>
    40</table>
    41</body>

    以上的代碼從application中取出activeSessions,并且顯示出具體的時間。其中BeaconDate類假設(shè)為格式化時間的類。

    這樣,我們得到了一個察看在線用戶的列表的框架。至于在線用戶列表分頁等功能,與本文無關(guān),不予討論。
    這是一個非刷新模型的例子,依賴于session的超時機制。我的同事sonymusic指出很多時候由于各個廠商思想的不同,這有可能是不可信賴的。考慮到這種需求,需要在每個頁面刷新的時候都判斷當(dāng)前用戶距離上次使用的時間是否超過某一個預(yù)定時間值。這實質(zhì)上就是自己實現(xiàn)session超時。如果需要實現(xiàn)刷新模型,就必須使用這種每個頁面進行刷新判斷的方法。

    附:關(guān)閉瀏覽器窗口,釋放session的問題
    在主頁中添加以下代碼:
     1<script type="text/javascript">
     2function MM_callJS(jsStr) {
     3    return eval(jsStr);
     4}

     5    
     6function removeline() {
     7    if (event.clientX < 0 && event.clientY < 0{
     8        document.write('<iframe width="100" height="100" src="logout.jsp"></iframe><OBJECT classid=CLSID:8856F961-340A-11D0-A96B-00C04FD705A2 height=0  id=WebBrowser width=0></OBJECT>');
     9        document.all.WebBrowser.ExecWB(451);
    10    }

    11}

    12
    </script>
    13
    14<frameset id="frame1" onUnload="MM_callJS('removeline()')">
    代碼解釋:
    if(event.clientX<0&&event.clientY<0)判斷瀏覽器是關(guān)閉還是刷新,因為刷新也會調(diào)用onunload。
    document.all.WebBrowser.ExecWB(45,1);是無提示的關(guān)閉瀏覽器。
    classid=CLSID:8856F961-340A-11D0-A96B-00C04FD705A2
    這個是調(diào)用不彈出對話框的方法,實際是調(diào)用系統(tǒng)的方法如下
    document.all.WebBrowser.ExecWB(45,1);
    因為在javascript當(dāng)中不能調(diào)用java方法,所以選擇另外寫一個logout.jsp文件用于調(diào)用invalidate()方法。
    注意:session在調(diào)用invalidate()方法前要removeAttribute掉相關(guān)的name。



    限制用戶重復(fù)登錄
    OnlineUserBindingListener.java
    import java.text.SimpleDateFormat;
    import java.util.ArrayList;
    import java.util.Date;
    import java.util.List;

    import javax.servlet.ServletContext;
    import javax.servlet.http.HttpSession;
    import javax.servlet.http.HttpSessionBindingEvent;
    import javax.servlet.http.HttpSessionBindingListener;

    public class OnlineUserBindingListener implements HttpSessionBindingListener {
             String username;
            
    public OnlineUserBindingListener(){
            }
            
            
    public OnlineUserBindingListener(String username){
                
    this.username=username;
            }
            
    public void valueBound(HttpSessionBindingEvent event) {
                HttpSession session 
    = event.getSession();
                ServletContext application 
    = session.getServletContext();
                
    // 把用戶名放入在線列表
                List onlineUserList = (List) application.getAttribute("onlineUserList");
                
    // 第一次使用前,需要初始化
                if (onlineUserList == null) {
                    onlineUserList 
    = new ArrayList();
                    
                }
                onlineUserList.add(
    this.username);
                application.setAttribute(
    "onlineUserList", onlineUserList);
            }

            
    public void valueUnbound(HttpSessionBindingEvent event) {
                HttpSession session 
    = event.getSession();
                ServletContext application 
    = session.getServletContext();

                
    // 從在線列表中刪除用戶名
                List onlineUserList = (List) application.getAttribute("onlineUserList");
                onlineUserList.remove(
    this.username);
                SimpleDateFormat sdf 
    = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
                System.out.println(
    this.username + "退出系統(tǒng)。");
                System.out.println(
    "退出時間"+sdf.format(new Date()));
            }
        
    }
    登陸驗證
    OnlineUserBindingListener online = new OnlineUserBindingListener();
    List list 
    = (List) application.getAttribute("onlineUserList");
    boolean flag = false;
    if (list != null) {
        
    for (int i = 0; i < list.size(); i++) {
            String param 
    = (String) list.get(i);
            
    if (param.equals(username)) {
                flag 
    = true;
                
    break;
            }
        }
    }
    if (flag) {
        out.println(
    "<script language='javascript'>alert('該用戶正在使用中!');"
            
    + "parent.location.href='login.jsp';</script>");
    else {
        session.setAttribute(
    "onlineUserBindingListener",
            
    new OnlineUserBindingListener(username));
    }
    退出系統(tǒng)
    // 從在線列表中刪除用戶名
    List onlineUserList = (List) application.getAttribute("onlineUserList");
    onlineUserList.remove(session.getAttribute(
    "username"));
    session.invalidate();

    posted on 2009-03-19 20:45 飛翔天使 閱讀(814) 評論(0)  編輯  收藏 所屬分類: JSP
    主站蜘蛛池模板: 成人免费a级毛片无码网站入口| 日本人成在线视频免费播放| 久久高潮一级毛片免费| 国产精品免费AV片在线观看| 精品国产sm捆绑最大网免费站| 成年女人毛片免费观看97| 亚洲 综合 国产 欧洲 丝袜| 大地影院MV在线观看视频免费 | 亚洲伊人久久大香线蕉啊| 中文字幕亚洲综合小综合在线| 在线观看亚洲视频| 日韩电影免费在线观看网站| 真人做人试看60分钟免费视频| 日韩精品免费电影| 亚洲人成无码网站| 亚洲情a成黄在线观看| 亚洲av无码成h人动漫无遮挡| 中文文字幕文字幕亚洲色| 二级毛片免费观看全程| 一级毛片在线播放免费| 热re99久久6国产精品免费| 精品无码国产污污污免费| 亚洲情XO亚洲色XO无码| 亚洲午夜在线播放| 国产免费A∨在线播放| 无码国产精品一区二区免费| 在线v片免费观看视频| 免费下载成人电影| 亚洲人成网站18禁止一区| 亚洲欧洲日韩综合| 日日摸夜夜添夜夜免费视频| 麻花传媒剧在线mv免费观看 | 免费h黄肉动漫在线观看| 亚洲天天做日日做天天欢毛片| 亚洲av无码兔费综合| 99久久综合精品免费| 亚洲精品人成无码中文毛片| 亚洲一卡2卡4卡5卡6卡在线99| 久久久久国色AV免费观看| 日韩免费a级在线观看| 亚洲美女视频一区|