MonoRail学习笔记五:定制服务实现自定义功能
在上一篇MonoRail学习笔记四:MonoRail基本流程分析 中我提到,MonoRail中可以自定义一些服务。比如可以定义自己的Url解析类,来实现http://localhost:***/index.rails 等http://localhost:***/*.rails 的效果。
具体步骤如下:
1、修改web.config文件,在monorail节中加入以下定义
<services>
<service id="UrlTokenizer" type="TestSiteNVelocity.CustomUrlTokenizer, TestSiteNVelocity" />
</services>
2、编写自己的CustomUrlTokenizer类
这里为了方便,我直接复制默认的Castle.MonoRail.Framework.Services.DefaultUrlTokenizer类,将复制好的类放入TestSiteNVelocity,改名为CustomUrlTokenizer,然后在此基础上修改。
当然,这个类中有很多方法,也就可以自定义很多功能,为了实现http://localhost:***/*.rails 的效果,我们只需要修改ExtractAreaControllerAction方法。
原方法:
private void ExtractAreaControllerAction(string rawUrl, out string area, out string controller, out string action)
{
string[] parts = rawUrl.Split('/');

if (parts.Length < 2)
{
throw new UrlTokenizerException("Url smaller than 2 tokens");
}

action = parts[parts.Length - 1];

int fileNameIndex = action.IndexOf('.');

if (fileNameIndex != -1)
{
action = action.Substring(0, fileNameIndex);
}

controller = parts[parts.Length - 2];

area = string.Empty;

if (parts.Length - 3 == 0)
{
area = parts[parts.Length - 3];
}
else if (parts.Length - 3 > 0)
{
StringBuilder areaSB = new StringBuilder();

for(int i = 0; i <= parts.Length - 3; i++)
{
if (parts[i] != null && parts[i].Length > 0)
{
areaSB.Append(parts[i]).Append('/');
}
}

if (areaSB.Length > 0)
{
areaSB.Length -= 1;
}

area = areaSB.ToString();
}
}
修改后的方法:
private void ExtractAreaControllerAction(string rawUrl, out string area, out string controller, out string action)
{
string[] parts = rawUrl.Split('/');

action = parts[parts.Length - 1];

int fileNameIndex = action.IndexOf('.');

if (fileNameIndex != -1)
{
action = action.Substring(0, fileNameIndex);
}

if (parts.Length < 2)
{
controller = "servlet"; ;
area = "";

return;
}

controller = parts[parts.Length - 2];

area = string.Empty;

if (parts.Length - 3 == 0)
{
area = parts[parts.Length - 3];
}
else if (parts.Length - 3 > 0)
{
StringBuilder areaSB = new StringBuilder();

for(int i = 0; i <= parts.Length - 3; i++)
{
if (parts[i] != null && parts[i].Length > 0)
{
areaSB.Append(parts[i]).Append('/');
}
}

if (areaSB.Length > 0)
{
areaSB.Length -= 1;
}

area = areaSB.ToString();
}
}
红色标示的为修改部分
其实意思很简单,就是当访问http://localhost:***/*.rails 形式的页面时,默认的controller类为ServletController类
接下来就可以按常规方式编写ServletController类和view部分
这样之后当我们访问http://localhost:***/index.rails时,调用的就是ServletCOntroller类的Index方法,当访问http://localhost:***/bag.rails 时,调用的就是ServletCOntroller类的Bag方法 ......
当然我们也同样可以定义
[DefaultAction("Index")]
public class ServletController : Controller
让所有未定义的http://localhost:***/*.rails访问调用ServletController的Index方法
这篇文章只是涉及了自定义服务的很小的一个点,我们还可以自定义:
ControllerFactory
ViewComponentFactory
ResourceFactory
EmailSender
EmailTemplateService
UrlBuilder
ValidatorRegistry
等很多服务,来实现特定效果
具体步骤如下:
1、修改web.config文件,在monorail节中加入以下定义



2、编写自己的CustomUrlTokenizer类
这里为了方便,我直接复制默认的Castle.MonoRail.Framework.Services.DefaultUrlTokenizer类,将复制好的类放入TestSiteNVelocity,改名为CustomUrlTokenizer,然后在此基础上修改。
当然,这个类中有很多方法,也就可以自定义很多功能,为了实现http://localhost:***/*.rails 的效果,我们只需要修改ExtractAreaControllerAction方法。
原方法:














































修改后的方法:

















































其实意思很简单,就是当访问http://localhost:***/*.rails 形式的页面时,默认的controller类为ServletController类
接下来就可以按常规方式编写ServletController类和view部分
这样之后当我们访问http://localhost:***/index.rails时,调用的就是ServletCOntroller类的Index方法,当访问http://localhost:***/bag.rails 时,调用的就是ServletCOntroller类的Bag方法 ......
当然我们也同样可以定义
[DefaultAction("Index")]
public class ServletController : Controller
让所有未定义的http://localhost:***/*.rails访问调用ServletController的Index方法
这篇文章只是涉及了自定义服务的很小的一个点,我们还可以自定义:
ControllerFactory
ViewComponentFactory
ResourceFactory
EmailSender
EmailTemplateService
UrlBuilder
ValidatorRegistry
等很多服务,来实现特定效果
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· DeepSeek如何颠覆传统软件测试?测试工程师会被淘汰吗?