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. ubuntu14.04 163sources.list

    deb http://mirrors.163.com/ubuntu/ trusty main restricted universe multiverse deb http://mirrors.163 ...

  2. c++转义字符、指针

    上篇博客的答案: 1: // DataTypeDemo.cpp : 定义控制台应用程序的入口点. 2: // 3:  4: #include "stdafx.h" 5: #incl ...

  3. java map的四种遍历

    四种遍历: public static void main(String[] args) { Map<String, String> map = new HashMap<String ...

  4. Unity3d 枚举某个目录下所有资源

    using UnityEngine; using System.Collections; using UnityEditor; using System.Collections.Generic; us ...

  5. frxReport 设计 (mtm)

    ► 设计  frxReport  frxReport 窗体上放一个  [frxReport] 的控件 双击 [frxReport]控件 进入设置模式 frxReport1.ShowReport() 方 ...

  6. ACM/ICPC 之 Bellman Ford练习题(ZOJ1791(POJ1613))

    这道题稍复杂一些,需要掌握字符串输入的处理+限制了可以行走的时间. ZOJ1791(POJ1613)-Cave Raider //限制行走时间的最短路 //POJ1613-ZOJ1791 //Time ...

  7. FFMpeg 滤镜中英文对照

    FFMpeg ver 20160213-git-588e2e3 滤镜中英文对照 2016.02.17 by 1CM T.. = Timeline support 支持时间轴 .S. = Slice t ...

  8. Effective C++ -----条款53:不要轻忽编译期的警告

    严肃对待编译器发出的警告信息.努力在你的编译器的最高(最严苛)警告级别下争取“无任何警告”的荣誉. 不要过度依赖编译器的报警能力,因为不同的编译器对待事情的态度并不相同.一旦移植到另一个编译器上,你元 ...

  9. Greedy:The Water Bowls(POJ 3185)

    水池 题目大意:给定一个20的数组,全都是0和1,可以翻一个数改变成另一个数(0或者1),但是其左右两边的数都会跟着变为原来的相反数,问你怎么用最小的操作数使全部数变成0 这一题的:满足 1:翻转次序 ...

  10. Luncence .Net 使用

    public partial class Form1 : Form { public Form1() { InitializeComponent(); } //标准分词 private void bu ...