ASP.NET 动态转静态页面的两种方法总结:(欢迎大家一起讨论)
由于搜索引擎对aspx页面收录和html页面收录率的差别以及页面资源占用问题,我们很多时候需要实现ASPX页面动态转静态。网上也有很多人
讨论其实现方法,本人实践后总结两种主流方法如下:
第一种方法:使用模板转换,步骤如下:1、建立MyConvert.cs类文件 using System; //记得添加以下三引用 using System.Text; using System.Web; using System.IO; namespace TesConvert { /// /// MyConvert 的摘要说明。 /// public class MyConvert { public MyConvert() { // // TODO: 在此处添加构造函数逻辑 // } public bool WriteFile(string strText,string strContent,string strAuthor) { string path = HttpContext.Current.Server.MapPath("/TesConvert/news/");//定义html文件存放路径 Encoding code = Encoding.GetEncoding("gb2312");//定义文字编码 // 读取模板文件 string temp = HttpContext.Current.Server.MapPath("/TesConvert/text.html"); StreamReader sr=null; StreamWriter sw=null; string str=""; try { sr = new StreamReader(temp, code); str = sr.ReadToEnd(); // 读取文件 } catch(Exception exp) { HttpContext.Current.Response.Write(exp.Message); HttpContext.Current.Response.End(); sr.Close(); } string htmlfilename=path + DateTime.Now.ToString("yyyyMMddHHmmss")+".html"; // 替换内容 // 这时,模板文件已经读入到名称为str的变量中了 str = str.Replace("ShowArticle",strText); //模板页中的ShowArticle str = str.Replace("title",strText); str = str.Replace("content",strContent); str = str.Replace("author",strAuthor); // 写文件 try { sw = new StreamWriter(htmlfilename,false,code); sw.Write(str); sw.Flush(); } catch(Exception ex) { HttpContext.Current.Response.Write(ex.Message); HttpContext.Current.Response.End(); } finally { sw.Close(); } return true; } } } 2、TestNews.aspx文件:添加三和TextBox分别为:tbx_Title、tbx_Content、tbx_Author和一个Button:btn_AddNews。 TestNews.aspx.cs文件 private void btn_AddNews_Click(object sender, System.EventArgs e) { MyConvert Hover = new MyConvert();
if(Hover.WriteFile(this.txb_Title.Text.ToString(),Server.HtmlDecode(this.txb_Content.Value),this.txb_Author.Text.ToString())) { Response.Write("添加成功"); } else { Response.Write("生成HTML出错!"); } } 3、添加模板text.html文件 ShowArticletitlecontentauthor说明:一.news文件夹必须赋予asp.net用户写入的权限。这是一个简单的实现例子,实际项目必须先将数据保存到数据库下面,在datagird中
调用数据库下面html文件的URL地址。二.默认情况下,我们是不能向TextBox、TextArea中添加html语法的,必须修改config文件,在
下面添加,但是这样做的话,整个项目中都允许键入html标签了,暂时还不知道其他的方。 缺点:这种方法是在ASP.net在页面所有内容生成后、输出内容前对页面内容进行操作以前曾说过用HttpModule来在Response前更改,不够灵活
,每行修改response,比较费力。
第二种方法:重写AttributeCollection.Render,比较灵活(msdn如是说:在呈现阶段,所有 ASP.NET 移动设备适配器都通过一个称为文本编写器的对象
来编写它们的输出。文本编写器对象是从 TextWriter 基类创建的。)可以写个基类,如:public class BasePage: System.Web.UI.Page{ public BasePage() { } protected override void Render(System.Web.UI.HtmlTextWriter writer) { string name=Request.Url.AbsolutePath.Substring(1,Request.Url.AbsolutePath.Length-1).Replace("aspx","htm"); string newurl=""; if(name.IndexOf("/")0) { newurl=Server.MapPath("../") + name; } else { newurl=Server.MapPath("./") + name; } MemoryStream ms = new MemoryStream(); StreamWriter sww = new StreamWriter(ms); StreamWriter swr = new StreamWriter(newurl); System.Web.UI.HtmlTextWriter htmlw = new HtmlTextWriter(swr); base.Render(htmlw); htmlw.Flush(); htmlw.Close(); string strLL = System.Text.Encoding.UTF8.GetString(ms.ToArray()); Response.Write(strLL); Response.Redirect(Request.Url.AbsoluteUri.Replace("aspx","htm"), true); }}然后在需要生成静态页面的页面中继承就可以了。
说明:这种办法是在Asp.net的生成动作完成之后,再进行一次转换。缺点:觉得本质上应该还是属于频繁post的aspx页面。
小弟目前还是菜鸟一只,请各路兄弟点评。

对于你说的第二种方法,需要配合自动查找静态文件方法。在页面的page_init事件中,首先查找本地是否已经有静态文件了,如果有,就直接重定向(例如 Server.Transfer(...))到那个页面。当需要重建哪一个页面的时候,删除其静态文件的就可以了,下次访问的时候会自动重建。
不过不论第一种还是第二种,我认为意义都不大,特别是与页面缓冲或者Ajax相比之下。
如果你转换的时候将所有url都自动转换为指向静态文件的,并且静态文件是真的静态也就是当asp.net被从服务器上删除的时候仍然可以通过web服务器浏览整个网站,似乎还有一点意义。
我是这样做的
新闻正文内容用你说的那个替换方法
我还采用了下面的生成方法,可是有个问题
如果你是用datagrid或datalist等东西绑顶的数据库的话,生成的静态页样式表有问题,很严重的说
public class GetPageHtml : System.Web.UI.Page{protected System.Web.UI.WebControls.Button WebClientButton;protected System.Web.UI.WebControls.Button WebRequestButton;protected System.Web.UI.WebControls.TextBox ContentHtml;protected System.Web.UI.WebControls.TextBox UrlText;private string PageUrl = "";
private void Page_Load(object sender, System.EventArgs e){}
#region Web Form Designer generated codeoverride protected void OnInit(EventArgs e){InitializeComponent();base.OnInit(e);}
/// /// 设计器支持所需的方法 - 不要使用代码编辑器修改/// 此方法的内容。/// private void InitializeComponent(){ this.WebClientButton.Click += new System.EventHandler(this.WebClientButton_Click);this.WebRequestButton.Click += new System.EventHandler(this.WebRequestButton_Click);this.Load += new System.EventHandler(this.Page_Load);}#endregion
private void WebClientButton_Click(object sender, System.EventArgs e){PageUrl = UrlText.Text;WebClient wc = new WebClient();wc.Credentials = CredentialCache.DefaultCredentials;
///方法一:Byte[] pageData = wc.DownloadData(PageUrl);ContentHtml.Text = Encoding.Default.GetString(pageData);
/// 方法二:/// ***************代码开始**********/// Stream resStream = wc.OpenRead(PageUrl);/// StreamReader sr = new StreamReader(resStream,System.Text.Encoding.Default);/// ContentHtml.Text = sr.ReadToEnd();/// resStream.Close();/// **************代码结束********/// wc.Dispose(); }
private void WebRequestButton_Click(object sender, System.EventArgs e){PageUrl = UrlText.Text;WebRequest request = WebRequest.Create(PageUrl);WebResponse response = request.GetResponse();Stream resStream = response.GetResponseStream(); StreamReader sr = new StreamReader(resStream, System.Text.Encoding.Default);ContentHtml.Text = sr.ReadToEnd();resStream.Close(); sr.Close();}}
我是用xml + xsl来实现的
在xml文档里定义需要的取出的数据
程序遍历xml文档,取出数据加入当前xml文档中
再使用xsl转换为html并输出
xsl转html是比较耗CPU的,如果访问量比较大,可以考虑用.net remoting把xsl转html放到另外的服务器上处理
推荐使用IHttpHandleFactory,先捕获url地址,再生成页面,超级简单public class HttpHand:IHttpHandlerFactory{#region 创建Http工厂
public void ReleaseHandler(IHttpHandler handler){
}
public IHttpHandler GetHandler(HttpContext context, string requestType, string url, string pathTranslated){//获取随意定义的文件的名字,包含扩展string name = url.Substring(url.LastIndexOf('/')+1);string filename = name.Substring(0,name.LastIndexOf('.'));filename="CommonFactory."+filename;
object obj = null; //一个类型,用于转IhttpHandler为抽象类型
//创建Httptry{Type tp = Type.GetType(filename); obj = System.Activator.CreateInstance(tp);}catch{
}return (IHttpHandler)obj;}
#endregion}
-----------------------------------------------------------------------public class LoginOut:IHttpHandler{
SqlFactory.Operation opea = new SqlFactory.Operation();public void Proce***equest(HttpContext context){if(context.User.Identity.Name!=string.Empty){string path =context.Request.QueryString["P"];
//清除用户登录表信息opea.voidDelSinLogin(context.User.Identity.Name); opea.voidAddWork(context.User.Identity.Name+"注销用户","注销成功");Sec.FormsAuthentication.SignOut();AlertStr.AlertBackWindow(context.Response,"注销成功,欢迎再来!",path);
}else{opea.voidAddWork("匿名用户注销","注销失败,原因:没有登录");AlertStr.AlertBackWindow(context.Response,"没有登录,无法注销",context.Request);}}
public bool IsReusable{get{// TODO: 添加 LoginOut.IsReusable getter 实现return true;}}

}