本文主要是介绍c# HttpWebRequest与HttpWebResponse,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
c# HttpWebRequest与HttpWebResponse 绝技如果你想做一些,抓取,或者是自动获取的功能,那么就跟我一起来学习一下Http请求吧。
本文章会对Http请求时的Get和Post方式进行详细的说明,
在请求时的参数怎么发送,怎么带Cookie,怎么设置证书,怎么解决 编码等问题,进行一步一步的解决。
* 如果要使用中间的方法的话,可以访问我的帮助类完全免费开源:
这个类是专门为HTTP的GET和POST请求写的,解决了编码,证书,自动带Cookie等问题。
C# HttpHelper,帮助类,真正的Httprequest请求时无视编码,无视证书,无视Cookie,网页抓取
1.第一招,根据URL地址获取网页信息
先来看一下代码
get方法
普通浏览 复制代码
- public static string GetUrltoHtml ( string Url, string type )
- {
- try
- {
- System. Net.WebRequest wReq = System. Net.WebRequest.Create (Url ) ;
- // Get the response instance.
- System. Net.WebResponse wResp = wReq.GetResponse ( ) ;
- System. IO.Stream respStream = wResp.GetResponseStream ( ) ;
- // Dim reader As StreamReader = New StreamReader(respStream)
- using ( System. IO.StreamReader reader = new System. IO.StreamReader (respStream, Encoding.GetEncoding (type ) ) )
- {
- return reader.ReadToEnd ( ) ;
- }
- }
- catch ( System.Exception ex )
- {
- //errorMsg = ex.Message;
- }
- return "" ;
- }
post方法
普通浏览 复制代码
- ///<summary>
- ///采用https协议访问网络
- ///</summary>
- ///<param name="URL">url地址</param>
- ///<param name="strPostdata">发送的数据</param>
- ///<returns></returns>
- public string OpenReadWithHttps ( string URL, string strPostdata, string strEncoding )
- {
- Encoding encoding = Encoding.Default ;
- HttpWebRequest request = (HttpWebRequest )WebRequest.Create (URL ) ;
- request.Method = "post" ;
- request.Accept = "text/html, application/xhtml+xml, */*" ;
- request.ContentType = "application/x-www-form-urlencoded" ;
- byte [ ] buffer = encoding.GetBytes (strPostdata ) ;
- request.ContentLength = buffer.Length ;
- request.GetRequestStream ( ).Write (buffer, 0, buffer.Length ) ;
- HttpWebResponse response = (HttpWebResponse )request.GetResponse ( ) ;
- using ( StreamReader reader = new StreamReader (response.GetResponseStream ( ), System. Text.Encoding.GetEncoding (strEncoding ) ) )
- {
- return reader.ReadToEnd ( ) ;
- }
- }
这招是入门第一式, 特点:
1.最简单最直观的一种,入门课程。
2.适应于明文,无需登录,无需任何验证就可以进入的页面。
3.获取的数据类型为HTML文档。
4.请求方法为Get/Post
2.第二招,根据URL地址获取需要验证证书才能访问的网页信息
先来看一下代码
get方法
普通浏览 复制代码
- /回调验证证书问题
- public bool CheckValidationResult ( object sender, X509Certificate certificate, X509Chain chain, SslPolicyErrors errors )
- {
- // 总是接受
- return true ;
- }
-
- /// <summary>
- /// 传入URL返回网页的html代码
- /// </summary>
- /// <param name="Url">URL</param>
- /// <returns></returns>
- public string GetUrltoHtml ( string Url )
- {
- StringBuilder content = new StringBuilder ( ) ;
-
- try
- {
- //这一句一定要写在创建连接的前面。使用回调的方法进行证书验证。
- ServicePointManager.ServerCertificateValidationCallback = new System. Net. Security.RemoteCertificateValidationCallback (CheckValidationResult ) ;
-
- // 与指定URL创建HTTP请求
- HttpWebRequest request = (HttpWebRequest )WebRequest.Create (Url ) ;
-
- //创建证书文件
- X509Certificate objx509 = new X509Certificate ( Application.StartupPath + "\\123.cer" ) ;
-
- //添加到请求里
- request.ClientCertificates.Add (objx509 ) ;
-
- // 获取对应HTTP请求的响应
- HttpWebResponse response = (HttpWebResponse )request.GetResponse ( ) ;
- // 获取响应流
- Stream responseStream = response.GetResponseStream ( ) ;
- // 对接响应流(以"GBK"字符集)
- StreamReader sReader = new StreamReader (responseStream, Encoding.GetEncoding ( "utf-8" ) ) ;
- // 开始读取数据
- Char [ ] sReaderBuffer = new Char [ 256 ] ;
- int count = sReader.Read (sReaderBuffer, 0, 256 ) ;
- while (count > 0 )
- {
- String tempStr = new String (sReaderBuffer, 0, count ) ;
- content.Append (tempStr ) ;
- count = sReader.Read (sReaderBuffer, 0, 256 ) ;
- }
- // 读取结束
- sReader.Close ( ) ;
- }
- catch (Exception )
- {
- content = new StringBuilder ( "Runtime Error" ) ;
- }
-
- return content.ToString ( ) ;
- }
post方法
普通浏览 复制代码
- //回调验证证书问题
- public bool CheckValidationResult ( object sender, X509Certificate certificate, X509Chain chain, SslPolicyErrors errors )
- {
- // 总是接受
- return true ;
- }
-
- ///<summary>
- ///采用https协议访问网络
- ///</summary>
- ///<param name="URL">url地址</param>
- ///<param name="strPostdata">发送的数据</param>
- ///<returns></returns>
- public string OpenReadWithHttps ( string URL, string strPostdata, string strEncoding )
- {
- // 这一句一定要写在创建连接的前面。使用回调的方法进行证书验证。
- ServicePointManager.ServerCertificateValidationCallback = new System. Net. Security.RemoteCertificateValidationCallback (CheckValidationResult ) ;
- Encoding encoding = Encoding.Default ;
- HttpWebRequest request = (HttpWebRequest )WebRequest.Create (URL ) ;
-
- //创建证书文件
- X509Certificate objx509 = new X509Certificate ( Application.StartupPath + "\\123.cer" ) ;
-
- //加载Cookie
- request.CookieContainer = new CookieContainer ( ) ;
-
- //添加到请求里
- request.ClientCertificates.Add (objx509 ) ;
- request.Method = "post" ;
- request.Accept = "text/html, application/xhtml+xml, */*" ;
- request.ContentType = "application/x-www-form-urlencoded" ;
- byte [ ] buffer = encoding.GetBytes (strPostdata ) ;
- request.ContentLength = buffer.Length ;
- request.GetRequestStream ( ).Write (buffer, 0, buffer.Length ) ;
- HttpWebResponse response = (HttpWebResponse )request.GetResponse ( ) ;
- using (StreamReader reader = new StreamReader (response.GetResponseStream ( ), System. Text.Encoding.GetEncoding (strEncoding ) ) )
- {
- return reader.ReadToEnd ( ) ;
- }
- }
这招是学会算是进了大门了,凡是需要验证证书才能进入的页面都可以使用这个方法进入,我使用的是证书回调验证的方式,证书验证是否通过在客户端验证,这样的话我们就可以使用自己定义一个方法来验证了,有的人会说那也不清楚是怎么样验证的啊,其它很简单,代码是自己写的为什么要那么难为自己呢,直接返回一个True不就完了,永远都是验证通过,这样就可以无视证书的存在了, 特点:
1.入门前的小难题,初级课程。
2.适应于无需登录,明文但需要验证证书才能访问的页面。
3.获取的数据类型为HTML文档。
4.请求方法为Get/Post
3.第三招,根据URL地址获取需要登录才能访问的网页信息
我们先来分析一下这种类型的网页,需要登录才能访问的网页,其它呢也是一种验证,验证什么呢,验证客户端是否登录,是否具用相应的凭证,需要登录的都要验证SessionID这是每一个需要登录的页面都需要验证的,那我们怎么做的,我们第一步就是要得存在Cookie里面的数据包括SessionID,那怎么得到呢,这个方法很多,使用ID9或者是火狐浏览器很容易就能得到,可以参考我的文章
提供一个网页抓取hao123手机号码归属地的例子 这里面针对ID9有详细的说明。
如果我们得到了登录的Cookie信息之后那个再去访问相应的页面就会非常的简单了,其它说白了就是把本地的Cookie信息在请求的时候捎带过去就行了。
看代码
get方法
普通浏览 复制代码
- /// <summary>
- /// 传入URL返回网页的html代码带有证书的方法
- /// </summary>
- /// <param name="Url">URL</param>
- /// <returns></returns>
- public string GetUrltoHtml ( string Url )
- {
- StringBuilder content = new StringBuilder ( ) ;
- try
- {
- // 与指定URL创建HTTP请求
- HttpWebRequest request = (HttpWebRequest )WebRequest.Create (Url ) ;
- request.UserAgent = "Mozilla/5.0 (compatible; MSIE 9.0; Windows NT 6.1; Trident/5.0; BOIE9;ZHCN)" ;
- request.Method = "GET" ;
- request.Accept = "*/*" ;
- //如果方法验证网页来源就加上这一句如果不验证那就可以不写了
- request.Referer = "http://sufei.cnblogs.com" ;
- CookieContainer objcok = new CookieContainer ( ) ;
- objcok.Add ( new Uri ( "http://sufei.cnblogs.com" ), new Cookie ( "键", "值" ) ) ;
- objcok.Add ( new Uri ( "http://sufei.cnblogs.com" ), new Cookie ( "键", "值" ) ) ;
- objcok.Add ( new Uri ( "http://sufei.cnblogs.com" ), new Cookie ( "sidi_sessionid", "360A748941D055BEE8C960168C3D4233" ) ) ;
- request.CookieContainer = objcok ;
-
- //不保持连接
- request.KeepAlive = true ;
-
- // 获取对应HTTP请求的响应
- HttpWebResponse response = (HttpWebResponse )request.GetResponse ( ) ;
-
- // 获取响应流
- Stream responseStream = response.GetResponseStream ( ) ;
-
- // 对接响应流(以"GBK"字符集)
- StreamReader sReader = new StreamReader (responseStream, Encoding.GetEncoding ( "gb2312" ) ) ;
-
- // 开始读取数据
- Char [ ] sReaderBuffer = new Char [ 256 ] ;
- int count = sReader.Read (sReaderBuffer, 0, 256 ) ;
- while (count > 0 )
- {
- String tempStr = new String (sReaderBuffer, 0, count ) ;
- content.Append (tempStr ) ;
- count = sReader.Read (sReaderBuffer, 0, 256 ) ;
- }
- // 读取结束
- sReader.Close ( ) ;
- }
- catch (Exception )
- {
- content = new StringBuilder ( "Runtime Error" ) ;
- }
-
- return content.ToString ( ) ;
- }
post方法。
普通浏览 复制代码
- ///<summary>
- ///采用https协议访问网络
- ///</summary>
- ///<param name="URL">url地址</param>
- ///<param name="strPostdata">发送的数据</param>
- ///<returns></returns>
- public string OpenReadWithHttps ( string URL, string strPostdata )
- {
- Encoding encoding = Encoding.Default ;
- HttpWebRequest request = (HttpWebRequest )WebRequest.Create (URL ) ;
- request.Method = "post" ;
- request.Accept = "text/html, application/xhtml+xml, */*" ;
- request.ContentType = "application/x-www-form-urlencoded" ;
- CookieContainer objcok = new CookieContainer ( ) ;
- objcok.Add ( new Uri ( "http://sufei.cnblogs.com" ), new Cookie ( "键", "值" ) ) ;
- objcok.Add ( new Uri ( "http://sufei.cnblogs.com" ), new Cookie ( "键", "值" ) ) ;
- objcok.Add ( new Uri ( "http://sufei.cnblogs.com" ), new Cookie ( "sidi_sessionid", "360A748941D055BEE8C960168C3D4233" ) ) ;
- request.CookieContainer = objcok ;
-
- byte [ ] buffer = encoding.GetBytes (strPostdata ) ;
- request.ContentLength = buffer.Length ;
-
- request.GetRequestStream ( ).Write (buffer, 0, buffer.Length ) ;
- HttpWebResponse response = (HttpWebResponse )request.GetResponse ( ) ;
- StreamReader reader = new StreamReader (response.GetResponseStream ( ), System. Text.Encoding.GetEncoding ( "utf-8" ) ) ;
- return reader.ReadToEnd ( ) ;
- }
特点:
1.还算有点水类型的,练习成功后可以小牛一把。
2.适应于需要登录才能访问的页面。
3.获取的数据类型为HTML文档。
4.请求方法为Get/Post
总结一下,其它基本的技能就这几个部分,如果再深入的话那就是基本技能的组合了
比如,
1. 先用Get或者Post方法登录然后取得Cookie再去访问页面得到信息,这种其它也是上面技能的组合,
这里需要以请求后做这样一步
这就是在你请求后可以得到当次Cookie的方法,直接取得返回给上一个方法使用就行了,上面我们都是自己构造的,在这里直接使用这个Cookie就可以了。普通浏览 复制代码
response.Cookies
2.如果我们碰到需要登录而且还要验证证书的网页怎么办,其它这个也很简单把我们上面的方法综合 一下就行了
如下代码这里我以Get为例子Post例子也是同样的方法
普通浏览 复制代码
- /// <summary>
- /// 传入URL返回网页的html代码
- /// </summary>
- /// <param name="Url">URL</param>
- /// <returns></returns>
- public string GetUrltoHtml ( string Url )
- {
- StringBuilder content = new StringBuilder ( ) ;
-
- try
- {
- //这一句一定要写在创建连接的前面。使用回调的方法进行证书验证。
- ServicePointManager.ServerCertificateValidationCallback = new System. Net. Security.RemoteCertificateValidationCallback (CheckValidationResult ) ;
-
- // 与指定URL创建HTTP请求
- HttpWebRequest request = (HttpWebRequest )WebRequest.Create (Url ) ;
-
- //创建证书文件
- X509Certificate objx509 = new X509Certificate ( Application.StartupPath + "\\123.cer" ) ;
-
- //添加到请求里
- request.ClientCertificates.Add (objx509 ) ;
-
- CookieContainer objcok = new CookieContainer ( ) ;
- objcok.Add ( new Uri ( "http://www.cnblogs.com" ), new Cookie ( "键", "值" ) ) ;
- objcok.Add ( new Uri ( "http://www.cnblogs.com" ), new Cookie ( "键", "值" ) ) ;
- objcok.Add ( new Uri ( "http://www.cnblogs.com" ), new Cookie ( "sidi_sessionid", "360A748941D055BEE8C960168C3D4233" ) ) ;
- request.CookieContainer = objcok ;
-
- // 获取对应HTTP请求的响应
- HttpWebResponse response = (HttpWebResponse )request.GetResponse ( ) ;
- // 获取响应流
- Stream responseStream = response.GetResponseStream ( ) ;
- // 对接响应流(以"GBK"字符集)
- StreamReader sReader = new StreamReader (responseStream, Encoding.GetEncoding ( "utf-8" ) ) ;
- // 开始读取数据
- Char [ ] sReaderBuffer = new Char [ 256 ] ;
- int count = sReader.Read (sReaderBuffer, 0, 256 ) ;
- while (count > 0 )
- {
- String tempStr = new String (sReaderBuffer, 0, count ) ;
- content.Append (tempStr ) ;
- count = sReader.Read (sReaderBuffer, 0, 256 ) ;
- }
- // 读取结束
- sReader.Close ( ) ;
- }
- catch (Exception )
- {
- content = new StringBuilder ( "Runtime Error" ) ;
- }
-
- return content.ToString ( ) ;
-
- }
3.如果我们碰到那种需要验证网页来源的方法应该怎么办呢,这种情况其它是有些程序员会想到你可能会使用程序,自动来获取网页信息,为了防止就使用页面来源来验证,就是说只要不是从他们所在页面或是域名过来的请求就不接受,有的是直接验证来源的IP,这些都可以使用下面一句来进入,这主要是这个地址是可以直接伪造的
普通浏览 复制代码
- request.Referer = <a href= "http://sufei.cnblogs.com"><a href=\ "http://sufei.cnblogs.com/\" target=\"_blank\">http://sufei.cnblogs.com</a></a>;
呵呵其它很简单因为这个地址可以直接修改。但是如果服务器上验证的是来源的URL那就完了,我们就得去修改数据包了,这个有点难度暂时不讨论。
4.提供一些与这个例子相配置的方法
过滤HTML标签的方法
URL转化的方法普通浏览 复制代码
/// <summary> /// 过滤html标签 /// </summary> /// <param name="strHtml ">html的内容</param> /// <returns></returns> public static string StripHTML(string stringToStrip) { // paring using RegEx // stringToStrip = Regex.Replace(stringToStrip, "</p (?:\\s* )> (?:\\s* )<p (?:\\s* )> ", "\n\n ", RegexOptions.IgnoreCase | RegexOptions.Compiled); stringToStrip = Regex.Replace(stringToStrip, " ", "\n ", RegexOptions.IgnoreCase | RegexOptions.Compiled); stringToStrip = Regex.Replace(stringToStrip, "\ "", "''", RegexOptions.IgnoreCase | RegexOptions.Compiled ) ; stringToStrip = StripHtmlXmlTags (stringToStrip ) ; return stringToStrip ; } private static string StripHtmlXmlTags ( string content ) { return Regex.Replace (content, "<[^>]+>", "", RegexOptions.IgnoreCase | RegexOptions.Compiled ) ; }
普通浏览 复制代码
- #region 转化 URL
-
- public static string URLDecode ( string text )
- {
- return HttpUtility.UrlDecode (text, Encoding.Default ) ;
- }
-
- public static string URLEncode ( string text )
- {
- return HttpUtility.UrlEncode (text, Encoding.Default ) ;
- }
-
- #endregion
提供一个实际例子,这个是使用IP138来查询手机号码归属地的方法,其它在我的上一次文章里都有,在这里我再放上来是方便大家阅读,这方面的技术其它研究起来很有意思,希望大家多提建议,我相信应该还有更多更好,更完善的方法,在这里给大家提供一个参考吧。感谢支持
上例子
普通浏览 复制代码
- /// <summary>
- /// 输入手机号码得到归属地信息
- /// </summary>
- /// <param name="number">手机号码</param>
- /// <returns>数组类型0为归属地,1卡类型,2区 号,3邮 编</returns>
- public static string [ ] getTelldate ( string number )
- {
- try
- {
- string strSource = GetUrltoHtml ( "http://www.ip138.com:8080/search.asp?action=mobile&mobile=" + number.Trim ( ) ) ;
- //归属地
- strSource = strSource.Substring (strSource.IndexOf (number ) ) ;
- strSource = StripHTML (strSource ) ;
- strSource = strSource.Replace ( "\r", "" ) ;
- strSource = strSource.Replace ( "\n", "" ) ;
- strSource = strSource.Replace ( "\t", "" ) ;
- strSource = strSource.Replace ( " ", "" ) ;
- strSource = strSource.Replace ( "-->", "" ) ;
- string [ ] strnumber = strSource.Split ( new string [ ] { "归属地", "卡类型", "邮 编", "区 号", "更详细", "卡号" }, StringSplitOptions.RemoveEmptyEntries ) ;
- string [ ] strnumber1 = null ;
- if (strnumber.Length > 4 )
- {
- strnumber1 = new string [ ] { strnumber [ 1 ].Trim ( ), strnumber [ 2 ].Trim ( ), strnumber [ 3 ].Trim ( ), strnumber [ 4 ].Trim ( ) } ;
- }
- return strnumber1 ;
- }
- catch (Exception )
- {
- return null ;
- }
- }
这个例子写是不怎么样,些地方是可以简化的,这个接口而且可以直接使用Xml得到,但我在这里的重点是让一些新手看看方法和思路风凉啊,呵呵
第四招,通过Socket访问
普通浏览 复制代码
///<summary> /// 请求的公共类用来向服务器发送请求 ///</summary> ///<param name="strSMSRequest">发送请求的字符串</param> ///<returns>返回的是请求的信息</returns> private static string SMSrequest ( string strSMSRequest ) { byte [ ] data = new byte [ 1024 ] ; string stringData = null ; IPHostEntry gist = Dns.GetHostByName ( "www.110.cn" ) ; IPAddress ip = gist.AddressList [ 0 ] ; //得到IP IPEndPoint ipEnd = new IPEndPoint (ip, 3121 ) ; //默认80端口号 Socket socket = new Socket (AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp ) ; //使用tcp协议 stream类型 try { socket.Connect (ipEnd ) ; } catch (SocketException ex ) { return "Fail to connect server\r\n" + ex.ToString ( ) ; } string path = strSMSRequest.ToString ( ).Trim ( ) ; StringBuilder buf = new StringBuilder ( ) ; //buf.Append("GET ").Append(path).Append(" HTTP/1.0\r\n"); //buf.Append("Content-Type: application/x-www-form-urlencoded\r\n"); //buf.Append("\r\n"); byte [ ] ms = System. Text.UTF8Encoding.UTF8.GetBytes (buf.ToString ( ) ) ; //提交请求的信息 socket.Send (ms ) ; //接收返回 string strSms = "" ; int recv = 0 ; do { recv = socket.Receive (data ) ; stringData = Encoding.ASCII.GetString (data, 0, recv ) ; //如果请求的页面meta中指定了页面的encoding为gb2312则需要使用对应的Encoding来对字节进行转换() strSms = strSms + stringData ; //strSms += recv.ToString(); } while (recv != 0 ) ; socket.Shutdown (SocketShutdown.Both ) ; socket.Close ( ) ; return strSms ; }
这篇关于c# HttpWebRequest与HttpWebResponse的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!