HTTP cookie

细说Cookie

阅读目录

· 开始

· Cookie 概述

· Cookie的写、读过程

· 使用Cookie保存复杂对象

· Js中读写Cookie

· CookieSession中的应用

· Cookie在身份验证中的应用

· Cookie的安全状况

· 如何在C#发请的请求中使用Cookie

· 重构与使用总结

· 补充

Cookie虽然是个很简单的东西,但它又是WEB开发中一个很重要的客户端数据来源,而且它可以实现扩展性很好的会话状态, 所以我认为每个WEB开发人员都有必要对它有个清晰的认识。本文将对Cookie这个话题做一个全面的描述, 也算是我对Cookie的认识总结。

回到顶部

Cookie 概述

Cookie是什么? Cookie 是一小段文本信息,伴随着用户请求和页面在 Web 服务器和浏览器之间传递。Cookie 包含每次用户访问站点时 Web 应用程序都可以读取的信息。

为什么需要Cookie 因为HTTP协议是无状态的,对于一个浏览器发出的多次请求,WEB服务器无法区分 是不是来源于同一个浏览器。所以,需要额外的数据用于维护会话。 Cookie 正是这样的一段随HTTP请求一起被传递的额外数据。

Cookie能做什么? Cookie只是一段文本,所以它只能保存字符串。而且浏览器对它有大小限制以及 它会随着每次请求被发送到服务器,所以应该保证它不要太大。 Cookie的内容也是明文保存的,有些浏览器提供界面修改,所以, 不适合保存重要的或者涉及隐私的内容。

Cookie 的限制。 大多数浏览器支持最大为 4096 字节的 Cookie。由于这限制了 Cookie 的大小,最好用 Cookie 来存储少量数据,或者存储用户 ID 之类的标识符。用户 ID 随后便可用于标识用户,以及从数据库或其他数据源中读取用户信息。 浏览器还限制站点可以在用户计算机上存储的 Cookie 的数量。大多数浏览器只允许每个站点存储 20 个 Cookie;如果试图存储更多 Cookie,则最旧的 Cookie 便会被丢弃。有些浏览器还会对它们将接受的来自所有站点的 Cookie 总数作出绝对限制,通常为 300 个。

通过前面的内容,我们了解到Cookie是用于维持服务端会话状态的,通常由服务端写入,在后续请求中,供服务端读取。 下面本文将按这个过程看看Cookie是如何从服务端写入,最后如何传到服务端以及如何读取的。

回到顶部

Cookie的写、读过程

Asp.net中,读写Cookie是通过使用HttpCookie类来完成的,它的定义如下:

public sealed class HttpCookie

{

    // 获取或设置将此 Cookie 与其关联的域。默认值为当前域。

    public string Domain { getset; }

    // 获取或设置此 Cookie 的过期日期和时间(在客户端)。

    public DateTime Expires { getset; }

    // 获取一个值,通过该值指示 Cookie 是否具有子键。

    public bool HasKeys { get; }

    // 获取或设置一个值,该值指定 Cookie 是否可通过客户端脚本访问。

    // 如果 Cookie 具有 HttpOnly 属性且不能通过客户端脚本访问,则为 true;否则为 false。默认为 false

    public bool HttpOnly { getset; }

    // 获取或设置 Cookie 的名称。

    public string Name { getset; }

    // 获取或设置要与当前 Cookie 一起传输的虚拟路径。默认值为当前请求的路径。

    public string Path { getset; }

    // 获取或设置一个值,该值指示是否使用安全套接字层 (SSL)(即仅通过 HTTPS)传输 Cookie

    public bool Secure { getset; }

    // 获取或设置单个 Cookie 值。默认值为空引用。

    public string Value { getset; }

    // 获取单个 Cookie 对象所包含的键值对的集合。

    public NameValueCollection Values { get; }

    // 获取 System.Web.HttpCookie.Values 属性的快捷方式。

    public string this[string key] { getset; }

}

 

Cookie写入浏览器的过程:我们可以使用如下代码在Asp.net项目中写一个Cookie 并发送到客户端的浏览器(为了简单我没有设置其它属性)。

HttpCookie cookie new HttpCookie("MyCookieName""string value");

Response.Cookies.Add(cookie);

 

我想很多人都写过类似的代码,但是,大家有没有想过:Cookie最后是如何发送到客户端的呢?我们打开Fiddler来看一下吧。

从上图,您应该能发现,我们在服务端写的Cookie,最后其实是通过HTTP的响应头这种途径发送到客户端的。每一个写入动作, 都会产生一个【Set-Cookie】的响应头。
浏览器正是在每次获取请求的响应后,检查这些头来接收Cookie的。

Asp.net获取Cookie的过程:我们可以使用如下代码在Asp.net项目中读取一个Cookie

HttpCookie cookie Request.Cookies["MyCookieName"];

if( cookie != null )

    labCookie1.Text cookie.Value;

else

    labCookie1.Text "未定义";

 

代码同样也很简单,还是类似的问题:大家有没有想过,Cookie是如何传到服务端的呢?我们还是继续使用Fiddler来寻找答案吧。

从图片中,我们可以发现,Cookie是放在请求头中,发送到服务端的。如果你一直刷新页面,就能发现, 每次HTTP请求,Cookie都会被发送。当然了,浏览器也不是发送它所接收到的所有Cookie,它会检查当前要请求的域名以及目录, 只要这二项目与Cookie对应的DomainPath匹配,才会发送。对于Domain则是按照尾部匹配的原则进行的。
所以,我在访问 www.cnblogs.com 时,浏览器并不会将我在浏览 www.163.com 所接收到的 Cookie 发出去。

删除Cookie:其实就是在写Cookie时,设置Expires为一个【早于现在时间的时间】。也就是:设置此Cookie已经过期, 浏览器接收到这个Cookie时,便会删除它们。

HttpCookie cookie new HttpCookie("MyCookieName"null);

cookie.Expires new DateTime(190011);

Response.Cookies.Add(cookie);

 

回到顶部

使用Cookie保存复杂对象

前面的示例代码大致演示了Cookie的读写操作。不过,我们平时可能希望将更复杂的【自定义类型】通过Cookie来保存, 那么又该如何操作呢?对于这个问题,我们定义一个类型来看看如何处理。

public class DisplaySettings 

{

    public int Style;

 

    public int Size;

    

    public override string ToString()

    {

        return string.Format("Style = {0}, Size = {1}"this.Style, this.Size);

    }    

}

 

上面的代码,我定义一个类型,用于保存用户在浏览页面时的显示设置。接下来,我将介绍二种方法在Cookie中保存并读取它们。

方法-1,经典做法。(注意前面给出的HttpCookie定义代码中的最后二个成员)

private void WriteCookie_2a()

{

    DisplaySettings setting new DisplaySettings { Style 1, Size 24 };

 

    HttpCookie cookie new HttpCookie("DisplaySettings1");

    cookie["Style"setting.Style.ToString();

    cookie["Size"setting.Size.ToString();

 

    Response.Cookies.Add(cookie);

}

 

private void ReadCookie_2a()

{

    HttpCookie cookie Request.Cookies["DisplaySettings1"];

    if( cookie == null )

        labDisplaySettings1.Text "未定义";

    else {

        DisplaySettings setting new DisplaySettings();

        setting.Style cookie["Style"].TryToInt();

        setting.Size cookie["Size"].TryToInt();

        labDisplaySettings1.Text setting.ToString();

    }

}

 

方法-2,将对象JSON序列化为字符串。

private void WriteCookie_2b()

{

    DisplaySettings setting new DisplaySettings { Style 2, Size 48 };

 

    HttpCookie cookie new HttpCookie("DisplaySettings2", setting.ToJson());

    Response.Cookies.Add(cookie);

}

 

private void ReadCookie_2b()

{

    HttpCookie cookie Request.Cookies["DisplaySettings2"];

    if( cookie == null )

        labDisplaySettings2.Text "未定义";

    else {

        DisplaySettings setting cookie.Value.FromJson<DisplaySettings>();

        labDisplaySettings2.Text setting.ToString();

    }

}

 

这段代码使用了我定义的二个扩展方法。

对于这二种方法,我个人更喜欢后者,因为它具有更好扩展性:如果类型增加了成员,不需要修改读写Cookie的代码。
不过,这种方式产生的有些字符,比如【双引号】,极少数浏览器(Opera)不支持,所以需要做UrlEncode或者Base64编码处理。
同理,对于第一种方法,遇到Value有【双引号】时,我们同样需要做UrlEncode或者Base64编码处理。

回到顶部

Js中读写Cookie

Cookie并非只能在服务端读写,在客户端的浏览器中也可以实现对它的读写访问。而且在JS中创建的Cookie对于服务端仍然有效(可见), 接下来我们来看看在JS中如何写入Cookie,演示代码将创建一个按钮,并在点击按钮后写入Cookie

<input type="button" onclick="WriteCookie();" value="WriteCookie" />

 

<script type="text/javascript">

    function WriteCookie() {

        var cookie "cookie_js=22222222; path=/";

        document.cookie cookie;

    }    

</script>

 

JS中写Cookie很简单,只要给document.cookie赋值一个Cookie字符串即可,至于格式,可以参考前面用Fiddle看到的结果。

再来看一下如何使用JS读取Cookie吧。请参考如下代码:

<input type="button" onclick="ReadCookie();" value="ReadCookie" />

 

<script type="text/javascript">

    function ReadCookie() {

        alert(document.cookie);

    }    

</script>

 

仍然是访问document.cookie,不过,这次我们得到却是全部的Cookie值,每个Key/Value项用分号分开,中间则用等号分开。 所以, 如果您想在JS中读取Cookie,一定要按照这个规则来拆分并解析您要读取的Cookie项。鉴于这样的操作有些繁琐, 我们可以jquery.cookie.js插件来轻松完成这个功能,有兴趣的朋友也可以看一下它是如何处理的。 这个插件的代码比较少,这里就直接贴出, 

注意哦:前面我们看到了HttpCookie有个HttpOnly属性,如果它为true,那么JS是读不到那个Cookie的,也就是说: 我们如果在服务端生成的Cookie不希望在JS中能被访问,可以在写Cookie时,设置这个属性。不过,通过一些工具,还是可以看到它们。

接下来,我们再来看看Asp.netCookie有哪些应用。

回到顶部

CookieSession中的应用

Asp.net中,HttpContext, Page对象都有个Session的对象,我们可以使用它来方便地在服务端保存一些与会话相关的信息。
前面我们也提到过,HTTP协议是无状态的,对于一个浏览器发出的多次请求,WEB服务器无法区分 是不是来源于同一个浏览器。 所以,为了实现会话,服务端需要一个会话标识ID能保存到浏览器,让它在后续的请求时都带上这个会话标识ID,以便让服务端知道 某个请求属于哪个会话,这样便可以维护与会话相关的状态数据。由于Cookie对于用户来说,是个不可见的东西,而且每次请求都会传递到 服务端,所以它就是很理想的会话标识ID的保存容器。在Asp.net中,默认也就是使用Cookie来保存这个ID的。注意:虽然Asp.net 2.0 也支持无Cookie的会话,但那种方式要修改URL,也有它的缺点,因此这种方法并没有广泛的使用。本文将不对这个话题做过多的分析, 就此略过无Cookie会话这种方式。

我们来看看Session是如何使用Cookie来保存会话标识ID的,在默认的Asp.net配置中,Web.config有着如下定义:

<sessionState mode="InProccookieName="ASP.NET_SessionIdcookieless="UseCookies"></sessionState>

 

 

如果我们执行以下操作:

Session["Key1"DateTime.Now;

 

 

此时,我们可以使用一些浏览器提供的工具来查看一下现在的Cookie情况。

从图片上看,这个Cookie的名字就是我们在配置文件中指出的名称,我们可以修改一下配置文件:

<sessionState cookieName="SK"></sessionState>

 

 

再来执行上面的写Session的操作,然后看Cookie

我们可以看到:SKCookie出现了。说明:在截图时我把名称为"ASP.NET_SessionId"Cookie删除了。

通过上面示例,我们可以得到结论,Session的实现是与Cookie有关的,服务端需要将会话标识ID保存到Cookie中。
这里再一次申明,除非你使用无Cookie的会话模式,否则Session是需要Cookie的支持。反过来,Cookie并不需要Session的支持。

回到顶部

Cookie在身份验证中的应用

我想很多人都在Asp.net的开发中使用过Form身份认证。对于一个用户请求, 我们可以在服务端很方便地判断它是不是代表一个已登录用户。

this.labStatus.Text (Request.IsAuthenticated "已登录"未登录");

 

 

那么,您有没有好奇过:Asp.net是如何识别一个请求是不是一个已登录用户发起的呢?说到这里,我们就要从用户登录说起了。 为了实现登录及Form认证方式,我们需要如下配置:

<authentication mode="Forms>

    <forms name="UserStatus"></forms>

</authentication>

 

接下来,我们需要实现用户登录逻辑。具体实现方式有很多,不过,最终的调用都是差不多的,如下代码所示:

private void SetLogin()

{

    System.Web.Security.FormsAuthentication.SetAuthCookie("fish"false);

}

 

只要执行了以上代码,我们就可以看到,前面的判断【Request.IsAuthenticated】返回true,最终会显示"已登录"。 为了探寻这个秘密,我们还是来看一下当前页面的Cookie情况。

果然,多出来一个Cookie,名称与我在配置文件中指定的名称相同。我们再来看看如果注销当前登录会是什么样子的:

private void SetLogout()

{

    System.Web.Security.FormsAuthentication.SignOut();

}

 

看到了吗,名为"UserStatus"Cookie不见了。此时如果你再去观察【Request.IsAuthenticated】,可以发现它此时返回 false。 或者,您也可以再试一次,登录后,直接删除名为"UserStatus"Cookie,也能发现登录状态将显示"未登录"。 或许,您还是有点不清楚前面我调用【System.Web.Security.FormsAuthentication.SetAuthCookie("fish", false);】后,Asp.net做了些什么, 回答这个问题其实很简单:自己用Reflector.exe去看一下Asp.net的实现吧。
这里为了更让您能信服登录与Cookie有关,我将直接创建一个Cookie看一下 Asp.net能不能认可我创建的Cookie,并认为登录有效。请看代码:

private void SetLogin()

{

    //System.Web.Security.FormsAuthentication.SetAuthCookie("fish", false);

 

    // 下面的代码和上面的代码在作用上是等效的。

    FormsAuthenticationTicket ticket new FormsAuthenticationTicket(

        2"fish"DateTime.Now, DateTime.Now.AddDays(30d), falsestring.Empty);

    string str FormsAuthentication.Encrypt(ticket);

 

    HttpCookie cookie new HttpCookie(FormsAuthentication.FormsCookieName, str);

    Response.Cookies.Add(cookie);

}

 

如果执行这段代码,您将发现:【Request.IsAuthenticated】返回true,登录状态会显示"已登录"
至此,我们可以得出一个结论: Form身份认证依赖CookieAsp.net就是每次检查我们在配置文件中指定的Cookie名称,并解密这个Cookie来判断当前请求用户的登录状态。

回到顶部

Cookie的安全状况

从以上图片,您应该能发现:浏览器能提供一些界面让用户清楚的观察我们在服务端写的Cookie, 甚至有些浏览器还提供很方便的修改功能。如下图所示:

所以,我们在服务端写代码读取Cookie时,尤其是涉及类型转换、反序列化或者解密时,一定要注意这些操作都有可能会失败。 而且上图也清楚的反映了一个事实:Cookie中的值都是“一目了然”的,任何人都能看到它们。所以,我们尽量不要直接在Cookie中 保存一些重要的或者敏感的内容。如果我们确实需要使用Cookie保存一些重要的内容,但又不希望被他人看懂, 我们可以使用一些加密的方法来保护这些内容。

1. 对于一些重要性不高的内容,我们可以使用Base64之类的简单处理方式来处理。

2. 对于重要性相对高一点的内容,我们可以利用.net提供的一些加密工具类,自己来设计加密方法来保护。不过, 密码学与加密解密并不是很简单的算法,因此,自己设计的加密方式可能不会很安全。

3. 重要的内容,我们可以使用.net提供的FormsAuthenticationTicket,FormsAuthentication来加密。我认为这种方式还是比较安全的。 毕竟前面我们也看过了,Asp.netForm身份认证就是使用这种方式来加密用户登录的身份标识的,所以,如果这种方式不安全, 也就意味着Asp.net的身份认证也不安全了。 如果您使用这种方式来加密,那么请注意:它产生的加密后文本还是比较大的, 前面我也提到过,每次请求时,浏览器都会带上与请求相匹配的所有Cookie,因此,这种Cookie会对传输性能产生一定的影响, 所以,请小心使用,切记不可过多的使用。

这里要补充一下:去年曾经出现过【Padding Oracle Attack】这个话题, 一些人甚至错误的认为是Asp.net加密方式不安全!如果您也是这样认为的,那么可以看一下这篇文章: 浅谈这次ASP.NETPadding Oracle Attack相关内容 ,以消除这个错误的认识。当然了,我们也可以从这个话题得到一些收获:解密失败时,不要给出过多的提示,就当没有这个Cookie存在。

回到顶部

如何在C#发请的请求中使用Cookie

前面我们一直在谈服务端与浏览器中使用Cookie,其实浏览器也是一个普通的应用程序,.net framework也提供一些类也能让我们 直接发起HTTP请求,下面我们来看一下如何在C#发请的请求中使用Cookie ,其实也很简单,主要是使用了CookieContainer类,请看以下演示代码:

private static string SendHttpRequestGet(string url, Encoding encoding, 

            CookieContainer cookieContainer)

    {

        ifstring.IsNullOrEmpty(url) )

            throw new ArgumentNullException("url");

 

        if( encoding == null )

            throw new ArgumentNullException("encoding");

 

        HttpWebRequest request (HttpWebRequest)WebRequest.Create(url);

        request.Method "GET";

        request.CookieContainer cookieContainer;

        

        usingWebResponse response request.GetResponse() ) {

            usingStreamReader reader new StreamReader(response.GetResponseStream(), encoding) ) {

                return reader.ReadToEnd();

            }

        }

    }

 

    private void SendHttpDEMO()

    {

        StringBuilder sb new StringBuilder();

        CookieContainer cookieContainer new CookieContainer();

 

        string url "http://www.taobao.com";

        SendHttpRequestGet(url, Encoding.Default, cookieContainer);

 

        // 后面可以继续发起HTTP请求,此时将会包含上次从服务器写入的Cookie

        //SendHttpRequestGet("同域名下的其它URL", Encoding.Default, cookieContainer);

 

        // 至此,我们可以显示取得了哪些Cookie

        CookieCollection cookies cookieContainer.GetCookies(new Uri(url));

        if( cookies != null ) {

            foreach( System.Net.Cookie cookie in cookies )

                sb.AppendLine(cookie.ToString());

        }

        txtCookies.Text sb.ToString();

    }

 

回到顶部

重构与使用总结

在前面的Asp.net示例代码中,我一直使用.net提供的HttpCookie类来操作Cookie,是为了展示用原始的方式来使用Cookie, 这些代码有点重复,也有点繁琐, 为此,我提供了几个简单的方法可以更容易的使用Cookie,也算是对Cookie使用的一个总结。

/// <summary>

/// 用于方便使用Cookie的扩展工具类

/// </summary>

public static class CookieExtension

{

    // 我们可以为一些使用频率高的类型写专门的【读取】方法

 

    /// <summary>

    /// 从一个Cookie中读取字符串值。

    /// </summary>

    /// <param name="cookie"></param>

    /// <returns></returns>

    public static string GetString(this HttpCookie cookie)

    {

        if( cookie == null )

            return null;

 

        return cookie.Value;

    }

 

    /// <summary>

    /// 从一个Cookie中读取 Int 值。

    /// </summary>

    /// <param name="cookie"></param>

    /// <param name="defaultVal"></param>

    /// <returns></returns>

    public static int ToInt(this HttpCookie cookie, int defaultVal)

    {

        if( cookie == null )

            return defaultVal;

 

        return cookie.Value.TryToInt(defaultVal);

    }

 

    /// <summary>

    /// 从一个Cookie中读取值并转成指定的类型

    /// </summary>

    /// <typeparam name="T"></typeparam>

    /// <param name="cookie"></param>

    /// <returns></returns>

    public static T ConverTo<T>(this HttpCookie cookie)

    {

        if( cookie == null )

            return default(T);

 

        return (T)Convert.ChangeType(cookie.Value, typeof(T));

    }

 

    /// <summary>

    /// 从一个Cookie中读取【JSON字符串】值并反序列化成一个对象,用于读取复杂对象

    /// </summary>

    /// <typeparam name="T"></typeparam>

    /// <param name="cookie"></param>

    /// <returns></returns>

    public static T FromJson<T>(this HttpCookie cookie)

    {

        if( cookie == null )

            return default(T);

 

        return cookie.Value.FromJson<T>();

    }

 

 

    /// <summary>

    /// 将一个对象写入到Cookie

    /// </summary>

    /// <param name="obj"></param>

    /// <param name="name"></param>

    /// <param name="expries"></param>

    public static void WriteCookie(this object obj, string name, DateTimeexpries)

    {

        if( obj == null )

            throw new ArgumentNullException("obj");

 

        ifstring.IsNullOrEmpty(name) )

            throw new ArgumentNullException("name");

        

 

        HttpCookie cookie new HttpCookie(name, obj.ToString());

 

        if( expries.HasValue )

            cookie.Expires expries.Value;

 

        HttpContext.Current.Response.Cookies.Add(cookie);

    }

 

    /// <summary>

    /// 删除指定的Cookie

    /// </summary>

    /// <param name="name"></param>

    public static void DeleteCookie(string name)

    {

        ifstring.IsNullOrEmpty(name) )

            throw new ArgumentNullException("name");

 

        HttpCookie cookie new HttpCookie(name);

 

        // 删除Cookie,其实就是设置一个【过期的日期】

        cookie.Expires new DateTime(190011);

        HttpContext.Current.Response.Cookies.Add(cookie);

    }

}

 

 

更完整的代码可以从本文的示例代码中获得。(文章底部有下载地址)

使用方式:

public static class TestClass

{

    public static void Write()

    {

        string str "中国";

        int aa 25;

        DisplaySettings setting new DisplaySettings { Style 3, Size 50 };

        DateTime dt new DateTime(2012111200);

 

        str.WriteCookie("Key1"DateTime.Now.AddDays(1d));

        aa.WriteCookie("Key2"null);

        setting.ToJson().WriteCookie("Key3"null);

        dt.WriteCookie("Key4"null);

    }

 

    public static void Read()

    {

        HttpRequest request HttpContext.Current.Request;

 

        string str request.Cookies["Key1"].GetString();

        int num request.Cookies["Key2"].ToInt(0);

        DisplaySettings setting request.Cookies["Key3"].FromJson<DisplaySettings>();

        DateTime dt request.Cookies["Key4"].ConverTo<DateTime>();

    }    

}

 

 

注意哦:以上代码中都是直接使用字符串"Key"的形式,这种方式对于大一些的程序在后期可能会影响维护。
所以建议:将访问Cookie所使用的Key能有一个类来统一的定义,或者将读写操作包装成一些属性放在一个类中统一的管理。

public static class CookieValues

{

    // 建议把Cookie相关的参数放在一起,提供 get / set 属性(或者方法)来访问,以避免"key"到处乱写

 

    public static string AAA

    {

        get return HttpContext.Current.Request.Cookies["Key1"].GetString(); }

    }

    public static int BBB

    {

        get return HttpContext.Current.Request.Cookies["Key2"].ToInt(0); }

    }

    public static DisplaySettings CCC

    {

        get return HttpContext.Current.Request.Cookies["Key3"].FromJson<DisplaySettings>(); }

    }

    public static DateTime DDD

    {

        get return HttpContext.Current.Request.Cookies["Key4"].ConverTo<DateTime>(); }

    }

}

 

回到顶部

补充

根据一些朋友提供的反馈,这里再补充4个需要注意的地方:

1. 如果使用Form登录验证且希望使用Cookie方式时,建议设置 cookieless="UseCookies", 因为这个参数的默认值是:cookieless="UseDeviceProfile"Asp.net可能会误判。 dudu就吃过亏。

<authentication mode="Forms>

    <forms name="MyCookieNamecookieless="UseCookies"></forms>

</authentication>

 

2. Cookie3个属性,一般我们可以不用设置,但它们的值可以在Web.config中指定默认值:

<httpCookies domain="www.123.comhttpOnlyCookies="truerequireSSL="false"/>

 

3. 虽然在写Cookie时,我们可以设置name, value之外的其它属性,但是在读取时,是读不到这些设置的。 其实在我的示例代码中有体现,我前面也忘记了说明了。

4. HttpRequest.Cookies 与 HttpResponse.Cookies 会有关系(很奇怪吧)。
以下代码演示了这个现象:

protected void Page_Load(object sender, EventArgs e)

{

    DateTime.Now.ToString().WriteCookie("t1"null);

 

    label1.Text ShowAllCookies();

 

    Guid.NewGuid().ToString().WriteCookie("t2"null);

 

    // 如果去掉下面代码,将会看到2t1 

    Response.Cookies.Remove("t1");

    Response.Cookies.Remove("t2");

}

 

private string ShowAllCookies()

{

    StringBuilder sb new StringBuilder();

 

    forint 0; i Request.Cookies.Count; i++ ) {

        HttpCookie cookie Request.Cookies[i];

        sb.AppendFormat("{0}={1}<br />", cookie.Name, cookie.Value);

    }

 

    return sb.ToString();

}

 

上面的试验代码将会一直显示 t1 Cookie ,这里就不再贴图了。

 

 

 

 

 

 

 

 

 

 

 

 

 

HTTP协议 () Cookie

CookieHTTP协议中非常重要的东西, 之前拜读了Fish Li 写的【细说Cookie, 让我学到了很多东西。Fish的这篇文章写得太经典了。 所以我这篇文章就没有太多内容了。 

最近我打算写一个系列的HTTP文章,我站在HTTP协议的角度, 说说我对Cookie的理解。

 

阅读目录

· Cookie是什么,有什么用,为什么要用到Cookie

· Cookie的分类

· Cookie存在哪里

· 使用和禁用Cookie

· Fiddler查看HTTP中的Cookie

· 网站自动登陆的原理

· 截获Cookie,冒充别人身份

· Cookie和文件缓存的区别

· Cookie泄露隐私

· P3P协议

 

Cookie是什么,有什么用,为什么要用到Cookie

请看Fish Li 写的【细说Cookie

 

Cookie的分类

可以大致把Cookie分为2类: 回话cookie和持久cookie

会话cookie: 是一种临时的cookie,它记录了用户访问站点时的设置和偏好,关闭浏览器,会话cookie就被删除了

持久cookie: 存储在硬盘上,(不管浏览器退出,或者电脑重启,持久cookie都存在), 持久cookie有过期时间

 

Cookie存在哪里

Cookie是存在硬盘上,  IEcookie的地方和Firefoxcookie的地方不一样。  不同的操作系统也可能存cookie的地方不一样。

不同的浏览器会在各自的独立空间存放Cookie, 互不干涉

以我的windows7, IE8为例,  cookie存在这: C:\Users\xiaoj\AppData\Local\Microsoft\Windows\Temporary Internet Files

注意: 缓存文件和cookie文件,是存在一起的都在这个目录下。

你也可以这样找打开IE,点击Tools->Internet Options->General Tab下的->Browsing history下的Setting按钮,弹出的对话框中点击View files.

 

不同的网站会有不同的cookie文件

 

使用和禁用Cookie

IE:   工具->Internet 选项 -> 隐私

 

Fiddler查看HTTP中的Cookie

浏览器把cookie通过HTTP Request 中的“Cookie: header”发送给Web服务器

Web服务器通过HTTP Response中的"Set-Cookie: header"cookie发送给浏览器

使用Fiddler可以清楚地看到cookieHTTP中传递。 Fiddler工具中可以清晰的看到Http Request 中的Cookie, 和Http Response中的cookie  

实例: 启动Fiddler,  启动浏览器访问一些购物网站,就可以看到。

 

 

网站自动登陆的原理

我们以”博客园自动登陆“的例子,来说明cookie是如何传递的。

大家知道博客园是可以自动登陆的。 如下图,这个是什么原理呢?

假如我已经在登陆页面输入了用户名,密码,选择了保存密码,登陆。

这时候,其实在你的机器上保存好了登陆的cookie, 不信你可以按照上节介绍方法去你的电脑上找下博客园的cookie  

当我下次访问博客园流程如下。

1. 用户打开IE浏览器,在地址栏上输入www.cnblogs.com.

2. IE首先会在硬盘中查找关于cnblogs.comcookie. 然后把cookie放到HTTP Request中,再把Request发给Web服务器。

3. Web服务器返回博客园首页(你会看到你已经登陆了)。

 

截获Cookie,冒充别人身份

通过上面这个例子,可以看到cookie是很重要的,识别是否是登陆用户,就是通过cookie。  假如截获了别人的cookie是否可以冒充他人的身份登陆呢?  当然可以, 这就是一种黑客技术叫Cookie欺骗。

利用Cookie 欺骗, 不需要知道用户名密码。就可以直接登录,使用别人的账户做坏事。

我知道有两种方法可以截获他人的cookie

1. 通过XSS脚步攻击, 获取他人的cookie. 具体原理可以看 [Web安全性测试之XSS]

2. 想办法获取别人电脑上保存的cookie文件(这个比较难)

 

拿到cookie后,就可以冒充别人的身份了。 这个过程我就不演示了。

Cookie和文件缓存的区别

很多人会把cookie和文件缓存弄混淆, 这两个完全是不一样的东西。唯一的相同之处可能是它们俩都存在硬盘上,而且是存在同一个文件夹下。

关于HTTP缓存请看这 【HTTP协议之缓存

我们在IE中可以选择分别删除Cookie和缓存文件

 

 

Cookie 泄露隐私

2013年央视的315晚会上, 曝光了很多不法公司利用Cookie跟踪并采集用户的个人信息,并转卖给网络广告商,形成了一条窃取用户信息的灰色产业链。从而实现广告准确投放。严重干扰了用户的正常网络应用,侵害了个人的隐私和利益。

我经常就在门户网站上发现广告位上显示的是我在电商网站上流量过的商品。  这就是我的cookie被泄露了。

 

目前在欧洲, 已经对Cookie立法, 如果网站需要保存用户的cookie, 必须弹出一个对话框,要用户确认后才能保存Cookie.

 

P3P协议

从上面看来, Cookie 是一个比较容易泄露用户隐私和危险的东西。  有没有办法保护个人用户隐私呢?    那就是P3P协议

P3P是一种被称为个人隐私安全平台项目(the Platform for Privacy Preferences)的标准,能够保护在线隐私权,使Internet冲浪者可以选择在浏览网页时,是否被第三方收集并利用自己的个人信息。如果一个 站点不遵守P3P标准的话,那么有关它的Cookies将被自动拒绝,并且P3P还能够自动识破多种Cookies的嵌入方式。p3p是由全球资讯联盟网 所开发的。

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值