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文件
ShowArticle
title
content
author
說明:一.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 code
override 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爲抽象類型
//創建Http
try
{
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 ProcessRequest(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;
}
}
}