http://www.sufeinet.com/plugin.php?id=keke_group

苏飞论坛

 找回密码
 马上注册

QQ登录

只需一步,快速开始

分布式系统框架(V2.0) 轻松承载百亿数据,千万流量!讨论专区 - 源码下载 - 官方教程

HttpHelper爬虫框架(V2.7-含.netcore) HttpHelper官方出品,爬虫框架讨论区 - 源码下载 - 在线测试和代码生成

HttpHelper爬虫类(V2.0) 开源的爬虫类,支持多种模式和属性 源码 - 代码生成器 - 讨论区 - 教程- 例子

查看: 35575|回复: 7

[教程] 关于”远程服务器返回错误: (417) Expectation failed“的解决方法

[复制链接]
发表于 2013-5-14 22:00:21 | 显示全部楼层 |阅读模式
这个问题还真有人问到,那我就写一下原因吧
我先来说一下这个问题可能出现在什么情况下,
如果你在使用的Httphelper类时,使用了代理 就有可能会报这个错误,这原因其实大家如果了解Http协议的话应该很容易就明白了。
我们来看看Http1.1协议中关于这个的说明吧。
  1. (100状态码(见10.1.1节)的目的在于允许客户端判定服务器是否愿意接受客户端发来的消息主体(基于请求头域)在客户端发送此请求消息主体前。 在有些情况下,如果服务器拒绝查看消息主体,这时客户端发送消息主体是不合适的或会降低效率。)
复制代码
就是说 ‍Expect:100-Continue的作用是,设定Client 和 Server在Post数据前需要进行 ‍“请求头域” 的数据匹配,相当于是握手。如果匹配则开始进行body 的内容,Post数据。否则,报错(417) Unkown。
也可以这样理解
允许客户端发request消息body之前先用request header试探一下server,看server要不要接收//request body,再决定要不要发request body。

不要急我们再来看看下面的一段
  1. (‍对HTTP/1.1代理服务器的要求:
  2. --- 若代理服务器接到一个请求,此请求包含值为"100-continue"的Expect请求头域,并且代理服务器可能知道下一站点的服务器遵循HTTP/1.1或更高版协议,或者不知道下一站点服务器的HTTP版本,那么它必须包含此Expect头域来转发此请求。
  3. --- 若代理服务器知道下一站点服务器版本是HTTP/1.0或更低,则它不能转发此请求,并且它必须以417(期望失败)状态响应。
  4. --- 代理服务器应当维护一个缓存,以记录最近访问下一站点服务器的HTTP版本号。
  5. --- 若接收到的请求来自于版本是HTTP/1.0(或更低)的客户端,并且此请求不含值为"100-continue"的Expect请求头域,那么代理服务器不能转发100(继续)响应。)
  6. 由于我们使用的是代理服务器,那个还有一种原因不能忽略,就是如果目标网页的HTTP的版本号为1.0或之前的版本,而代理服务器的本版为1.1或以上。这么这是,代理服务器将不会转发我们的Post请求,并报错‍(417) Unkown。
  7. 再看wireshark的包信息,其中明确可以看出,协议的版本号为HTTP1.1。这样,我们基本上可以确定‍(417) Unkown的原因:

  8. 握手失败,请求头域类型不匹配。
复制代码
其实关键就在的Http1.1限制了要先握手,我们这样想一下,如果我们选择不握手是不是就行了,
当然是可以的。
处理方法如下
可以直接在Http请求之前加上这么一句
[code=csharp] System.Net.ServicePointManager.Expect100Continue = false;[/code]
这句的意思就是不握手直接请求
其实我们还可以直接设置request的属性也是一样的,上面的方法太绝对了,一下就限制死了
[code=csharp]request.ServicePoint.Expect100Continue = false;[/code]
还有一个咱是在WebConfig里进行配置
[code=html]<configuration>
<system.net>
<settings>
           <servicePointManager expect100Continue="false" />
</settings>
</system.net>
</configuration>[/code]
其实 这个还有一个更明示的解释
就是当发送的数据超过1024时,是否寻询问接收多出的数据
当然我们直接选择False就是不询问了。
这个在我的类里面还暂时没有这样的方法
request.ServicePoint.Expect100Continue
去修改上面的属性,这个在下一个版本中我一定更新上,如果大家需要的话可以直接自己先添加一个属性。
先应下急
最后感谢用户cyberarmy提出的这个问题http://www.sufeinet.com/thread-3133-1-1.html

本帖被以下淘专辑推荐:



1. 开通SVIP会员,免费下载本站所有源码,不限次数据,不限时间
2. 加官方QQ群,加官方微信群获取更多资源和帮助
3. 找站长苏飞做网站、商城、CRM、小程序、App、爬虫相关、项目外包等点这里
发表于 2013-10-18 23:11:03 | 显示全部楼层
感谢分享!~
回复

使用道具 举报

发表于 2014-9-10 20:02:58 | 显示全部楼层
本帖最后由 Have_A_Dream 于 2014-9-10 20:06 编辑

@站长苏飞 我现在做的这个网站post,也是报这个错,根据教程设置后依然如此

抓包工具抓包信息

抓包工具抓包信息




[C#] 纯文本查看 复制代码
HttpHelper http = new HttpHelper();
            HttpItem item = new HttpItem()
            {
                URL = url4,//URL     必需项    
                Method = "POST",//URL     可选项 默认为Get   
                IsToLower = false,//得到的HTML代码是否转成小写     可选项默认转小写   
                Cookie = setcookie,//字符串Cookie     可选项    
                Postdata = buf,//Post数据     可选项GET时不需要写   
                UserAgent = "Mozilla/5.0 (compatible; MSIE 9.0; Windows NT 6.1; Trident/5.0)",//用户的浏览器类型,版本,操作系统     可选项有默认值   
                ContentType = "application/x-www-form-urlencoded",//返回类型    可选项有默认值   
                Allowautoredirect = true,//是否根据301跳转     可选项     
                ProxyIp = IP,//代理服务器ID     可选项 不需要代理 时可以不设置这三个参数     
                ResultType = ResultType.String
            };
            //item.Header.Add("x-requested-with", "XMLHttpRequest");
            //System.Net.ServicePointManager.Expect100Continue = false;
            HttpResult result = http.GetHtml(item);
            string html = result.Html;
            string cookie = result.Cookie;

 楼主| 发表于 2014-9-10 20:11:29 | 显示全部楼层
  //System.Net.ServicePointManager.Expect100Continue = false;这个设置了吗,两个同时设置
发表于 2014-9-10 20:12:31 | 显示全部楼层
站长苏飞 发表于 2014-9-10 20:11
//System.Net.ServicePointManager.Expect100Continue = false;这个设置了吗,两个同时设置

设置了,两个开设置了运行后还是一样的
发表于 2014-9-13 00:05:53 | 显示全部楼层




@站长苏飞

快来救救我吧,自己实在没办法解决了,困惑好几天了
发表于 2014-9-13 00:29:47 | 显示全部楼层
本帖最后由 Have_A_Dream 于 2014-9-13 14:19 编辑

发表于 2016-12-3 16:25:40 | 显示全部楼层
1. 发送一个请求, 包含一个Expect:100-continue, 询问Server使用愿意接受数据
2. 接收到Server返回的100-continue应答以后, 才把数据POST给Server
于是,这样就有了一个问题, 并不是所有的Server都会正确应答100-continue, 比如lighttpd, 就会返回417 “Expectation Failed”, 则会造成逻辑出错。
您需要登录后才可以回帖 登录 | 马上注册

本版积分规则

QQ|手机版|小黑屋|手机版|联系我们|关于我们|广告合作|苏飞论坛 ( 豫ICP备18043678号-2)

GMT+8, 2024-11-24 12:19

© 2014-2021

快速回复 返回顶部 返回列表