我一直在面对类似的问题。然而,VirtualPathProvider仅仅是为了实现如此小的收益而过于沉重 - 更不用说它似乎有可能在安全方面有点风险实施。我已经发现了两个可能的变通:
1)使用反射来获取你想要的东西:
var page = HttpContext.Current.Handler as Page;
string text = "<table><tr><td>Testing!!!</td></tr></table>";
var systemWebAssembly = System.Reflection.Assembly.GetAssembly(typeof(Page));
var virtualPathType = systemWebAssembly.GetTypes().Where(t => t.Name == "VirtualPath").FirstOrDefault(); // Type.GetType("System.Web.VirtualPath");
var createMethod = virtualPathType.GetMethods(System.Reflection.BindingFlags.Static | System.Reflection.BindingFlags.Public).Where(m => m.Name == "Create" && m.GetParameters().Length == 1).FirstOrDefault();
object virtualPath = createMethod.Invoke(null, new object[]
{
page.AppRelativeVirtualPath
});
var template = (ITemplate)typeof(TemplateParser).GetMethod("ParseTemplate", System.Reflection.BindingFlags.Static | System.Reflection.BindingFlags.NonPublic).Invoke(null, new object[]{text, virtualPath, true});
2)使用一个有点哈克变通:
var page = HttpContext.Current.Handler as Page;
string text = "<table><tr><td>Testing!!!</td></tr></table>";
string modifiedText = string.Format("<asp:UpdatePanel runat=\"server\"><ContentTemplate>{0}</ContentTemplate></asp:UpdatePanel>", text);
var control = page.ParseControl(modifiedText);
var updatePanel = control.Controls[0] as UpdatePanel;
var template = updatePanel.ContentTemplate;
我公开的管理员认为这两者都不是很好的解理想情况下,在.Net框架中会有这种方法。类似于:
public class TemplateParser
{
public static ITemplate ParseTemplate(string content, string virtualPath, bool ignoreParserFilter)
{
return TemplateParser.ParseTemplate(string content, VirtualPath.Create(virtualPath), ignoreParserFilter);
}
}
这将缓解整个实施VirtualPathProvider的需要。也许我们会在ASP.NET vNext中看到:-)
我在Sitefinity也看到过这个,如果我没有记错的话,我认为这是最好的选择。或者你可以为ascx标记创建你自己的解析器,但这不是微不足道的。 – Robert 2011-03-22 15:27:06
是的,我认为很多基于.NET的CMS和其他可扩展应用程序都使用这种方法。它看起来像是微不足道的,我发现很难理解为什么有一个物理的ASCX文件和一个字符串变量有什么不同。 – 2011-03-22 20:16:20