拦截asp.net输出流做处理

本文标题是指对已经生成了HTML的页面做一些输出到客户端之前的处理。

方法的原理是:把Response的输出重定向到自定义的容器内,也就是我们的StringBuilder对象里,在HTML所有的向页面输出都变成了向StringBuilder输出,然后我们对StringBuilder处理完成之后,再把Response的输出重定向到原来的页面上,然后再通过Response.Write方法把StringBuilder的内容输出到页面上

这里之所以用反射,是因为Response对象的OutPut属性是只读的,通过反编译该类的程序集发现,OutPut实际上是内部私有成员 _writer来实现输出的。因此通过反射来改写该成员的值以实现输出流的重定向。

[c-sharp] view plaincopy

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Web;
  5. using System.Web.UI;
  6. using System.Web.UI.WebControls;
  7. using System.Text;
  8. using System.IO;
  9. using System.Reflection;
  10. public partial class _Default : System.Web.UI.Page
  11. {
  12. StringBuilder content = new StringBuilder();
  13. TextWriter tw_old, tw_new;
  14. FieldInfo tw_field;
  15. protected void Page_Load(object sender, EventArgs e)
  16. {
  17. var context = HttpContext.Current;
  18. tw_old = context.Response.Output;//Response原来的OutPut
  19. tw_new = new StringWriter(content);//一个StringWriter,用来获取页面内容
  20. var type_rp = context.Response.GetType();
  21. //通过反射获取对象的私有字段
  22. tw_field = type_rp.GetField("_writer", System.Reflection.BindingFlags.Public | System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance);
  23. tw_field.SetValue(context.Response, tw_new);
  24. }
  25. protected override void Render(HtmlTextWriter writer)
  26. {
  27. base.Render(writer);
  28. //替换回Response的OutPut
  29. tw_field.SetValue(HttpContext.Current.Response, tw_old);
  30. //做自己的处理
  31. content.AppendLine("<!--江湖小子-->");
  32. HttpContext.Current.Response.Write(content.ToString());
  33. }
  34. }
  35. 方法二,用HttpModul来实现:
  36. using System;
  37. using System.Collections.Generic;
  38. using System.Linq;
  39. using System.Web;
  40. using System.Web.UI;
  41. using System.IO;
  42. using System.Text;
  43. using System.Reflection;
  44. /// <summary>
  45. ///HttpModule 的摘要说明
  46. /// </summary>
  47. public class HttpModule : IHttpModule
  48. {
  49. private HttpApplication _contextApplication;
  50. private TextWriter tw_new, tw_old;
  51. private StringBuilder _content;
  52. private FieldInfo tw_field;
  53. public void Init(HttpApplication context)
  54. {
  55. _contextApplication = context;
  56. _contextApplication.PreRequestHandlerExecute += new EventHandler(_contextApplication_PreRequestHandlerExecute);
  57. }
  58. public void Dispose()
  59. {
  60. _contextApplication = null;
  61. _contextApplication.Dispose();
  62. }
  63. public void _contextApplication_PreRequestHandlerExecute(object sender, EventArgs e)
  64. {
  65. HttpContext context = _contextApplication.Context;
  66. var _page = context.Handler as System.Web.UI.Page;
  67. _page.Unload += new EventHandler(_page_Unload);
  68. _content = new StringBuilder();
  69. tw_old = context.Response.Output;//Response原来的OutPut
  70. tw_new = new StringWriter(_content);//一个StringWriter,用来获取页面内容
  71. var type_rp = context.Response.GetType();
  72. tw_field = type_rp.GetField("_writer", System.Reflection.BindingFlags.Public | System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance);
  73. tw_field.SetValue(context.Response, tw_new);
  74. }
  75. void _page_Unload(object sender, EventArgs e)
  76. {
  77. //替换回Response的OutPut
  78. tw_field.SetValue(HttpContext.Current.Response, tw_old);
  79. //做自己的处理
  80. _content.AppendLine("<!--江湖小子-->");
  81. HttpContext.Current.Response.Write(_content.ToString());
  82. }
  83. }
  84. 方法三:
  85. public class HttpModule : IHttpModule
  86. {
  87. private HttpApplication _contextApplication;
  88. private TextWriter tw_new, tw_old;
  89. private StringBuilder _content;
  90. private FieldInfo tw_field;
  91. public void Init(HttpApplication application)
  92. {
  93. _contextApplication = application;
  94. _contextApplication.BeginRequest += new EventHandler(_contextApplication_BeginRequest);
  95. _contextApplication.EndRequest +=new EventHandler(_contextApplication_EndRequest);
  96. }
  97. void _contextApplication_BeginRequest(object sender, EventArgs e)
  98. {
  99. _content = new StringBuilder();
  100. tw_old = _contextApplication.Response.Output;
  101. tw_new = new StringWriter(_content);
  102. var type_rp = _contextApplication.Response.GetType();
  103. tw_field = type_rp.GetField("_writer", System.Reflection.BindingFlags.Public | System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance);
  104. tw_field.SetValue(_contextApplication.Response, tw_new);
  105. }
  106. void _contextApplication_EndRequest(object sender, EventArgs e)
  107. {
  108. tw_field.SetValue(_contextApplication.Response, tw_old);
  109. //做自己的处理
  110. _content.AppendLine("<!--jhxz-->");
  111. _contextApplication.Response.Write(_content.ToString());
  112. }
  113. public void Dispose()
  114. {
  115. _contextApplication = null;
  116. _contextApplication.Dispose();
  117. }
  118. }

最后还是推荐一篇好文:码农欧洲出差的一点小插曲

时间: 2024-11-05 20:39:42

拦截asp.net输出流做处理的相关文章

拦截asp.net输出流做处理, 拦截HTML文本

对已经生成了HTML的页面做一些输出到客户端之前的处理 方法的原理是:把Response的输出重定向到自定义的容器内,也就是我们的StringBuilder对象里,在HTML所有的向页面输出都变 成了向StringBuilder输出,然后我们对StringBuilder处理完成之后,再把Response的输出重定向到原来的页面上,然后再通 过Response.Write方法把StringBuilder的内容输出到页面上 这里之所以用反射,是因为Response对象的OutPut属性是只读的,通过

拦截asp.net输出流并进行处理的方法

本文实例主要实现对已经生成了HTML的页面做一些输出到客户端之前的处理. 方法的实现原理是:把Response的输出重定向到自定义的容器内,也就是我们的StringBuilder对象里,在HTML所有的向页面输出都变成了向StringBuilder输出,然后我们对StringBuilder处理完成之后,再把Response的输出重定向到原来的页面上,然后再通过Response.Write方法把StringBuilder的内容输出到页面上. 这里之所以用反射,是因为Response对象的OutPu

拦截asp.net mvc输出流做处理, 拦截HTML文本(asp.net MVC版)

以前的一个贴子写过一个webForm的拦截HTML输出流的版本,最近用到mvc时用同样的方式发生一些问题. 如下图 查了好久也不知道啥原因. 好吧, 我最后选择放弃. 想起以前自定义Response.Filter 时,里面Write方法可以获取页面流的信息. 这次我借用HttpModule实现拦截HTML内容输出流,下面看代码 一.HtmlHttpModule.cs    定义一个新类继承HttpModule using System; using System.Collections.Gene

ASP.NET MVC 做的网站项目

感谢博客园团队日夜为广大需要获取知识人们所做的奉献 博客园团队您们辛苦了 ASP.NET MVC 实现有论坛功能的网站(有iis发布网站 这是之前写的... www.lazyfitness.cn 经过一个月的修正 通过ASP.NET MVC 所做的网站www.lazyfitness.cn正式发布了!!! 这个网站不再像之前的哪样页面简陋功能单一了(毕竟之前的是一周做的...) 丰富了很多功能,还有后台管理员哦~ 值得一提的是,这个全部是基于APS.NET MVC框架实现的,我在博客园其实看到用A

ASP.NET CORE做的网站运行在docker上(不用dockerfile文件部署)

原文:ASP.NET CORE做的网站运行在docker上(不用dockerfile文件部署) 按网上的做法用dockerfile文件是可以弄得出来的,http://www.docker.org.cn/article/119.html, 不过我想把网站文件放在外面硬盘目录,再映射进去,这样只要在硬盘目录中修改CSHTML文件后重启一下容器就行了 步骤如下: 1. vs中建立ASP.NET CORE网站,类名为coreweb1 2. 发布到c:\temp\coreweb1目录 3. 先在本地CMD

ASP.NET输出流至少要有256个字节的数据后Response.Flush方法才会生效

很多时候我们写的asp.net程序会因为做很多操作,所以会花上一分钟甚至几分钟时间.为了使软件使用者能够耐心的等待程序的执行,我们经常会希望有一个进度条来表示程序执行的状态.或者最起码要显示一个类似: “数据载入中”,“正在保存数据” 等的说明性文字.此时我们就会用到Response.Flush().他会将缓冲区中编译完成的数据先发送到客户端. 但是有很多时候,我们发现即使我们使用了Response.Flush(),但是并没有将已经在输出流中的数据发到客户端.客户端呈献给我们的依然是白屏.经过反

使用asp.net dateaable 做统计表并且网页上显示的表和导出的Excel 一模一样 根据领导看报表需求两个报表和一起 (add algorithm)

1.首先使用    .net   控件 <tr> <td> <asp:Table ID="tabBill" runat="server" Style=" overflow: auto;border-collapse:collapse;width: 100%;table-layout: fixed;width: 1880px"> </asp:Table> </td> <td>

asp.net mvc,做 301 永久重定向

以下代码为 asp.net mvc 4.0 代码做的 301 永久重定向 string url = “http://www.csdn.net/test.html” Response.StatusCode = 301;      Response.Status = "301 Moved Permanently";      Response.AppendHeader("Location", url));      Response.AppendHeader("

20151229:AJax :用asp.net方式做一个查询

aspx代码: <%@ Page Language="C#" AutoEventWireup="true" CodeFile="Default.aspx.cs" Inherits="_Default" %> <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head runat="se