远程主机关闭了连接错误,Flush()或End()方法导致的

在.NET里想要输出一段流一般需要用到以下方法:

Response.Buffer = false;
Response.AddHeader("Connection", "Keep-Alive");
Response.ContentType = "application/octet-stream";
Response.HeaderEncoding = System.Text.Encoding.UTF8;
Response.AddHeader("Content-Disposition", "attachment;filename=" + HttpUtility.UrlEncode(FileName,System.Text.Encoding.UTF8).Replace("+", "%20"));
Response.AddHeader("Content-Length", r.Length.ToString());
byte[] buffer;
buffer = r.ToArray();
Response.BinaryWrite(buffer);
r.Close(); //关闭下载文件 
Response.Flush();
Response.End(); //结束文件下载

但是程序在运行时会报出“正在中止线程”的错误,可以确定的是引起错误的代码就是Response.End();

所以需要删除此方法。

在IIS 5.5下这样做问题就解决了,不过在IIS 6.0和7.0 下又会报出另外一个错误“与远程主机通信时发生错误。错误代码是 0x8007xxxx。”

在网上翻查了一翻资料后,终于找到了解决方法:就是将Response.End();换成Response.Close();

 

总结一下,帮助对Response.Close()的解释是关闭到客户端的连接。对Response.End()的解释是停止该页的执行,并引发Application_EndRequest。

也就说用Response.End()程序就直接停止后面的工作了转而触发Application_EndRequest,那么当程序在后面还有代码需要运行时,程序就会抛出ThreadAbortException的异常。还有需要了解的就是end方法在不抛出异常时还会调用flush的方法。

接着来,Response.Close()方法是关闭了连接,也就说程序顺利执行完了所有代码后关闭了连接。

对于只运行Flush()后报出的“与远程主机通信时发生错误”,我的理解是当执行了Flush()后在底层马上开始向客户端发送数据,但是Flush()貌似只能指示程序开始连接发送,却没有停止关闭的标识,导致程序报出异常。当然这是在IIS 7.0下会出错(可能在IIS 7.0 开始在调用Flush()后需要显示关闭socket连接)。

所以以后再在用完Flush()后最好加上Close()。就像数据库连接一样,用完就关闭连接。

 

P.s 如果您用了以上方法仍旧未能解决问题,我的建议是使用终极方案了(本方案仅供参考,没有办法的办法):

try{ Response.Flush();  }
catch{}
finally
{
Response.Close();
}

 

使用以上方法,可能会导致又一个新的错误出现:
“异常类型: ThreadAbortException。 异常消息: 正在中止线程”
感谢您的耐心,看了这么久,我这里给出最佳解决方案代码:

try{ Response.Flush();  }
catch{}
finally
{
System.Web.HttpContext.Current.ApplicationInstance.CompleteRequest();



以下是相关原理解析,感兴趣的可以看看:

PRB:在使用 Response.End、Response.Redirect 或 server.Transfer 时出现 ThreadAbortException

症状 
如果使用 Response.End、Response.Redirect 或 Server.Transfer 方法,将出现 ThreadAbortException 异常。您可以使用 try-catch 语句捕获此异常。

原因 
Response.End 方法终止页的执行,并将此执行切换到应用程序的事件管线中的 Application_EndRequest 事件。不执行 Response.End 后面的代码行。
此问题出现在 Response.Redirect 和 Server.Transfer 方法中,因为这两种方法均在内部调用 Response.End。

解决方案 
要解决此问题,请使用下列方法之一: 
• 对于 Response.End,调用 HttpContext.Current.ApplicationInstance.completeRequest 方法而不是 Response.End 以跳过 Application_EndRequest 事件的代码执行。 
• 对于 Response.Redirect,请使用重载 Response.Redirect(String url, bool endResponse),该重载对 endResponse 参数传递 false 以取消对 Response.End 的内部调用。例如: Response.Redirect ("nextpage.aspx", false); 如果使用此替代方法,将执行 Response.Redirect 后面的代码。 
• 对于 Server.Transfer,请改用 Server.Execute 方法。

 经测试, Response.End 才能正常的终止服务器继续输出内容到浏览器。 
======================================= 
测试代码:   
 protected void Page_Load(object sender, EventArgs e)
    {
        try
        {
            Response.Write("11");
            Context.ApplicationInstance.CompleteRequest();
            Response.Write("22");
            Response.End();
        }
        catch
        {
            Response.Write("33");
        }
    }
=======================================

来看看Response.End()方法的实现:

public void End()
{
    if (this._context.IsInCancellablePeriod)
    {
        InternalSecurityPermissions.ControlThread.Assert();
        Thread.CurrentThread.Abort(new HttpApplication.CancelModuleException(false));
    }
    else if (!this._flushing)
    {
        this.Flush();
        this._ended = true;
        if (this._context.ApplicationInstance != null)
        {
            this._context.ApplicationInstance.CompleteRequest();
        }
    }
}

在来看看CompleteRequest()方法的实现:

internal void CompleteRequest()
{
    this._requestCompleted = true;
    if (HttpRuntime.UseIntegratedPipeline)
    {
        HttpContext context = this._application.Context;
        if ((context != null) && (context.NotificationContext != null))
        {
            context.NotificationContext.RequestCompleted = true;
        }
    }
}

通过以上代码,可见Response.End()是终止服务器输出,但有可能会throw出线程异常,不处理即可,ms官方对该问题进行了解释,如果你不希望throw出线程错误,则可以采用相关解决方法。

CompleteRequest()是通知结束请求。



  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值