socket 一个websocke对应一个socketclient对象
using System;
using System.Collections.Concurrent;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Fleck;
namespace DB
{
/// <summary>
/// 客户端网络状态
/// </summary>
///
public enum NetStateEnum
{
[Description("已连接")]
Connected = 1,
[Description("已发送")]
SendData = 2,
[Description("已接收")]
ReceiveData = 3,
[Description("已解析")]
ParseData = 4,
[Description("已离线")]
Disconnected = 5,
[Description("上报超时")]
ReportTimeout = 6
}
/// <summary>
/// 记录每一个Socket连接
/// </summary>
public class SocketConnectionInfo
{
private const int SocketDataBufferSize = 1024;//如果单包数据字节数大于该值则需要加大此值,以免分包
/// <summary>
/// 构造函数将一个套接字和一个客户号码
/// </summary>
/// <param name="socket">套接字</param>
/// <param name="connectionId">设备唯一号</param>
internal SocketConnectionInfo(IWebSocketConnection socket, string connectionId)
{
ConnectionId = connectionId;
CurrentSocket = socket;
DataBuffer = new byte[SocketDataBufferSize];
}
/// <summary>
/// 析构函数
/// </summary>
~SocketConnectionInfo()
{
DataBuffer = null;
CurrentSocket = null;
}
public void Close()
{
if (CurrentSocket != null/* && CurrentSocket.Connected*/)
{
CurrentSocket.Close();
}
}
/// <summary>
/// 客户端的套接字
/// </summary>
public IWebSocketConnection CurrentSocket { get; set; }
/// <summary>
/// 由客户机发送缓冲区来存储数据
/// </summary>
public byte[] DataBuffer { get; set; }
/// <summary>
/// 当前实际接收的数据字节数,与属性DataBuffer结合使用以确定实际接收的数据
/// </summary>
///
public int DataBufferLen { get; set; }
/// <summary>
/// 数据接收时的系统时间
/// </summary>
public DateTime ReceivedTimeFromServer { get; set; }
/// <summary>
/// 最后一次接收的数据里的采集时间
/// </summary>
///
public DateTime ReceivedTimeFromClient { get; set; }
/// <summary>
/// 是否在线
/// </summary>
public bool IsAlive
{
get
{
if (CurrentSocket != null)
{
return CurrentSocket.IsAvailable;
}
else
{
return false;
}
}
}
/// <summary>
/// 用于标识socket连接
/// </summary>
public string ConnectionId { get; set; }
public NetStateEnum NetDataState { get; set; }
public byte[] LastUnParsedBytes { get; set; } // 缓存上次未解析的数据 缓存在每个连接中
public object ParsedEntity { get; set; }
/// <summary>
/// 是否禁用,由用户手工更改。禁用后的连接不处理其收发数据,收到数据直接抛弃。防止客户端数据高频发送无效数据。
/// </summary>
public bool IsDisabled { get; set; }
public bool IsLoggedIn { get; set; } //是否登录成功,只有登录成功的情况下才可以后续通信交互
/// <summary>
/// 当前使用的终端编号,每次通信都有可能修改
/// </summary>
public string CurrentPileCode { get; set; }
public string DeviceId { get; set; }
public string TemporaryHint { get; set; }
public string MessageTypeName { get; internal set; }
}
public class SocketConnectionInfoFactory
{
/// <summary>
/// 所有客户端Socket连接的集合,通过socket对象索引
/// </summary>
private ConcurrentDictionary<IWebSocketConnection, SocketConnectionInfo> dictionary = new ConcurrentDictionary<IWebSocketConnection, SocketConnectionInfo>();
public ConcurrentDictionary<IWebSocketConnection, SocketConnectionInfo> GetItems()
{
return dictionary;
}
public SocketConnectionInfo BindSocketConnectionInfo(IWebSocketConnection socket, string connectionId)
{
SocketConnectionInfo socketConnectionInfo;
if (dictionary.ContainsKey(socket))
{
socketConnectionInfo = dictionary[socket];
}
else
{
socketConnectionInfo = new SocketConnectionInfo(socket, connectionId);
dictionary.TryAdd(socket, socketConnectionInfo);
}
return socketConnectionInfo;
}
public SocketConnectionInfo GetSocketConnectionInfo(string uniqueId)
{
SocketConnectionInfo socketConnectionInfo = null;
foreach (var item in dictionary)
{
if (string.Compare(item.Value?.ConnectionId, uniqueId, true) == 0)
{
if (item.Value.ReceivedTimeFromServer >= socketConnectionInfo?.ReceivedTimeFromServer)
{
Remove(socketConnectionInfo.CurrentSocket);
}
else
{
socketConnectionInfo = item.Value;
}
}
}
return socketConnectionInfo;
}
public void Remove(IWebSocketConnection socket)
{
if (socket != null)
{
dictionary.TryRemove(socket, out SocketConnectionInfo value);
try
{
//判断此连接是否可用
if (socket.IsAvailable)
{
socket.Close();
}
}
catch
{
throw;
}
socket.Close();
value = null;
}
}
public void RemoveAll()
{
foreach (var item in dictionary)
{
item.Value?.Close();
}
dictionary.Clear();
}
public int Count
{
get
{
if (dictionary == null)
{
return 0;
}
return dictionary.Count;
}
}
}
}
调用方式
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace DB
{
public class DBHelper
{
private static object obj = new object();
private static SocketConnectionInfoFactory webSocket = null;
public static SocketConnectionInfoFactory GetInstance()
{
if (webSocket == null)
{
lock (obj)
{
if (webSocket == null)
{
webSocket = new SocketConnectionInfoFactory();
}
}
}
return webSocket;
}
}
}
socket 一个websocke对应一个socketclient对象的更多相关文章
- .Net开发笔记(十九) 创建一个可以可视化设计的对象
阅读本篇博客之前需要了解VS窗体设计器的工作原理,详细可参见本系列博客(十).(十一).(十二).必须需要知道的一条结论就是:处于窗体设计器(Form Designer)中的任何组件(包含控件,下同) ...
- threadlocal精髓是为每一个线程保证一个共享对象,保证一个,保证是同一个
threadlocal精髓是为每一个线程保证一个共享对象,保证一个,保证同一个线程中是同一个共享对象. 如果是静态变量是共享的话,那必须同步,否则尽管有副本,还是会出错,故C错
- Tiny Mapper是一个.net平台开源的对象映射组件
NET平台开源项目速览(14)最快的对象映射组件Tiny Mapper 阅读目录 1.Tiny Mapper基本介绍 2.Tiny Mapper 基本使用 3.Tiny Mapper 指定配置使用 ...
- 27、通过visual s'tudio 验证 SOCKET编程:搭建一个TCP服务器
本文就是在windows下进行socket编程,搭建一个TCP客户端. 在visual studio下编程,首先在windows下进行初始化(这点在linux下是不需要的): /* 初始化 Winso ...
- 通过一个uri获取一个Bitmap对象
Android 开发过程中,可能会用到的,通过一个uri获取一个Bitmap对象 private Bitmap getBitmapFromUri(Uri uri){ try { // 读取ur ...
- C#中如果用await关键字来await一个为null的Task对象会抛出异常
await & async模式是C#中一个很重要的特性,可以用来提高异步程序(多线程程序)的执行效率.但是如果尝试用await关键字来await一个为null的Task对象,会导致程序抛出Nu ...
- Object.assign() 从一个或多个源对象复制到目标对象
Object.assign()方法用于将所有可枚举属性的值从一个或多个源对象复制到目标对象.它将返回目标对象. 1.语法: Object.assign(target, ... , sources) 参 ...
- 创建一个doc对象时候 如果读取了一个已存在的xml对象时候 该xml对象的结构已存在doc中 当改变该doc结构时候 不需要创建新的doc对象
创建一个doc对象时候 如果读取了一个已存在的xml对象时候 该xml对象的结构已存在doc中 当改变该doc结构时候不 需要创建新的doc对象 直接添加即可 他会同步过去
- iter创建一个可以被迭代的对象
#!/usr/bin/env python obj = iter([11,22,33,44]) #iter 创建一个可以被迭代的对象 print(obj) r1 = next(obj) print(r ...
随机推荐
- JavaScript this 的指向问题
原文作者:SegmentFault ——写bug 原文链接:https://segmentfault.com/a/1190000015438195 this的指向已经是一个老生常谈的问题,每逢面试都要 ...
- python xlwt 设置单元格样式
使用xlwt中的Alignment来设置单元格的对齐方式,其中horz代表水平对齐方式,vert代表垂直对齐方式. VERT_TOP = 0x00 上端对齐 VERT_CENTER = 0x01 居中 ...
- &和&&,|和||的用法区别
&和&&的区别是,&会执行两边,不管第一个是否成立&&只会执行一边,如果第一个条件为假,则不会走第二个条件举例public class Test2{ p ...
- Python多线程笔记(二)
Lock对象 原语锁(互斥锁)是一个同步原语,状态是"已锁定"或者"未锁定"之一.两个方法acquire()和release()用于修改锁的状态.如果状态为已锁 ...
- 将List<E>内对象按照某个字段排序
主要用到java.util的Collections类 Collections.sort(list); 其中,E必须实现Comparable<E>接口
- ID生成算法(一)——雪花算法
JavaScript生成有序GUID或者UUID,这时就想到了雪花算法. 原理介绍: snowFlake算法最终生成ID的结果为一个64bit大小的整数,结构如下图: 解释: 1bit.二进制中最高位 ...
- P3975 (后缀自动机sort)
题目链接: https://www.luogu.org/problem/P3975 题意: 求出所有字串的第k大子串 有两种,第一种对于出现在不同位置的相同子串算作一个子串 第二种,对于不同位置的子串 ...
- 在使用avalon框架的时候,用ms-duplex双工绑定,在template上是有数据渲染的,但是js里面却是undefined
controller绑定是用于圈定某个VM的作用域范围,因为有的页面,你用的对象一致,而这两个作用域里面有相同的数据,那么很有可能是另一个作用域里面的东西影响了这个作用域,所以在指定作用域的给不同的命 ...
- Docker网络原则入门:EXPOSE,-p,-P,-link
如果你已经构建了一些多容器的应用程序,那么肯定需要定义一些网络规则来设置容器间的通信.有多种方式可以实现:可以通过--expose参数在运行时暴露端口,或者在Dockerfile里使用EXPOSE指令 ...
- Android中View大小的确定过程
View and ViewGroup 安卓中有5种基本的 ViewGroup: FrameLayout RelativeLayout LinearLayout TableLayout Absolute ...