构建可靠的网络服务

网络程序应该注册成为系统服务,以保证其自启动以及稳定可靠运行!

这一场,讲讲怎么建立一个生产级别的网络服务。

老规矩,先上源码:https://github.com/NewLifeX/NewLife.Net

系统服务功能,由网络库的兄弟框架,X组件的Agent来支撑,以前也叫XAgent,网上搜索 NewLife XAgent 可以找到不少文章。

XAgent在X组件里面很年轻,才10年,设计于2008年,上海陆家嘴。

〇、最终效果

先来看看最终效果,大家也可以telnet net.newlifex.com 1234 来看效果

图片

左边窗口就是这次要讲的网络服务程序,工作在调试模式。

右边窗口是上一次的EchoTest客户端,连接左边网络服务。

一、建立控制台项目

建立一个控制台项目,通过nuget引用NewLife.Core

新建一个服务类 MyService,继承自泛型基类 AgentServiceBase

Program.Main里面增加一行引导程序:

class Program
{    static void Main(String[] args)
    {        // 引导进入我的服务控制类        MyService.ServiceMain();
    }
}

下面就开始慢慢完善我们的服务类MyService

public MyService()
{
    ServiceName = "EchoAgent";
    DisplayName = "回声服务";
    Description = "这是NewLife.Net的一个回声服务示例!";    // 准备两个工作线程,分别负责输出日志和向客户端发送时间
    ThreadCount = 2;
    Intervals = new[] { 1, 5 };
}

定一些基本参数,看效果图可以猜到用途

图片

服务名、显示名、描述,就这么多!

ThreadCount = 2指定两个工作线程,Intervals指定它们的轮询周期分别是1秒和5秒

系统服务的标准动作就是启动和停止

MyNetServer _Server;/// <summary>开始服务</summary>/// <param name="reason"></param>protected override void StartWork(String reason)
{    // 实例化服务端,指定端口,同时在Tcp/Udp/IPv4/IPv6上监听
    var svr = new MyNetServer
    {
        Port = 1234,
        Log = XTrace.Log
    };
    svr.Start();

    _Server = svr;     base.StartWork(reason);
}/// <summary>停止服务</summary>/// <param name="reason"></param>protected override void StopWork(String reason)
{
    _Server.TryDispose();
    _Server = null;    base.StopWork(reason);
}

我们重载启动函数,初始化网络服务,并重启停止函数来销毁网络服务。

这里的MyNetServer从上一个例程拷贝过来。

网络服务做一个成员资源,避免被GC回收。

XAgent默认带来多线程任务调度,其核心是 Work(Int32 index)

/// <summary>调度器让每个任务线程定时执行Work,index标识任务</summary>/// <param name="index"></param>/// <returns></returns>public override Boolean Work(Int32 index)
{    switch (index)
    {        case 0: ShowStat(_Server); break;        case 1: SendTime(_Server); break;
    }    return false;
}private String _last;/// <summary>显示服务端状态</summary>/// <param name="ns"></param>private void ShowStat(NetServer ns)
{    var msg = ns.GetStat();    if (msg == _last) return;

    _last = msg;

    WriteLog(msg);
} /// <summary>向所有客户端发送时间</summary>/// <param name="ns"></param>private void SendTime(NetServer ns)
{    var str = DateTime.Now.ToFullString() + Environment.NewLine;    var buf = str.GetBytes();
    ns.SendAllAsync(buf);
}

XAgent内部设计有一个任务调度器,它会实际创建2个线程(ThreadCount指定),每个线程定时执行Work(Int32 index)函数,index参数用于标识哪一个任务线程。

我们这只需要一个很简单的switch,0号线程负责输出服务端状态,每秒一次,1号线程负责给连接到服务端的所有会话发送服务器当前时间。

多说几句XAgent:

1,任务线程具有较高线程优先级,比一般线程有更多机会得到CPU时间

2,调度器有个最高优先级的管理线程,负责监管所有任务线程,如果任务线程崩溃或者超时,它会干掉并新建

3,管理线程还负责监控线程数、句柄数、内存占用等

二、开发调试

既然是控制台项目,先跑起来看看:

图片

红色字体显示重要信息,黄色字体显示菜单,常用功能是235。

我们选择5,循环调试,其实就是在控制台里面模拟服务工作流程,让网络服务跑起来。

底下日志可以看到,它监听了4个套接字。

2是安装服务,也就是把当前应用安装成为Windows服务,这里特别注意,一般需要管理员权限,才能安装成功,除非关闭系统UAC。

3是启动服务,只有在安装了服务之后,才能看到。

所以,XAgent程序,既是开发调试控制台程序,也是安装卸载、启动停止服务的操作台,更是Windows服务程序本身!

细心的同学可以发现,安装好的Windows服务实质上就是 EchoAgent.exe -s,带有-s参数。

三、安装服务

最后,我们把它安装到一台公网服务器上,tcp://net.newlifex.com:1234,telnet上去看看效果

图片图片

从日志文件可以看到,它的应用类型 ApplicationType 是 Service,也就是Windows Service。

下面的日志,在A0线程(也就是0号任务线程)输出服务端状态。

在线1/1,当前在线/最大在线

发送 2/20/0,共发送2次,最大速度每秒20字节,当前速度每秒0字节

既然有A0线程,同样也会有A1,还会有An(ThreadCount>n),可用于区分不同任务线程输出的日志。

至此,我们的Windows网络服务程序开发完成,并安装到公网服务器上,持续对外提供Echo服务!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

智能大石头

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值