在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);
}
这个缺省的请求包装使用是简单,但对中文的兼容性不太好,过两天有空再自己写个强一点的吧,呵呵
最终类的使用就比较简单了,在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
::URL::http://www.west-wind.com/presentations/aspnetruntime/aspnetruntime.asp
2.Executing ASMX files without a web server
::URL::http://radio.weblogs.com/0105476/stories/2002/10/24/executingAsmxFilesWithoutAWebServer.html
3.ASP. NET Client-side Hosting with Cassini
::URL::http://msdn.microsoft.com/msdnmag/issues/03/01/CuttingEdge/
4.Using ASP.NET Runtime in Desktop Applications
::URL::http://www.codeguru.com/cs_internet/UsingAspRuntime.html
放下
flier_lu 发表于 >2004-2-9 20:32:17← 保存该日志到本地
作者:Kriss
大侠有空还是写个功能完善点的WEB宿主吧:)
cassini里的那个是针对http和socket的,我似乎用不到啊
我是在一个类似于新闻系统里用到,准备根据ASPX文件产生HTML文件,所以只要用这个宿主把解析结果存起来就行了
现在的问题还是子目录的问题
实在不行就只好把文件移到根目录解析了:(
flier_lu 在 Kriss 的文章中回复道:
呵呵,列入计划吧,技术上是问题不大的。不过最近手头工作比较忙,一堆技术预研工作和BLog的功课要做,只能说争取了 😛
作者:Kriss
但这样解决似乎仍有一个问题,Create(vPath,pPath) 创建的是一个Web应用程序宿主,肯定会在其目录寻找dll以及web.config等文件。如果我指定目录为 d:hooyeehelp,那help因为不是独立的web应用程序,应该会出错,说找不到dll和web.config
我从asp.net网站下载过微软自己的那个WEB宿主的例子看过,它那里创建好宿主后,好像是可以访问直接子目录的文件的
flier_lu 在 Kriss 的文章中回复道:
你是指 ASP.NET 上实现的 cassini 那个 ASP.NET 宿主 Web 服务器的例子吗?人家可以实现了一个完整的 SimpleWorkerRequest 的子类,而我例子中偷懒直接用SimpleWorkerRequest 完成,功能上当然会有很大差距,呵呵,差了近千行代码呢 😛
作者:Kriss
我使用了这个方法,利用WebHost从aspx产生静态html,但是有一个问题,如果我请求的文件不是默认路径,而是某个子目录下面的,会出现无法找到aspx文件的信息:
WebHost host = WebHost.Create("/hooyee","c:\hooyee\web";
string html;
html = host.RequestPage("Default.aspx";
这样是没有问题的
html = host.RequestPage("Help/Default.aspx";
这样产生的HTML文本里是错误信息:404错误,无法找到指定资源
我查了一些文章但还是找不到解决方法。特来请教
另外,中文问题我也碰到了,如果从RequestPage中获取 Stream 的话,中文是乱码。但如果利用上面的方法,先返回到 string ,再把这个string保存到文件里,就没有问题了。有没有别的解决方案?
flier_lu 在 Kriss 的文章中回复道:
这个404错误的原因是你指定虚拟路径的地方错误。host.RequestPage的参数应该只是页面的名字,如果需要打开子目录下页面,则应该在WebHost.Create下面指定,如将WebHost host = WebHost.Create("/hooyee","c:\hooyee\web");
改为WebHost host = WebHost.Create("/hooyee","c:\hooyee\web\Help");
就可以打开 Help 子目录下的页面了
至于那个中文问题,我现在也没有什么好的解决办法,希望等有空把ASP.NET的机制完整分析一遍后,能找到解决方法。😃
作者:lionsky_net@hotmail.com
哦,一时马虎,没有注意到,呵呵
作者:lionskynet_@hotmail.com
好文章,不过为什么我的SDK找不到WebHost?
flier_lu 在 lionskynet_@hotmail.com 的文章中回复道:
这个WebHost是我自己写的一个简单的包装类啊,呵呵。主要是对host的建立和页面请求做了简单的封装。
以下为引用:
using System;
using System.IO;
using System.Text;
using System.Web;
using System.Web.Hosting;
using System.Runtime.Remoting;
namespace Navigator.WebHost
{
/// <summary>
/// Summary description for WebHost.
/// </summary>
public class WebHost : MarshalByRefObject
{
#if false
// 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;
}
#else
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;
}
#endif
static public WebHost Create(string name)
{
return Create(name, Environment.CurrentDirectory);
}
static public WebHost Create()
{
return Create(string.Empty, Environment.CurrentDirectory);
}
private AppDomain _appDomain = null;
private string _virtualDir;
private string _baseDir;
protected void setApplicationDomain(AppDomain domain)
{
_appDomain = domain;
}
protected void setVirtualDirectory(string dir)
{
_virtualDir = dir;
}
protected void setBaseDirectory(string dir)
{
_baseDir = dir;
}
public AppDomain ApplicationDomain
{
get
{
return _appDomain;
}
}
public string VirtualDirectory
{
get
{
return _virtualDir;
}
}
public string BaseDirectory
{
get
{
return _baseDir;
}
}
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);
}
}
}