C# 模拟多用户登录,出现超时诡异问题。求教。

MouseClick 2014-09-26 06:42:51
写了一个Winform模拟一个网站的登录,多用户登录后签到拿积分的。在实现中出现一个超时的问题,非常奇怪找不出原因,求高手赐教。
代码如下

private static List<User> userList;
private void btnSign_Click(object sender, EventArgs e)
{
loadUserInfo(); //加载用户数据,会自动new 一个List<User>
if (userList != null && userList.Count > 0)
{
for (int i = 0; i < userList.Count; i++)
{
Thread.Sleep(1000);
User user = userList[i];
string userId = Des.DecryptDES(user.Id, "yakexiek");
string passWord = user.PassWord;
string userName = "";
string signString = ""; //获取签到后的信息

CookieCollection curCookie = new CookieCollection();
// HttpWebResponse response = null;
//获取登陆页面的cookie
response = HttpHelper.CreateGetHttpResponse(indexUrl, null, null, null);//这里第二次运行出现问题。
curCookie = HttpHelper.GetAllCookiesFromHeader(response.Headers["Set-Cookie"], host);

//POST 登陆页面,发送账号密码
IDictionary<string, string> parameters = new Dictionary<string, string>();
parameters.Add("Username", userId);
parameters.Add("Password", passWord);
response = HttpHelper.CreatePostHttpResponse(loginUrl, parameters, null, null, curCookie);
curCookie = response.Cookies;


//模拟登陆showSystem.jsp,
response = HttpHelper.CreateGetHttpResponse(showSystemUrl, null, null, curCookie);

//模拟登陆openForm页面,进行用户别名解析
response = HttpHelper.CreateGetHttpResponse(openFormUrl, null, null, curCookie);
userName = getUserName(HttpHelper.GetResponseString(response, Encoding.UTF8));

//签到
response = HttpHelper.CreateGetHttpResponse(signUrl + userName, null, null, curCookie);
signString = HttpHelper.GetResponseString(response);
userList.RemoveAt(i);

}

}

}
///
HttpHelper.CreateGetHttpResponse

private static readonly string DefaultUserAgent = "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.2; SV1; .NET CLR 1.1.4322; .NET CLR 2.0.50727)";
/// <summary>
/// 创建GET方式的HTTP请求
/// </summary>
public static HttpWebResponse CreateGetHttpResponse(string url, int? timeout, string userAgent, CookieCollection cookies)
{
HttpWebRequest request = null;
if (url.StartsWith("https", StringComparison.OrdinalIgnoreCase))
{
//对服务端证书进行有效性校验(非第三方权威机构颁发的证书,如自己生成的,不进行验证,这里返回true)
ServicePointManager.ServerCertificateValidationCallback = new RemoteCertificateValidationCallback(CheckValidationResult);
request = WebRequest.Create(url) as HttpWebRequest;
request.ProtocolVersion = HttpVersion.Version10; //http版本,默认是1.1,这里设置为1.0
}
else
{
request = WebRequest.Create(url) as HttpWebRequest;
}
request.Method = "GET";
request.UserAgent = DefaultUserAgent;
if (!string.IsNullOrEmpty(userAgent))
{
request.UserAgent = userAgent;
}
if (timeout.HasValue)
{
request.Timeout = timeout.Value;
}
//设置代理UserAgent和超时
//request.UserAgent = userAgent;
//request.Timeout = timeout;
if (cookies != null)
{
request.CookieContainer = new CookieContainer();
request.CookieContainer.Add(cookies);
}
return request.GetResponse() as HttpWebResponse;
}
/// <summary>
/// Post请求
/// </summary>
/// <param name="url"></param>
/// <param name="parameters"></param>
/// <param name="timeout"></param>
/// <param name="userAgent"></param>
/// <param name="cookies"></param>
/// <param name="strRefer"></param>
/// <returns></returns>
public static HttpWebResponse CreatePostHttpResponse(string url, IDictionary<string, string> parameters, int? timeout, string userAgent, CookieCollection cookies,string strRefer)
{
HttpWebRequest request = null;
//如果是发送HTTPS请求
if (url.StartsWith("https", StringComparison.OrdinalIgnoreCase))
{
ServicePointManager.ServerCertificateValidationCallback = new RemoteCertificateValidationCallback(CheckValidationResult);
request = WebRequest.Create(url) as HttpWebRequest;
request.ProtocolVersion = HttpVersion.Version10;
}
else
{
request = WebRequest.Create(url) as HttpWebRequest;
}
request.Method = "POST";
request.ContentType = "application/x-www-form-urlencoded";

//设置代理UserAgent和超时
//request.UserAgent = userAgent;
//request.Timeout = timeout;
if (!string.IsNullOrEmpty(userAgent))
{
request.UserAgent = userAgent;
}
else
{
request.UserAgent = DefaultUserAgent;
}

if (timeout.HasValue)
{
request.Timeout = timeout.Value;
}
if (cookies != null)
{
request.CookieContainer = new CookieContainer();
request.CookieContainer.Add(cookies);
}
if (!string.IsNullOrEmpty(strRefer))
{
request.Referer = strRefer;
}
//发送POST数据
if (!(parameters == null || parameters.Count == 0))
{
StringBuilder buffer = new StringBuilder();
int i = 0;
foreach (string key in parameters.Keys)
{
if (i > 0)
{
buffer.AppendFormat("&{0}={1}", key, parameters[key]);
}
else
{
buffer.AppendFormat("{0}={1}", key, parameters[key]);
i++;
}
}
byte[] data = Encoding.ASCII.GetBytes(buffer.ToString());
using (Stream stream = request.GetRequestStream())
{
stream.Write(data, 0, data.Length);
}
}
// string[] values = request.Headers.GetValues("Content-Type");
HttpWebResponse resposne = (HttpWebResponse)request.GetResponse();
resposne.Cookies = request.CookieContainer.GetCookies(request.RequestUri);
return resposne;

}

/// <summary>
/// 创建POST方式的HTTP请求
/// </summary>
public static HttpWebResponse CreatePostHttpResponse(string url, IDictionary<string, string> parameters, int? timeout, string userAgent, CookieCollection cookies)
{

return CreatePostHttpResponse(url,parameters,timeout,userAgent,cookies,null);
}


第一次模拟登陆成功,并签到了。
第二次在response = HttpHelper.CreateGetHttpResponse(indexUrl, null, null, null);这个地方,卡住了,过了一会出现超时。
不知道什么问题。
...全文
266 2 打赏 收藏 转发到动态 举报
写回复
用AI写文章
2 条回复
切换为时间正序
请发表友善的回复…
发表回复
MouseClick 2014-09-27
  • 打赏
  • 举报
回复
问题已经解决,每次结束都response.Close();,没有进行request.Close()。还是非常感谢。
__lhy 2014-09-27
  • 打赏
  • 举报
回复
对于GetResponse或GetRequestStream超时死掉的原因,可能是: 1.DefaultConnectionLimit是默认的2,而当前的Http的connection用完了,导致后续的GetResponse或GetRequestStream超时死掉 ==>> 默认系统只支持同时存在2个http的connection ==>> 使用HttpWebRequest之后如果没有close,则会占用1个http的connection,所以如果超过2次使用HttpWebRequest而没有close,那么就用完系统的http的connection,之后再去使用HttpWebRequest,GetResponse就会死掉 解决办法: 办法1: 每次使用完HttpWebRequest,使用 1 2 req.Close(); req=null; 去关闭对应的http connection 最好对应的HttpWebResponse也要close: 1 2 resp.Close(); resp = null; 方法2: 修改DefaultConnectionLimit的值,改为足够大,比如: 1 System.Net.ServicePointManager.DefaultConnectionLimit = 200; 2.系统中Http相关的资源没有正确释放,导致后续GetResponse或GetRequestStream超时死掉 就像我此处遇到的,可能是之前调用http相关函数,没有正确完全释放资源,导致虽然DefaultConnectionLimit给了足够大,但是还是会死掉,此时在http请求代码之前去做一次垃圾回收,则后续http的GetResponse或GetRequestStream就正常了,就不会超时死掉了。 参考代码如下: 1 2 3 4 5 System.GC.Collect(); req = (HttpWebRequest)WebRequest.Create(constSkydriveUrl); setCommonHttpReqPara(ref req); resp = (HttpWebResponse)req.GetResponse(); 3.Http的GET请求时,不要手动设置ContentLength的值 这个是参考这里:HttpWebRequest.GetResponse() hangs the second time it is called而记录于此的,也许有人是此原因,所以可供参考一下。 即Http的GET请求,不要添加类似如下的代码: 1 2 if (m_contentLength > 0) httpWebRequest.ContentLength = m_contentLength; 不要去手动修改对应的ContentLength的值,C#的http相关库函数,会自动帮你计算的。 注:POST方法中,的确是要手动填充数据和算出数据大小,然后手动给ContentLength赋值的。 4.其他可能的一些原因 (1)关于KeepAlive的问题 如果Http的请求,是设置了KeepAlive=true的话,那么对应的http的connection会和服务器保持连接的。 所以如果上述办法都不能解决超时的问题,可以尝试将keepAlive设置为false试试,看看能否解决。 (2)关于Sleep 有些人好像是通过在http请求前,加了对应的Sleep,结果解决了此问题。需要的人,也可以试试。 (3)HttpWebRequest的Timeout 一般来说,既然超时了,往往是由于错误使用函数或者网络有问题导致的,所以实际上此处对于有些人去把HttpWebRequest的Timeout的值改的更大,往往都是没用的。 只不过,万一是由于网络响应慢而导致超时,那么倒是可以尝试,将HttpWebRequest的Timeout的值改为更大。 (其中HttpWebRequest的Timeout默认的值是100,000 milliseconds ==100 seconds) 参考代码: 1 req.Timeout = 5 * 60 * 1000; // 5 minutes

110,534

社区成员

发帖
与我相关
我的任务
社区描述
.NET技术 C#
社区管理员
  • C#
  • Web++
  • by_封爱
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

让您成为最强悍的C#开发者

试试用AI创作助手写篇文章吧