最近有个基于tcp socket 协议和设备交互需求,想到了新生命团队的各种组件,所以决定用NewLife网络库作为服务端来完成一系列的信息交互.
第一,首先说一下我们需要实现的功能需求吧
1,首先客户有一堆自动售货机的设备,设备连接socket服务端后 定时发送设备实时状态作为心跳信息,并且服务端需要下发信息予以确认。
2,需要知道设备的实时在线状态
3,设备需要实现微信,支付宝扫码支付需求,当客户买东西的时候选择扫码支付时,设备上报产品价格信息,支付方式,服务器下发微信或者支付宝的当面付二维码。
4,当客户扫码支付后,服务器需要下发支付结果。
5,设备上报出货成功或失败信息,服务器予以回复确认
6,设备上报现金收款的收款或者退款信息,服务器保存并且下发回复确认。
7,服务器后台需要能够主动的向设备推送广告信息,并且能够实现掉线设备的广告推送
第二,我们来分析这个可靠的服务器的大致结构是如何的(仅包含设备实时状态和支付交互)
1,想要服务稳定可靠,那么首先考虑的是程序的运行方式,当然是以一个windows服务为宿主的方式来运行最为可靠,那么我们可以使用TopShelf来构建这样一个服务宿主
2,实现设备的在线状态实时可知,我们需要定时的去判断这个设备是否在定时给服务器发送心跳信息,这个项目中我们以超时未发送心跳信息为标准来衡量他是否在线。
3,设备数据上报涉及到粘包的处理,设备上报的数据是怎么样一个协议,如何来解决粘包的问题,这里我们借助Newlife网络库的管道模式来处理粘包
4,微信支付宝的扫码支付如何及时知道客户付款状态,并且及时的下发给设备。
第三,我们需要用到的第三方类库有哪些
1,首先使用的网络库是Newlife 新生命团队的网络库
2,构建windows服务使用的是TopShelf
3,状态的信息保存使用Redis,这里我们使用CSRedisCore 这个类库
4,微信支付宝本次使用的是PaySharp这个库,二维码生成使用QrCoder库
第四,开始构建的基本的Socket服务吧
1,首先创建一个windows服务,引入Topshelf后
static void Main(string[] args) { var rc = HostFactory.Run(x => //1 { x.Service<DianSocketMain>(s => //2 { s.ConstructUsing(name =>new DianSocketMain()); //3 s.WhenStarted(tc => tc.Start()); //4 s.WhenStopped(tc => tc.Stop()); //5 }); x.RunAsLocalSystem(); //6 x.SetDescription("售货机socket服务端"); //7 x.SetDisplayName("Vending machine socket"); //8 x.SetServiceName("Vendingmachinesocket"); //9 x.OnException(s => { LogerHelper.WriteErroLog("test",s.Message); }); }); //10 }
DianSocketMain类就是具体的Socket服务端,来看下是如何启动一个基本的socket服务端
public class DianSocketMain { /// <summary> ///网络库服务端 20181225 /// </summary> private NetServer _newLifeServer; /// <summary> /// /// </summary> public void Start() { try { try { //redis初始化 string redisHost = CoomHelper.GetAppSettings("RedisHost", "127.0.0.1:6379"); string redisPwd = CoomHelper.GetAppSettings("RedisPwd", ""); string prefix = "{device_server}:"; string redisConn = $"{redisHost},password={redisPwd},defaultDatabase=0,poolsize=50,ssl=false,writeBuffer=10240,prefix={prefix}"; var csredis = new CSRedis.CSRedisClient(redisConn); RedisHelper.Initialization(csredis); } catch (Exception e) { XTrace.WriteLine(e.Message); } //启动客户端 StartNewLifeListenClient(); } catch (Exception e) { XTrace.WriteException(e); } } /// <summary> /// 停止win服务 /// </summary> public void Stop() { try { if (_newLifeServer != null) { _newLifeServer.Dispose(); } } catch (Exception e) { XTrace.WriteException(e); } } /// <summary> /// 监听客户端 /// </summary> private void StartNewLifeListenClient() { try { XTrace.WriteLine("当前tcp端口号为:" + 9046); //创建监听地址和端口 IPAddress ipAddress = IPAddress.Parse("127.0.0.1"); var svr = new NetServer(ipAddress, 9046, NetType.Tcp) { Log = XTrace.Log, SessionTimeout = 60 }; svr.Add<ReciveFilter>();//粘包处理管道 svr.Received += new EventHandler<ReceivedEventArgs>(NewlifeRecive);//数据接收 //svr.Error += new EventHandler<ExceptionEventArgs>(NewlifeError);//错误处理 //svr.OnDisposed+=new EventHandler(); svr.Start(); _newLifeServer = svr; //XTrace.WriteLine("会话超时时间为:"+svr.SessionTimeout); } catch (Exception ex) { XTrace.WriteLine(ex.Message); } } /// <summary> /// 数据接收 /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private void NewlifeRecive(object sender, ReceivedEventArgs e) { INetSession session = (INetSession)sender; var pk = e.Message as Packet; if (pk.Count == 0) { XTrace.WriteLine("数据包解析错误"); return; } } }
这里包含的一个有redis的初始化,粘包处理,数据接收的一个基本服务端,下面我们再讲如何解决粘包的问题。