Self Host 使得Nancy 能够在任意application 中启动,无论是console 还是windows service。这期我们使用的版本是Nancy v0.4.0。

Demo

首先看一下使用方式:

class Program
{
static void Main(string[] args)
{
var nancyHost = new NancyHost(new Uri("http://localhost:8888/nancy/"));
nancyHost.Start();
Console.WriteLine("Nancy now listening - navigate to http://localhost:8888/nancy/. Press enter to stop");
Console.ReadKey();
nancyHost.Stop();
Console.WriteLine("Stopped. Good bye!");
}
}

这是一个控制台的例子,我们在Main方法中直接创建一个NancyHost的实例,然后调用nancyHost.Start()。warning: 当前版本的实现为早期版本,仅仅是为了演示而使用,请勿deploy在生产环境中。

   public NancyHost(Uri baseUri, INancyBootstrapper bootStrapper)
{
this.baseUri = baseUri;
listener = new HttpListener();
listener.Prefixes.Add(baseUri.ToString()); bootStrapper.Initialise();
engine = bootStrapper.GetEngine();
}

NancyHost 需要两个参数,Uri,以及INancyBootstrapper。Uri这里我们传入的是new Uri("http://localhost:8888/nancy/"。INancyBootstrapper这个参数可以省略,Nancy 启动的时候会反射加载所有实现了INancyBootstrapper的启动器实现。INancyBootstrapper可以用来作依赖注入框架的启动器,实现该接口的类有:

可以看到除了测试用的fake实现,都是一些IOC框架(Ninject,StructureMap,Unity......)。NancyBootstrapperLocator类具体负责寻找实现了该接口的启动器,如果找不到,就会使用默认的TinyIoCContainer作为IOC 容器。

static NancyBootstrapperLocator()
{
// Get the first non-abstract implementation of INancyBootstrapper if one exists in the
// app domain. If none exist then just use the default one.
var bootstrapperInterface = typeof(INancyBootstrapper);
var defaultBootstrapper = typeof(DefaultNancyBootstrapper); var locatedBootstrappers = from assembly in AppDomain.CurrentDomain.GetAssemblies()
where !assembly.IsDynamic
from type in assembly.SafeGetExportedTypes()
where !type.IsAbstract
where bootstrapperInterface.IsAssignableFrom(type)
where type != defaultBootstrapper
select type; var bootstrapperType = locatedBootstrappers.FirstOrDefault() ?? defaultBootstrapper; Bootstrapper = (INancyBootstrapper) Activator.CreateInstance(bootstrapperType);
}

我们可以看到所有默认接口都指定了Default的实现。

private IEnumerable<TypeRegistration> BuildDefaults()
{
return new[]
{
new TypeRegistration(typeof(IRouteResolver), DefaultRouteResolver),
new TypeRegistration(typeof(INancyEngine), DefaultNancyEngine),
new TypeRegistration(typeof(IModuleKeyGenerator), DefaultModuleKeyGenerator),
new TypeRegistration(typeof(IRouteCache), DefaultRouteCache),
new TypeRegistration(typeof(IRouteCacheProvider), DefaultRouteCacheProvider),
new TypeRegistration(typeof(IRoutePatternMatcher), DefaultRoutePatternMatcher),
new TypeRegistration(typeof(IViewLocator), DefaultViewLocator),
new TypeRegistration(typeof(IViewFactory), DefaultViewFactory),
new TypeRegistration(typeof(INancyContextFactory), DefaultContextFactory),
new TypeRegistration(typeof(INancyModuleBuilder), DefaultNancyModuleBuilder),
new TypeRegistration(typeof(IResponseFormatter), DefaultResponseFormatter)
};
}

Nancy 面向接口编程做的比较的不错。接下去,我们继续完成NancyHost的初始化,这里创建一个HttpListener的实例。HttpListener是System.dll中提供的一个类:

  /// <summary>
/// Provides a simple, programmatically controlled HTTP protocol listener. This class cannot be inherited.
/// </summary>
public sealed class HttpListener : IDisposable

有了这个类就方便了很多,我们不需要自己去处理socket。接下去我们启动NancyHost。

    public void Start()
{
shouldContinue = true; listener.Start();
thread = new Thread(Listen);
thread.Start();
}

这里就是这个简易版本的问题所在了,程序只是启动了一个线程去处理所有的http请求。在线程中,将所有的context,request作了一轮适配器一样的转换,最终将NancyResponse转变会HttpListener需要的Response。不过虽然同样是一个死循环的写法,while (shouldContinue)的做法明显是优于while(true)的,无则加勉,大家要有这个意识。

改进Self Host

单线程阻塞当然不妥当,不过这样就基本实现了一个简易的Self Host。接下去,让我们把视野切换到版本1.4.1。

   private void StartListener()
{
if (this.TryStartListener())
{
return;
} if (!this.configuration.UrlReservations.CreateAutomatically)
{
throw new AutomaticUrlReservationCreationFailureException(this.GetPrefixes(), this.GetUser());
} if (!this.TryAddUrlReservations())
{
throw new InvalidOperationException("Unable to configure namespace reservation");
} if (!TryStartListener())
{
throw new InvalidOperationException("Unable to start listener");
}
}
}

同样是HttpListener,这里的启动更加的成熟,增加了不少错误处理的判断。

    public void Start()
{
this.StartListener(); try
{
this.listener.BeginGetContext(this.GotCallback, null);
}
catch (Exception e)
{
this.configuration.UnhandledExceptionCallback.Invoke(e); throw;
}
}

在启动完HttpListener后,这里最大的不同是使用异步回调的方式,不再是单线程阻塞模式。

private void GotCallback(IAsyncResult ar)
{
try
{
var ctx = this.listener.EndGetContext(ar);
this.listener.BeginGetContext(this.GotCallback, null);
this.Process(ctx);
}
catch (Exception e)
{
this.configuration.UnhandledExceptionCallback.Invoke(e); try
{
this.listener.BeginGetContext(this.GotCallback, null);
}
catch
{
this.configuration.UnhandledExceptionCallback.Invoke(e);
}
}
}

采用标准的回调处理,每个请求将会在它自己的独立线程中执行。

private void Process(HttpListenerContext ctx)
{
try
{
var nancyRequest = this.ConvertRequestToNancyRequest(ctx.Request);
using (var nancyContext = this.engine.HandleRequest(nancyRequest))
{
try
{
ConvertNancyResponseToResponse(nancyContext.Response, ctx.Response);
}
catch (Exception e)
{
this.configuration.UnhandledExceptionCallback.Invoke(e);
}
}
}
catch (Exception e)
{
this.configuration.UnhandledExceptionCallback.Invoke(e);
}
}

至此,self host相关内容结束。

当当当当 - つづく

.NET Nancy 详解(四) Self Host的更多相关文章

  1. .NET DLL 保护措施详解(四)各操作系统运行情况

    我准备了WEB应用程序及WinForm应用程序,分别在WIN SERVER 2012/2008/2003.Win7/10上实测,以下为实测结果截图: 2012 2008 2003 WIN7 WIN10 ...

  2. logback -- 配置详解 -- 四 -- <filter>

    附: logback.xml实例 logback -- 配置详解 -- 一 -- <configuration>及子节点 logback -- 配置详解 -- 二 -- <appen ...

  3. pika详解(四) channel 通道

    pika详解(四) channel 通道   本文链接:https://blog.csdn.net/comprel/article/details/94662394 版权 ​ channel通道 通道 ...

  4. View绘制详解(四),谝一谝layout过程

    上篇博客我们介绍了View的测量过程,这只是View显示过程的第一步,第二步就是layout了,这个我们一般译作布局,其实就是在View测量完成之后根据View的大小,将其一个一个摆放在ViewGro ...

  5. tomcat 配置文件server.xml 详解 Connector Engine Host Context

    目录 一 server.xml 1.1 server 配置 1.2 service 配置 1.3 Executor 1.4 Connector 配置 1.5 Engine 其他tocmat 文章 一 ...

  6. C++11 并发指南六(atomic 类型详解四 C 风格原子操作介绍)

    前面三篇文章<C++11 并发指南六(atomic 类型详解一 atomic_flag 介绍)>.<C++11 并发指南六( <atomic> 类型详解二 std::at ...

  7. Android Studio 插件开发详解四:填坑

    转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/78265540 本文出自[赵彦军的博客] 在前面我介绍了插件开发的基本流程 [And ...

  8. webRTC中语音降噪模块ANS细节详解(四)

    上篇(webRTC中语音降噪模块ANS细节详解(三))讲了噪声的初始估计方法以及怎么算先验SNR和后验SNR. 本篇开始讲基于带噪语音和特征的语音和噪声的概率计算方法和噪声估计更新以及基于维纳滤波的降 ...

  9. Hadoop伪分布安装详解(四)

    目录: 1.修改主机名和用户名 2.配置静态IP地址 3.配置SSH无密码连接 4.安装JDK1.7 5.配置Hadoop 6.安装Mysql 7.安装Hive 8.安装Hbase 9.安装Sqoop ...

随机推荐

  1. 17.3---阶乘尾多少个0(CC150)

    思路,其实这题easy,就是看多少个5. 答案: public static int getFactorSuffixZero(int n) { // write code here int res = ...

  2. python egg文件解压

    unzip 就可以了. 由于项目需要将某些版本的库打包,然后 sys.path.insert方式引用(避免升级包导致某些旧的系统崩掉). 在将egg文件打包时,发现不可用.但相关模块的__path__ ...

  3. 应用HTK搭建语音拨号系统2:创建单音素HMM模型

    选自:http://maotong.blog.hexun.com/6204849_d.html 苏统华 哈尔滨工业大学人工智能研究室 2006年10月30日 声明:版权所有,转载请注明作者和来源 该系 ...

  4. 1个简单的Log

    #pragma once #include <windows.h> #include <process.h> class CLogger { public: static CR ...

  5. Js 的 this 是什么

    this指的是调用函数的那个对象,谁调用函数,谁就是this 更多: http://www.cnblogs.com/justany/archive/2012/11/01/the_keyword_thi ...

  6. SharePoint2010母版页想要的定制

    查找<div id="s4-ribbonrow" class="s4-pr s4-ribbonrowhidetitle"用style="disp ...

  7. 【leetcode】Edit Distance

    Edit Distance Given two words word1 and word2, find the minimum number of steps required to convert  ...

  8. 无IDE时编译和运行Java

    最近 Java subreddit 出现了一篇”在没有IDE的情况下编译Java包” 的帖子,这个帖子抛出了这么一个问题,“是否存在一个命令可以编译一组处于同一文件夹下独立包内的java文件的方法(这 ...

  9. 【XLL API 函数】xlGetName

    以字符串格式返回 DLL 文件的长文件名. 原型 Excel12(xlGetName, LPXLOPER12 pxRes, 0); 参数 这个函数没有参数 属性值和返回值 返回文件名和路径 实例 \S ...

  10. IOS - 消息推送原理和实现

    一.消息推送原理: 在实现消息推送之前先提及几个于推送相关概念,如下图1-1: 1.Provider:就是为指定IOS设备应用程序提供Push的服务器,(如果IOS设备的应用程序是客户端的话,那么Pr ...