使用Microsoft Url Rewrite Module进行URL重写及Postback后保持URL的解决方案
Microsoft URL Rewrite Module 是微软推出的asp.net url重写模块,仅支持IIS7,要在IIS6上进行 url重写,可以采用一些第三方的模块,比如 urlrewriting.net。首先去下载Rewrite Module并安装。安 装完毕后,在IIS管理工具中就会出现Url Rewrite图标。
双击你需要进行URL重写的网站或者虚拟目录的URL Rewrite按钮,可以进行规则的编辑。微软的重写 模块非常强大,有很多功能,还可以自己写程序对其进行扩展。本文仅介绍最常用和实用的利用正则表达 式进行重写。重写的基本原理是当服务器接收到一个请求的时候,利用正则表达式匹配当前输入的url, 然后根据一个规则将这个url转换为实际的url。例如,我们要把 localhost/Game/game.aspx? ID=SomeName转换为 localhost/SomeName/,也就是每个游戏名都作为一个独立的目录。于是,我们要匹配 输入的url的格式是 ([^/?]*)/$,这个正则表达式不包括根目录,例如localhost/,它的含义是所有非/和 ?的字符,并且以/结尾,按照我们的要求,它的实际地址是~/Game/game.aspx,参数是正则表达式中第一 个圆括号匹配到的内容。实际的url中会用到这个正则表达式的反向引用。反向引用的格式是 {R:n},其 中n是一个整数。n等于0的时候是整个正则表达式,n大于0的时候是从左往右数第n个捕获组的值。在这里 个例子中,我们的实际url就是 Game/Game.aspx?ID={R:1}。对于正则表达式的反向引用不太熟悉的人, 微软也提供了一个很方便的功能,Test Pattern
在这里可以很方便的 测试自己的正则表达式是否正确。完成后,点击Apply,就完成了一条规则的设置。它会在web.config文 件中添加如下内容:
<system.webServer>
<rewrite>
<rules>
<rule name="Game">
<match url="([^/?]*)/$" />
<action type="Rewrite" url="Game/Game.aspx?ID={R:1}" appendQueryString="false" />
</rule>
</rules>
</rewrite>
</system.webServer>
熟悉的话可以直接在web.config中修改,比较方便。不过 ,VS2008并没有智能提示,反而提示rewrite节错误。
当有多条重写规则的时候,asp.net从上到 下依次匹配,因此,如果我们有如下规则
<rewrite>
<rules>
<rule name="News">
<match url="News/$" />
<action type="Rewrite" url="Game/News.aspx" appendQueryString="false" />
</rule>
<rule name="Game">
<match url="([^/?]*)/$" />
<action type="Rewrite" url="Game/Game.aspx?ID={R:1}" appendQueryString="false" />
</rule>
</rules>
</rewrite>
那么 localhost/News/将会被转换为 Game/News.aspx 而不是 Game/Game.aspx?ID=News。如果对正则 表达比较熟悉,重写url是没什么困难的。
最后讲一个很困扰的问题,也就是当实际的url中带有参数的时候,如果页面上某个控件执行了 postback的操作,那么浏览器上的url又会重新带上参数,变成类似http://localhost/goldshop/Aion/? ID=Aion 的样子。这是因为IIS实际处理的页面还是那个带有参数的真实url,因此postback的时候还是回 发到原始的url上。解决方案主要参考 Jeffery Zhao的博文重提URL Rewrite(3):在URL Rewrite后保 持PostBack地址。主要想法就是利用一个Control Adapter改写html中form控件的输出方式。将它的 action改成我们重写过后的url而不再是原始的url。这个Adapter的代码如下:
using System.IO;
using System.Web;
using System.Web.UI;
namespace Sample.Web.UI.Adapters
{
public class FormRewriterControlAdapter :
System.Web.UI.Adapters.ControlAdapter
{
protected override void Render(HtmlTextWriter writer)
{
base.Render(new RewriteFormHtmlTextWriter(writer));
}
}
public class RewriteFormHtmlTextWriter : HtmlTextWriter
{
public RewriteFormHtmlTextWriter(HtmlTextWriter writer)
: base(writer)
{
this.InnerWriter = writer.InnerWriter;
}
public RewriteFormHtmlTextWriter(TextWriter writer)
: base(writer)
{
this.InnerWriter = writer;
}
public override void WriteAttribute(string name, string value, bool fEncode)
{
if (name == "action")
{
HttpContext context = HttpContext.Current;
if (context.Items["ActionAlreadyWritten"] == null)
{
value = context.Request.RawUrl;
context.Items["ActionAlreadyWritten"] = true;
}
}
base.WriteAttribute(name, value, fEncode);
}
}
}
把这个代码放到App_Code文件夹中,另外在 App_Browser文件夹中新建一个browser文件,内容如下:
<browsers>
<browser refID="Default">
<controlAdapters>
<adapter controlType="System.Web.UI.HtmlControls.HtmlForm"
adapterType="Sample.Web.UI.Adapters.FormRewriterControlAdapter" />
</controlAdapters>
</browser>
</browsers>
用来指定Control Adapter。这样就OK了。