asp.net2.0 の urlMappings は非常に使いやすいですが、残念ながら正規表現はサポートされていません。ただし、幸いなことに、IHttpModule を使用すると、
どのような種類のリクエストであっても、最初に IHttpModule を通過するため、優れた機能が提供されます
。URL 書き換えの機会:
以下は私が作成した IHttpModule です
。
System.Web の
パブリック クラス ReWriteModule:IHttpModule
を使用します。
{
public ReWriteModule()
{
}
パブリックオーバーライド文字列 ToString()
{
this.GetType().ToString() を返します。
}
void IHttpModule.Dispose()
{
}
プライベート静的 System.Xml.XmlDocument ルールドキュメント = null;
プライベート静的 System.Xml.XmlDocument GetRuleConfig(System.Web.HttpContext アプリ)
{
if (ruleDoc == null)
{
rulesDoc = 新しい System.Xml.XmlDocument();
rulesDoc.Load(app.Server.MapPath("~/rule.xml"));
}
ルールドキュメントを返します。
}
public static string GetUrl(System.Web.HttpContext cxt,string path)
{
System.Xml.XmlDocument doc = GetRuleConfig(cxt);
System.Xml.XmlNodeList lst= doc.GetElementsByTagName("RewriterRule");
文字列パット="";
foreach (最初の System.Xml.XmlNode 番目)
{
System.Xml.XmlNodeList sub = nd.ChildNodes[0].ChildNodes;
foreach(サブの System.Xml.XmlNode chk)
{
pat = "^" + chk.InnerText+"$";
System.Text. RegularExpressions.Regex reg = new System.Text. RegularExpressions.Regex(pat, System.Text. RegularExpressions.RegexOptions.Compiled | System.Text. RegularExpressions.RegexOptions.IgnoreCase);
if(reg.IsMatch(パス))
{
return reg.Replace(path, nd.ChildNodes[1].InnerText);
}
}
}
null を返します
。
void IHttpModule.Init(HttpApplication コンテキスト)
{
context.BeginRequest += デリゲート(オブジェクト送信者、EventArgs e)
{
System.Web.HttpContext cxt = context.Context;
if (cxt.Request.ContentType != "画像/pjpeg")
{
文字列タイプ = cxt.Request.ContentType.ToLower();
文字列パス = cxt.Request.Path;
文字列 apppath = cxt.Request.ApplicationPath;
パス = path.Remove(0, apppath.Length);
パス = "~" + パス;
string newUrl = GetUrl(cxt, path.TrimEnd().TrimStart());
if (newUrl != null)
{
cxt.Response.Filter = 新しいResponseFilter(cxt.Response.Filter,cxt.Request.Path);
cxt.Response.Write("要求されたパス:" + パス);
cxt.Response.Write("<BR>");
cxt.Response.Write("リダイレクト先 URL: " + newUrl);
cxt.Response.Write("<BR>");
cxt.RewritePath(newUrl);
} // すべてのリクエストを処理するために必要な場合に使用されます
//それ以外
//{
// cxt.Response.Write(cxt.Request.Path + "<BR>");
// cxt.Response.Write("要求したリソースが存在しないか、アクセスする権限がありません!");
// cxt.Response.Flush();
// cxt.Response.End();
//}
}
};
}
URL が書き換えられると、元の WEBFORM のアクションが変更されるため、要求されたリソースが存在しないということが簡単に発生する可能性があり
ます
。 DXしてみれば一目瞭然! ! !
私たちが持っているのはこの ResponseFilter だけです。これは次のように実装されています。
public class ResponseFilter:System.IO.Stream
{
public ResponseFilter(System.IO.Stream シンク,string _str)
{
_sink = シンク;
//
// TODO: ここにコンストラクター ロジックを追加します
//
this.str = _str;
}
プライベート文字列 str = "";
プライベート System.IO.Stream _sink;
プライベートロング_位置;
private System.Text.Encoding end=System.Text.Encoding.GetEncoding("GB18030");
private System.Text.StringBuilder oOutput = new System.Text.StringBuilder();
// Stream の次のメンバーはオーバーライドする必要があります。
パブリックオーバーライドブール CanRead
{
取得 { true を返します }
public
オーバーライド bool CanSeek
{
取得 { true を返します }
パブリック
オーバーライドブール CanWrite
{
取得 { true を返します }
オーバーライド
の長い長さ
{
取得 { 0 を返す }
パブリック
オーバーライドロングポジション
{
取得 { 位置を返す }
セット { _位置 = 値 }
public
オーバーライド long Seek(ロング オフセット、System.IO.SeekOrigin 方向)
{
return _sink.Seek(オフセット、方向);
パブリック
オーバーライド void SetLength(長い長さ)
{
_sink.SetLength(長さ);
オーバーライド
void Close()
{
_sink.Close();
パブリック
オーバーライド void Flush()
{
_sink.Flush();
public override int Read(byte[] バッファ, int オフセット, int カウント
)
{
return _sink.Read(バッファ, オフセット, カウント);
}
// Write メソッドが実際にフィルタリングを実行します。
public override void Write(byte[] バッファ、int オフセット、int カウント)
{
string szBuffer = System.Text.UTF8Encoding.UTF8.GetString(バッファ、オフセット、カウント);
文字列 ap="アクション="";
int pos=-1;
if ((pos=szBuffer.IndexOf(ap) )!= -1)
{
int epos = szBuffer.IndexOf(""", pos + ap.Length+1);
if (エポス != -1)
{
szBuffer= szBuffer.Remove(pos + ap.Length, epos - pos - ap.Length);
szBuffer
= szBuffer.Insert(pos + ap.Length, this.str);
byte[] data = System.Text.UTF8Encoding.UTF8.GetBytes(szBuffer);
_sink.Write(データ, 0, データ.長さ);
}
それ以外
{
oOutput.Append(szBuffer);
}
//次の段落を使用して、<Head></head> の間のコンテンツを変更できます。
//Regex oEndFile = new Regex("</head>", RegexOptions.IgnoreCase|RegexOptions.Compiled);
//if (oEndFile.IsMatch(szBuffer))
//{
// //データの最後のバッファを追加します
// //バッファ内のデータの最後の部分を追加します
// oOutput.Append(szBuffer);
// //クライアントへの完全な応答を取得します
// //完全なクライアント戻りデータを送信する
// 文字列 szCompleteBuffer = oOutput.ToString().ToLower();
// int ipos = szCompleteBuffer.IndexOf("<タイトル>");
// int epos = szCompleteBuffer.IndexOf("</title>",ipos+7);
// 文字列 sp = szCompleteBuffer.Substring(ipos+7, epos - ipos );
// szCompleteBuffer = szCompleteBuffer.Remove(ipos+7,sp.Length-7);
// szCompleteBuffer = szCompleteBuffer.Insert(ipos + 7, "dhz");
// // szCompleteBuffer = szCompleteBuffer.Replace(sp, "dhz");
// //一致しないため、元のデータを書き出す
// //一致しないため、ソースコードが書き込まれます
// byte[] データ = System.Text.UTF8Encoding.UTF8.GetBytes(szCompleteBuffer);
// _sink.Write(data, 0, data.Length);
//}
//それ以外
//{
// oOutput.Append(szBuffer);
//}
}
}
//////待機ルールは、xml ファイルを使用して次のように構成されます。
もちろん、web.config
<?xml version="1.0"coding="utf-8"
のカスタム構成セクションを通じて実行することもできます。? >
<ルール>
<リライタールール>
<探す>
<LookFor>~/(d{4})/(d{2}).html</LookFor>
<LookFor>~/(d{4})/(d{2})/</LookFor>
<LookFor>~/(d{4})/(d{2})</LookFor>
<LookFor>~/(d{4})/(d{2})/index.html</LookFor>
</探す>
<SendTo>~/Pro.aspx?year=$1&month=$2</SendTo>
</リライタールール>
<リライタールール>
<探す>
<LookFor>~/pc</LookFor>
</探す>
<SendTo>~/Test2.aspx</SendTo>
</リライタールール>
</ルール>
//このルールは適切に記述されていません。たとえば、最初のルールは正規表現を使用して実行できます。でも、現時点では、アンチキャプチャグループの概念を使用する必要があるようです。 !