为什么foreach(HttpCookie cookie in Request.Cookies)会出错

本文探讨了在ASP.NET中使用foreach遍历Request.Cookies时遇到的问题及原因分析。由于HttpCookieCollection继承自NameObjectCollectionBase,导致类型转换失败。文章深入剖析了问题根源,并提供了正确的遍历方法。

原文: http://www.cnblogs.com/dudu/archive/2004/12/21/80118.html

原文的评论也很有价值

   第一次使用foreach(HttpCookie cookie in Request.Cookies)的时候, 我怎么也没想到它会出错,错误信息竟然是“指定的转换无效。”。Request.Cookies的类型是HttpCookieCollection,怎么会出错呢?HttpCookieCollection难道有与众不同的地方?
    既然不能转换,那cookie究竟是什么类型?我们用代码测试一下:
   

foreach ( object  cookie  in  Request.Cookies)
ExpandedBlockStart.gifContractedBlock.gif            
{
                Response.Write(cookie.GetType().ToString()
+"<br>");
            }


原来cookie全变成了System.String类型, 上面的cookie变量的值就是Request.Cookies中每个Cookie的Name。下面的两段代码,结果是一样的:
 

foreach ( object  cookie  in  Request.Cookies)
ExpandedBlockStart.gifContractedBlock.gif            
{
                Response.Write(cookie
+"<br>");
            }


 

foreach ( string  cookie  in  Request.Cookies.AllKeys)
ExpandedBlockStart.gifContractedBlock.gif            
{
                Response.Write(cookie
+"<br>");
            }

 

 


那我们如何枚举Request.Cookies?通过索引,我也不用多说了,大家都知道。  

 

for ( int  i = 0 ;i < Request.Cookies.Count;i ++ )
ExpandedBlockStart.gifContractedBlock.gif            
{
                Response.Write(Request.Cookies[i].Name
+":"+Request.Cookies[i].Value+"<br>");
            }

 


而Syste.Net.CookieCollection不存在这个问题,那HttpCookieCollection与CookieCollection有什么区别呢?我们比较一下它们的基类与接口:
HttpCookieCollection 继承了NameObjectCollectionBase,NameObjectCollectionBase实现了ICollection, IEnumerable, ISerializable, IDeserializationCallback接口
CookieCollection实现了ICollection, IEnumerable接口

我想应该在NameObjectCollectionBase中可以找到答案。
首先我们复习一下foreach的工作原理,先看下面的代码:

 

 

namespace  Test
ExpandedBlockStart.gifContractedBlock.gif
{
class Class1
ExpandedSubBlockStart.gifContractedSubBlock.gif
{
   
static void Main(string[] args)
ExpandedSubBlockStart.gifContractedSubBlock.gif   
{
      
ArrayList array = new ArrayList();
      array.Add(
"A");
      array.Add(
"B");
      array.Add(
"C");

      
foreach (string item in array)
ExpandedSubBlockStart.gifContractedSubBlock.gif      
{
         Console.WriteLine(item);
      }

   }

}


在编译的时候,C#编辑器会对每一个foreach 区域进行转换,转换成下面的代码:


 

IEnumerator enumerator  =  array.GetEnumerator();
try  
ExpandedBlockStart.gifContractedBlock.gif
{
   
string item;
   
while (enumerator.MoveNext()) 
ExpandedSubBlockStart.gifContractedSubBlock.gif   
{
      item 
= (string) enumerator.Current;
      Console.WriteLine(item);
   }

}

finally  
ExpandedBlockStart.gifContractedBlock.gif
{
   IDisposable d 
= enumerator as IDisposable;
   
if (d != null) d.Dispose();
}


对于foreach(HttpCookie cookie in Request.Cookies), 转换的结果应该是这样:

IEnumerator enumerator  =  HttpCookieCollection.GetEnumerator();
try  
ExpandedBlockStart.gifContractedBlock.gif
{
   HttpCookie item;
   
while (enumerator.MoveNext()) 
ExpandedSubBlockStart.gifContractedSubBlock.gif   
{
      item 
= (HttpCookie) enumerator.Current;
    }

}

finally  
ExpandedBlockStart.gifContractedBlock.gif
{
   IDisposable d 
= enumerator as IDisposable;
   
if (d != null) d.Dispose();
}

 


因为上面的enumerator.Current返回的是System.String类型,所以会出现“指定的转换无效”的错误。
用Reflector查看NameObjectCollectionBase的源代码,可以发现 HttpCookieCollection.GetEnumerator()调用的是NameObjectCollectionBase中的GetEnumerator(),而GetEnumerator()返回的是NameObjectCollectionBase内部的一个类NameObjectKeysEnumerator的实例,这个类实现了IEnumerator。 
上面的enumerator.Current实际上是调用的NameObjectKeysEnumerator的Current属性,而Current属性中又调用了NameObjectCollectionBase的BaseGetKey。头都昏了! 为了找到问题的真正原因,只能继续。BaseGetKey的代码是这样的:

 

 

protected   string  BaseGetKey( int  index)
ExpandedBlockStart.gifContractedBlock.gif
{
      NameObjectCollectionBase.NameObjectEntry entry1 
= (NameObjectCollectionBase.NameObjectEntry) this._entriesArray[index];
      
return entry1.Key;
}

 

 


答案终于找到,NameObjectCollectionBase返回的是NameObjectEntry(这也是NameObjectCollectionBase的一个内部类)的一个成员Key, 这个Key就是string类型的。
问题的原因虽然找到,但为什么要这样设计,暂时还搞不明白,只能等以后慢慢研究。当然,如果有高人解惑就最好了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值