秋梦无痕

一场秋雨无梦痕,春夜清风冻煞人。冬来冷水寒似铁,夏至京北蟑满城。

Avatar

在WinForm程序中嵌入ASP.NET

发信人: flier (小海 [寻找风车中]), 信区: DotNET
标 题: 在WinForm程序中嵌入ASP.NET
发信站: BBS 水木清华站 (Mon Feb 9 20:53:20 2004), 转信

在WinForm程序中嵌入ASP.NET

现在的流行趋势是桌面程序Web化,Web程序桌面化,呵呵。最终目标就是你中有我,我中有你。例如MSN Explorer就是一个很好的展示,让用户在使用的时候分不清什么时候是在本地什么时候是在网络。而这类程序往往需要有一个后台服务器如IIS的支持,这对大多数桌面应用来说too heavy了。本着简单就是美的设计思想,这里给出一个轻量级的解决方法,把ASP.NET嵌入到普通WinForm桌面程序中去。
因为安全以及其它一些方面的原因,在使用ASP.NET引擎之前,必须建立一个新的AppDomain。简单的方法是直接使用ApplicationHost.CreateApplicationHost函数为指定的虚拟目
录和物理路径建立ASP.NET引擎宿主的实例,如
// should create a subdirectory ./bin and copy the assembly to it
static public WebHost Create(string name, string path)
{
if(!name.StartsWith(new string(Path.AltDirectorySeparatorChar, 1)))
{
name = Path.AltDirectorySeparatorChar + name;
}

WebHost host = (WebHost)ApplicationHost.CreateApplicationHost(
typeof(WebHost), name, path);

host.setVirtualDirectory(name);
host.setBaseDirectory(path);

return host;
}

但这样建立的程序有个BT的要求,他会在指定目录的bin子目录中去尝试载入宿主类型(WebHost)的assembly,也就是说你必须把程序在bin子目录下复制一份,非常不爽。解决方法是自己手工完成整个建立过程,如下:

static public WebHost Create(string virtualDir, string physicalDir)
{
if(!virtualDir.StartsWith(new string(Path.AltDirectorySeparatorChar, 1)))
{
virtualDir = Path.AltDirectorySeparatorChar + virtualDir;
}

if(!physicalDir.EndsWith(new string(Path.DirectorySeparatorChar, 1)))
{
physicalDir += Path.DirectorySeparatorChar;
}

AppDomainSetup setup = new AppDomainSetup();

setup.ApplicationName = "APP_" + Guid.NewGuid().ToString();
setup.ConfigurationFile = "web.config";

AppDomain domain = AppDomain.CreateDomain("ASPHOST_" + Guid.NewGuid().ToString(), null, setup);

domain.SetData(".appDomain", "*");
domain.SetData(".appPath", physicalDir);
domain.SetData(".appVPath", virtualDir);
domain.SetData(".domainId", domain.FriendlyName);
domain.SetData(".hostingVirtualPath", virtualDir);
domain.SetData(".hostingInstallDir", HttpRuntime.AspInstallDirectory);

WebHost host = (WebHost)domain.CreateInstanceAndUnwrap(
typeof(WebHost).Module.Assembly.FullName, typeof(WebHost).FullName);

host.setApplicationDomain(domain);
host.setVirtualDirectory(virtualDir);
host.setBaseDirectory(physicalDir);

return host;

}

这儿的一堆domain.SetData是传递参数给ASP.NET引擎。然后在那个appdomain中建立新的宿主类型的实例。这样就避免多份代码的尴尬。而使用ASP.NET就比较简单了,在宿主类中使用HttpRuntime.ProcessRequest函数处理特定请求。简单一点的话,可以直接用SimpleWorkerRequest包装请求,生成页面到一个指定的TextWriter中,如

private void DoRequest(string page, string query, TextWriter writer)
{
HttpRuntime.ProcessRequest(new SimpleWorkerRequest(page, query, writer));
}

public void RequestPage(string page, string query, Stream stream)
{
DoRequest(page, query, new StreamWriter(stream));
}

public void RequestPage(string page, Stream stream)
{
RequestPage(page, null, stream);
}

public string RequestPage(string page, string query)
{
using(StringWriter writer = new StringWriter())
{
DoRequest(page, query, writer);

return writer.ToString();
}
}

public string RequestPage(string page)
{
return RequestPage(page, string.Empty);
}

这个缺省的请求包装使用是简单,但对中文的兼容性不太好,过两天有空再自己写个强一点的吧,呵呵(注:吴雨认为使用UTF-8发送即可)

最终类的使用就比较简单了,在WinForm程序中建立一个singleton模式的属性

static private WebHost.WebHost _host = null;

public WebHost.WebHost Host
{
get
{
if(_host == null)
{
_host = WebHost.WebHost.Create();
}
return _host;
}
}

然后请求指定的asp.net页面,如

HTML = Host.RequestPage(_page);

即可完成从动态的asp.net脚本到静态html的转换。嵌入WinForm程序中,还可以通过Host类型完成两者之间的双向通讯,实现互相控制。

参考资料:
1.Using the ASP.Net Runtime for extending desktop applications with dynamic HTML Scripts
2.Executing ASMX files without a web server
3.ASP. NET Client-side Hosting with Cassini
4.Using ASP.NET Runtime in Desktop Applications

评论已关闭