C#獲取URL引數值

CHCH998發表於2018-07-28
原文地址為:C#獲取URL引數值

在寫程式的時候,我們經常需要對頁面進行傳引數,比如page?id=1234,那麼在page這個頁面中就直接可以使用string id = Request.QueryString["id"]; 來獲取引數id的值1234了。這是一個人人都知道的基礎知識。

上面的方法:Request.QueryString,它會把傳入的URL進行分析,並把結果儲存在一個鍵值(key value)的Collection中,我們只要通過設定key值,就可以返回這個key所對應的value了。

假設這個URL不是通過請求進來的,我們是沒有辦法通過Request來獲取URL的引數值的,我們是否可以通過一個字串string url = "http://www.google.com/page?id=1234"中分析出Collection[key]=value呢?而且很多時候我們的URL的引數都是經過UrlEncode編碼的,這個編碼通常來說會是Encoding.UTF8或者Encoding.GetEncoding("gb2312"),需要提醒你的是:Baidu的URL是使用gb2312的,而Google是使用UTF8的。那對於傳入的字串我們如何確認是該使用gb2312還是UTF8來解碼呢?

在Baidu或者Google搜尋到相關的解決辦法中,你會看到這樣的說法,那就是通過正規表示式去匹配URL,例如:Regex urlRegex = new Regex(@"(?:^|\?|&)courseid=(\d*)(?:&|$)"); 但是我比較懷疑通過這個方法分析出來的資料完整性,它有辦法確保匹配出所有資料嗎?後來我想到了模擬Microsoft .NET Framework中Request的方法來解決這個問題。其實上面的string id = Request.QueryString["id"]; 也可以寫成 System.Collections.Specialized.NameValueCollection col = Request.QueryString; string strID = col[”id”];所以讓我們來構造這個NameValueCollection吧!


程式碼部分 

///   <summary>
///  測試.
///   </summary>
public   void  Test()
{
    
string  pageURL  =   " http://www.google.com.hk/search?hl=zh-CN&source=hp&q=%E5%8D%9A%E6%B1%87%E6%95%B0%E7%A0%81&aq=f&aqi=g2&aql=&oq=&gs_rfai= " ;
    Uri uri 
=   new  Uri(pageURL);
    
string  queryString  =  uri.Query;
    NameValueCollection col 
=  GetQueryString(queryString);
    
string  searchKey  =  col[ " q " ];
    
// 結果 searchKey = "博彙數碼"
}    

///   <summary>
///  將查詢字串解析轉換為名值集合.
///   </summary>
///   <param name="queryString"></param>
///   <returns></returns>
public   static  NameValueCollection GetQueryString( string  queryString)
{
    
return  GetQueryString(queryString,  null true );
}

///   <summary>
///  將查詢字串解析轉換為名值集合.
///   </summary>
///   <param name="queryString"></param>
///   <param name="encoding"></param>
///   <param name="isEncoded"></param>
///   <returns></returns>
public   static  NameValueCollection GetQueryString( string  queryString, Encoding encoding,  bool  isEncoded)
{
    queryString 
=  queryString.Replace( " ? " "" );
    NameValueCollection result 
=   new  NameValueCollection(StringComparer.OrdinalIgnoreCase);
    
if  ( ! string .IsNullOrEmpty(queryString))
    {
        
int  count  =  queryString.Length;
        
for  ( int  i  =   0 ; i  <  count; i ++ )
        {
            
int  startIndex  =  i;
            
int  index  =   - 1 ;
            
while  (i  <  count)
            {
                
char  item  =  queryString[i];
                
if  (item  ==   ' = ' )
                {
                    
if  (index  <   0 )
                    {
                        index 
=  i;
                    }
                }
                
else   if  (item  ==   ' & ' )
                {
                    
break ;
                }
                i
++ ;
            }
            
string  key  =   null ;
            
string  value  =   null ;
            
if  (index  >=   0 )
            {
                key 
=  queryString.Substring(startIndex, index  -  startIndex);
                value 
=  queryString.Substring(index  +   1 , (i  -  index)  -   1 );
            }
            
else
            {
                key 
=  queryString.Substring(startIndex, i  -  startIndex);
            }
            
if  (isEncoded)
            {
                result[MyUrlDeCode(key, encoding)] 
=  MyUrlDeCode(value, encoding);                    
            }
            
else
            {
                result[key] 
=  value;
            }
            
if  ((i  ==  (count  -   1 ))  &&  (queryString[i]  ==   ' & ' ))
            {
                result[key] 
=   string .Empty;
            }
        }
    }
    
return  result;
}

///   <summary>
///  解碼URL.
///   </summary>
///   <param name="encoding"> null為自動選擇編碼 </param>
///   <param name="str"></param>
///   <returns></returns>
public   static   string  MyUrlDeCode( string  str, Encoding encoding)
{
    
if  (encoding  ==   null )
    {
        Encoding utf8 
=  Encoding.UTF8;
        
// 首先用utf-8進行解碼                     
         string  code  =  HttpUtility.UrlDecode(str.ToUpper(), utf8);
        
// 將已經解碼的字元再次進行編碼.
         string  encode  =  HttpUtility.UrlEncode(code, utf8).ToUpper();
        
if  (str  ==  encode)
            encoding 
=  Encoding.UTF8;
        
else
            encoding 
=  Encoding.GetEncoding( " gb2312 " );
    }
    
return  HttpUtility.UrlDecode(str, encoding);
}

 

說明 

  1. 對於下面的編碼方式中,Baidu是使用gb2312,而Google是使用UTF8的,所以這些編碼如何解釋呢?解決辦法就是如MyUrlDeCode方法所寫的,把編碼進行一次解碼,再把解碼後的字串進行編碼,再對字串進行比較,如果是相同的,那就是使用了這種編碼的。還要注意進行ToUpper字串。
  2. // 博彙數碼.
    string  baidu  =  HttpUtility.UrlDecode( " wd=%B2%A9%BB%E3%CA%FD%C2%EB " );
    string  google  =  HttpUtility.UrlDecode( " q=%E5%8D%9A%E6%B1%87%E6%95%B0%E7%A0%81 " ); 
  3. 關於return GetQueryString(queryString, nulltrue); 如果你傳入的是null,那就是表示你無法確認編碼是gb2312還是UTF8,當你確認編碼的,可以直接傳入編碼型別,比如return GetQueryString(queryString, Encoding.UTF8true); 
  4. 需要注意queryString = queryString.Replace("?""");把?給過濾掉。
  5. %B2%A9%BB%E3%CA%FD%C2%EB = %B2%A9 %BB%E3 %CA%FD %C2%EB(博彙數碼gb2312)
  6. %E5%8D%9A%E6%B1%87%E6%95%B0%E7%A0%81 = %E5%8D%9A %E6%B1%87 %E6%95%B0 %E7%A0%81(博彙數碼utf-8)

轉載請註明本文地址:C#獲取URL引數值

相關文章