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对象的更多相关文章

  1. .Net开发笔记(十九) 创建一个可以可视化设计的对象

    阅读本篇博客之前需要了解VS窗体设计器的工作原理,详细可参见本系列博客(十).(十一).(十二).必须需要知道的一条结论就是:处于窗体设计器(Form Designer)中的任何组件(包含控件,下同) ...

  2. threadlocal精髓是为每一个线程保证一个共享对象,保证一个,保证是同一个

    threadlocal精髓是为每一个线程保证一个共享对象,保证一个,保证同一个线程中是同一个共享对象. 如果是静态变量是共享的话,那必须同步,否则尽管有副本,还是会出错,故C错

  3. Tiny Mapper是一个.net平台开源的对象映射组件

    NET平台开源项目速览(14)最快的对象映射组件Tiny Mapper   阅读目录 1.Tiny Mapper基本介绍 2.Tiny Mapper 基本使用 3.Tiny Mapper 指定配置使用 ...

  4. 27、通过visual s'tudio 验证 SOCKET编程:搭建一个TCP服务器

    本文就是在windows下进行socket编程,搭建一个TCP客户端. 在visual studio下编程,首先在windows下进行初始化(这点在linux下是不需要的): /* 初始化 Winso ...

  5. 通过一个uri获取一个Bitmap对象

    Android 开发过程中,可能会用到的,通过一个uri获取一个Bitmap对象 private Bitmap getBitmapFromUri(Uri uri){  try  {   // 读取ur ...

  6. C#中如果用await关键字来await一个为null的Task对象会抛出异常

    await & async模式是C#中一个很重要的特性,可以用来提高异步程序(多线程程序)的执行效率.但是如果尝试用await关键字来await一个为null的Task对象,会导致程序抛出Nu ...

  7. Object.assign() 从一个或多个源对象复制到目标对象

    Object.assign()方法用于将所有可枚举属性的值从一个或多个源对象复制到目标对象.它将返回目标对象. 1.语法: Object.assign(target, ... , sources) 参 ...

  8. 创建一个doc对象时候 如果读取了一个已存在的xml对象时候 该xml对象的结构已存在doc中 当改变该doc结构时候 不需要创建新的doc对象

    创建一个doc对象时候 如果读取了一个已存在的xml对象时候 该xml对象的结构已存在doc中 当改变该doc结构时候不 需要创建新的doc对象 直接添加即可 他会同步过去

  9. iter创建一个可以被迭代的对象

    #!/usr/bin/env python obj = iter([11,22,33,44]) #iter 创建一个可以被迭代的对象 print(obj) r1 = next(obj) print(r ...

随机推荐

  1. socket、端口、进程的关系

    本文属网络编程部分.socket的引入是为了解决不同计算机间进程间通信的问题. 端口是TCP/IP协议中的概念,描述的是TCP协议上的对应的应用,可以理解为基于TCP的系统服务,或者说系统进程!如下图 ...

  2. STM32 HAL库 UART 串口读写功能笔记

    https://www.cnblogs.com/Mysterious/p/4804188.html STM32L0 HAL库 UART 串口读写功能 串口发送功能: uint8_t TxData[10 ...

  3. qsort中的函数指针,及函数解释

    函数指针有何用 函数指针的应用场景比较多,以库函数qsort排序函数为例,它的原型如下: void qsort(void *base,size_t nmemb,size_t size , int(*c ...

  4. PostgreSQL 多版本的实现与Innodb和oracle的差别

    PostgreSQL与oracle或InnoDB的多版本实现最大的区别在于最新版本和历史版本是否分离存储,PostgreSQL不分,而oracle和InnoDB分,而innodb也只是分离了数据,索引 ...

  5. 利用 BackgroundService 固定时间间隔执行某动作

    继承 BackgroundService 类: 为什么会写这个东西呢?本人在写消息队列的时候思考过一个问题——比如,每5秒从队列里面取一条消息(一条消息里面又包含了1000条数据),要把这1000条数 ...

  6. express搭建web服务器、路由、get、post请求、multer上传文件、EJS模板引擎的使用

    express官网 postman工具下载地址  multer的npm文档地址 express模板引擎怎么使用  地址:http://www.expressjs.com.cn/guide/using- ...

  7. myeclipse导入项目

    因为现在项目大部分都是maven项目所以导入的时候因该选择

  8. Mysql之数据库设计规范

    1. 三大范式首先要明白”范式(NF)”是什么意思.按照教材中的定义,范式是“符合某一种级别的关系模式的集合,表示一个关系内部各属性之间的联系的合理化程度”.数据库范式也分为1NF,2NF,3NF,B ...

  9. Spring —— @Async注解的使用

    参考文档 Spring Boot使用@Async实现异步调用:自定义线程池 Spring Boot使用@Async实现异步调用:ThreadPoolTaskScheduler线程池的优雅关闭

  10. 标准6轴机器人正反解(1)-坐标系和MDH参数表

    刚来新公司不久,部门给安排了新人作业,我被分到的任务是求标准6轴机器人的正反解,以及利用就近原则选择最优解.从今天开始,逐步将这部分内容总结出来: 本文以及后续文章均使用改进DH法: 连杆坐标系: 坐 ...