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;
}
}
}