频道直达 - 学院 - 下载 - 交易 - 特效 - 字库 - 手册 -排名-工具- 繁體
网页教学网站开发 设为首页
加入收藏
联系我们
建站搜索: 常用广告代码   用户注册 | 用户登陆
您当前的位置:中国建站之家 -> 网站开发设计技术教程 -> Asp.Net教程 -> ASP.NET 1.1 无 Cookie SessionID 重写

ASP.NET 1.1 无 Cookie SessionID 重写

作者:圈圈  来源:本站整理  发布时间:2007-2-28 6:04:44  发布人:圈圈

减小字体 增大字体

   浏览器的会话使用存储在 SessionID 属性中的唯一标识符进行标识。会话 ID 使 ASP.NET 应用程序能够将特定的浏览器与 Web 服务器上相关的会话数据和信息相关联。会话 ID 的值在浏览器和 Web 服务器间通过 Cookie 进行传输,如果指定了无 Cookie 会话,则通过 URL 进行传输。
         ASP.NET 通过自动在页的 URL 中插入唯一的会话 ID 来保持无 Cookie 会话状态。例如,下面的 URL 已被 ASP.NET 修改,以包含唯一的会话 ID lit3py55t21z5v55vlm25s55:
               http://www.example.com/s(lit3py55t21z5v55vlm25s55)/orderform.aspx
         如果一个包含无 Cookie SessionID 的链接被多个浏览器共享时(可能通过搜索引擎或其他程序),此行为可能导致对会话数据的意外共享。可以通过禁用会话标识符的回收来降低多个客户端共享会话数据的可能性。为此,将 sessionState 配置元素的 regenerateExpiredSessionId 属性设置为 true。这样,在使用已过期的会话 ID 发起无 Cookie 会话请求时,将生成一个新的会话 ID。
                                                                                                                               ——摘自 MSDN

         .NET2.0中我们已可以通过重写SessionIDManager 类来改变SessionID 的生成机制和验证方法来防止会话数据的意外共享(即出现多个浏览器被识别为同一个会话,共用一个Session),可在.NET1.1中却没有相关的类让我们改变SessionID 的生成机制(封装死了),但受一篇文章的启发,我们可以在SessionID 生成之后对它进行处理。文章是老外写的,由于本人阅读能力有限,偶可没时间去看一大版唧唧歪歪的鹰文,直接下了代码来看。还好代码不算多,思路也很清晰,大概了解了他实现重写SessionID的原理,我改了下在无Cookie 会话中完美实现了。
相关代码
using System;
using System.Web;
using System.Web.SessionState;
using System.Web.Security;
using System.Configuration;
using System.Security.Cryptography;
using System.Runtime.Serialization;
using System.Globalization;
using System.Text;

public class SecureSessionModule : IHttpModule
{
    private static string _ValidationKey = null;

    public void Init (HttpApplication app)
    {
        if (_ValidationKey == null)
            _ValidationKey = GetValidationKey ();
        app.AcquireRequestState+=new EventHandler(app_AcquireRequestState);
    }

    void app_AcquireRequestState (Object sender, EventArgs e)
    {
        _ValidationKey=GetValidationKey();//每天生成一个KEY提高安全性

        HttpContext current  = ((HttpApplication) sender).Context;

        //将处理后的SessionID存在Session["ASP.NET_SessionID"]中
        string sessionid = GetSession (current, "ASP.NET_SessionID");

        if (sessionid != null)
        {
            if (sessionid.Length <= 24)
                RedirectUrl(current);

            string id = sessionid.Substring (0, 24);
            string mac1 = sessionid.Substring (24);

            string mac2 = GetSessionIDMac (id, current.Request.UserHostAddress, current.Request.UserAgent, _ValidationKey);

            // 用户客户端信息发生的变化,比对失败
            if (String.CompareOrdinal(mac1, mac2) != 0)
            {
                RedirectUrl(current);
            }
        }
        else
        {
            RedirectUrl(current);
        }
    }

    private void RedirectUrl(HttpContext current)
    {
         //重定向页面以重新生成新的SessionID
         current.Response.Redirect(current.Request.Url.ToString(),true);
    }

    private string GetValidationKey ()
    {
        string key = DateTime.Now.ToShortDateString();

        return key;
    }

    private string GetSession (HttpContext current, string name)
    {
        object id = FindSession(current.Session,name);
        if (id == null)
        {
            // 将用户客户端信息加密存储在Session中以便比对
            id= current.Session.SessionID+GetSessionIDMac (current.Session.SessionID, current.Request.UserHostAddress,
                current.Request.UserAgent, _ValidationKey);
            current.Session[name]  = id;
        }
        return id.ToString();
    }

    private object FindSession (HttpSessionState session, string name)
    {
        return session[name];
    }

    private string GetSessionIDMac (string id, string ip, string agent, string key)
    {
        StringBuilder builder = new StringBuilder (id, 512);       
        builder.Append (ip);
        builder.Append (agent);

        using (HMACSHA1 hmac = new HMACSHA1 (Encoding.UTF8.GetBytes (key)))
        {
            return Convert.ToBase64String (hmac.ComputeHash (
                Encoding.UTF8.GetBytes (builder.ToString ())));
        }
    }

     public void Dispose () {}
}相关配置如下:
<configuration>
  <system.web>
    <httpModules>
      <add name="SecureSession" type="SecureSessionModule,SecureSessionModule" />
    </httpModules>
  </system.web>
</configuration>

       大家看了代码后就会知道,它实现的原理主要是因为不可能有相同IP电脑客户端同时访问一台服务器,当出现SessionID相同但他们客户端信息不同时就自动将后一个访问的客户端重定向以新建一个会话。
     
       遗憾的是在WAP上应用时就有问题了,由于客户端信息没IP唯一标识(移动不给手机号信息了),所以如果相同型号的手机访问时就无法区分,不知哪位高人有没更好的解决办法,还望不吝赐教

题外话:工作忙,时间紧,抄得多,写得少,有问题,请留言。欢迎大家多交流沟通~~~
http://www.cnblogs.com/outman2008/archive/2007/02/25/655804.html


将本文收藏到QQ书签与更多好友分享
[打 印]
[] [返回上一页] [收 藏]
∷相关文章评论∷    (评论内容只代表网友观点,与本站立场无关!) [更多评论...]
精彩推荐
热门文章
· 注册码大全二
· 注册码大全四
· 注册码大全一
· 要10G免费网络硬盘的请进..
· 通过google 赶快来赚美金..
· 注册码大全十
· 头像-qq头像(qq新头像)4..
· 让你轻松架设FTP服务器1..
· 注册码大全三
· 梦幻背景图片7
· 卡通动物图片6
· 网页制作素材-按钮素材2..
· 让你轻松架设FTP服务器5..
· 风景图片8
· 注册码大全九
· 让你轻松架设FTP服务器2..
关注此文读者还看过
· 大话XML(2)XML和HTML的比..
· 在网页调用 WINDOWS 控件..
· 首届行业网站的年会闭幕..
· ACCESS默认保存路径的修..
· 美国在线副总裁淡出日常..
· 关于录音,新手教程
· 关于session的一些问题
· 把数字转换成英语或是汉..
· 用asp做access的远程接口..
· 计数器的另一用法:自动..
· 防沉迷“催生”Q版网游
· Photoshop应用实例 打造..
· 网页制作中关于CSS设置字..
· 搭车免费强制安装 美流氓..
· Flash MX2004入门与进阶..
· MySQL数据库技术(27)
相关文章
· ASP.NET连接Access和SQL Se..
· Asp.net中防止用户多次登录..
· ASP.NET 与 Ajax 的实现方式..
· ASP.net平台社区软件Discuz..
· ASP.NET AJAX中的异步..
· 解析:如何在 ASP.NET 中下..
· ASP.Net中保护自定义的服务..
· ASP.NET中的doPostBack脚本..
· 不使用VS进行ASP.NET Membe..
· 关于ASP.NET编程中的嵌套If..
· ASP.NET 2.0 页面状态持续程..
· ASP.NET中的日期与时间的处..
· Asp.net编程中的数组基础实..
· 简单介绍 ASP.NET 中的运算..
· Asp.Net 构架(Http请..
· Asp.net中的Popup控件..
关于本站 - 网站帮助 - 广告合作 - 下载声明 - 友情连接 - 网站地图 - 人才招聘
网站合作、内容监督、商务咨询:QQ: 9576619
Copyright ? 2005--2008 中国建站之家版权所有
粤ICP备05092265号