SupperSocket深入浅出
这篇文章出要是SuperSocket底层如何接收数据
Process(ArraySegment<byte> segment) 获取加载数据(直到数据全部接收后返回)
namespace SuperSocket.ProtoBase
{
/// <summary>
/// The default pipeline processor
/// </summary>
/// <typeparam name="TPackageInfo">The type of the package info.</typeparam>
public class DefaultPipelineProcessor<TPackageInfo> : IPipelineProcessor
where TPackageInfo : IPackageInfo
{
private IReceiveFilter<TPackageInfo> m_ReceiveFilter; private IReceiveFilter<TPackageInfo> m_FirstReceiveFilter; private BufferList m_ReceiveCache; private int m_MaxPackageLength; /// <summary>
/// Initializes a new instance of the <see cref="DefaultPipelineProcessor{TPackageInfo}"/> class.
/// </summary>
/// <param name="receiveFilter">The initializing receive filter.</param>
/// <param name="maxPackageLength">The max package size.</param>
public DefaultPipelineProcessor(IReceiveFilter<TPackageInfo> receiveFilter, int maxPackageLength = )
{
m_FirstReceiveFilter = m_ReceiveFilter = receiveFilter;
m_ReceiveCache = new BufferList();
m_MaxPackageLength = maxPackageLength;
} private void PushResetData(ArraySegment<byte> raw, int rest)
{
var segment = new ArraySegment<byte>(raw.Array, raw.Offset + raw.Count - rest, rest);
m_ReceiveCache.Add(segment);
} private IList<IPackageInfo> GetNotNullOne(IList<IPackageInfo> left, IList<IPackageInfo> right)
{
if (left != null)
return left; return right;
} /// <summary>
/// Processes the input segment.
/// </summary>
/// <param name="segment">The input segment.</param>
/// <returns>
/// the processing result
/// </returns>
public virtual ProcessResult Process(ArraySegment<byte> segment)
{
var receiveCache = m_ReceiveCache; receiveCache.Add(segment); var rest = ; var currentReceiveFilter = m_ReceiveFilter; SingleItemList<IPackageInfo> singlePackage = null; List<IPackageInfo> packageList = null; while (true)
{
var lastItemLength = receiveCache.Last.Count;
var packageInfo = currentReceiveFilter.Filter(receiveCache, out rest); if (currentReceiveFilter.State == FilterState.Error)
{
return ProcessResult.Create(ProcessState.Error);
}
//最大缓存空间
if (m_MaxPackageLength > )
{
var length = receiveCache.Total; if (length > m_MaxPackageLength)
{
return ProcessResult.Create(ProcessState.Error, string.Format("Max package length: {0}, current processed length: {1}", m_MaxPackageLength, length));
}
} var nextReceiveFilter = currentReceiveFilter.NextReceiveFilter; // don't reset the filter if no request is resolved
if(packageInfo != null)
currentReceiveFilter.Reset(); if (nextReceiveFilter != null)
{
currentReceiveFilter = nextReceiveFilter;
m_ReceiveFilter = currentReceiveFilter;
} // continue receive
if (packageInfo == null)
{
if (rest > )
{
var last = receiveCache.Last; if(rest != lastItemLength)
{
PushResetData(segment, rest);
} continue;
} return ProcessResult.Create(ProcessState.Cached, GetNotNullOne(packageList, singlePackage));
} if (packageList != null)
{
packageList.Add(packageInfo);
}
else if (singlePackage == null)
singlePackage = new SingleItemList<IPackageInfo>(packageInfo);
else
{
if (packageList == null)
packageList = new List<IPackageInfo>(); packageList.Add(singlePackage[]);
packageList.Add(packageInfo);
singlePackage = null;
} if (packageInfo is IBufferedPackageInfo // is a buffered package
&& (packageInfo as IBufferedPackageInfo).Data is BufferList) // and it uses receive buffer directly
{
// so we need to create a new receive buffer container to use
m_ReceiveCache = receiveCache = new BufferList(); if (rest <= )
{
return ProcessResult.Create(ProcessState.Cached, GetNotNullOne(packageList, singlePackage));
}
}
else
{
m_ReceiveCache.Clear(); if (rest <= )
{
return ProcessResult.Create(ProcessState.Completed, GetNotNullOne(packageList, singlePackage));
}
} PushResetData(segment, rest);
}
} /// <summary>
/// cleanup the cached the buffer by resolving them into one package at the end of the piple line
/// </summary>
/// <returns>return the processing result</returns>
public void Reset()
{
m_ReceiveCache.Clear();
m_FirstReceiveFilter.Reset(); if (m_ReceiveFilter != m_FirstReceiveFilter)
m_ReceiveFilter = m_FirstReceiveFilter;
} /// <summary>
/// Gets the received cache.
/// </summary>
/// <value>
/// The cache.
/// </value>
public BufferList Cache
{
get { return m_ReceiveCache; }
}
}
}
SupperSocket深入浅出的更多相关文章
- SupperSocket深入浅出(二)
如果还没有看SuperStock深入浅出(一) ,请先看 这一章,主要说下命令是如果运行的.刚开始的时候会发现拷别人的代码命令是可以运行的,在修改的过程中突然发现命令无效了? 这里什么原因?,我先把代 ...
- SupperSocket深入浅出(一)
花了几天时间了解了SupperSocket工作原理,各各类之间的工作关系.SupperSocket大部资料网上都有,但写的都不适合初学者. 今天花点时间写下这几天的学习成果,一方面是为了将来更好的回顾 ...
- 【深入浅出jQuery】源码浅析--整体架构
最近一直在研读 jQuery 源码,初看源码一头雾水毫无头绪,真正静下心来细看写的真是精妙,让你感叹代码之美. 其结构明晰,高内聚.低耦合,兼具优秀的性能与便利的扩展性,在浏览器的兼容性(功能缺陷.渐 ...
- 【深入浅出jQuery】源码浅析2--奇技淫巧
最近一直在研读 jQuery 源码,初看源码一头雾水毫无头绪,真正静下心来细看写的真是精妙,让你感叹代码之美. 其结构明晰,高内聚.低耦合,兼具优秀的性能与便利的扩展性,在浏览器的兼容性(功能缺陷.渐 ...
- 深入浅出Struts2+Spring+Hibernate框架
一.深入浅出Struts2 什么是Struts2? struts2是一种基于MVC的轻量级的WEB应用框架.有了这个框架我们就可以在这个框架的基础上做起,这样就大大的提高了我们的开发效率和质量,为公司 ...
- DOM 事件深入浅出(二)
在DOM事件深入浅出(一)中,我主要给大家讲解了不同DOM级别下的事件处理程序,同时介绍了事件冒泡和捕获的触发原理和方法.本文将继续介绍DOM事件中的知识点,主要侧重于DOM事件中Event对象的属性 ...
- DOM 事件深入浅出(一)
在项目开发时,我们时常需要考虑用户在使用产品时产生的各种各样的交互事件,比如鼠标点击事件.敲击键盘事件等.这样的事件行为都是前端DOM事件的组成部分,不同的DOM事件会有不同的触发条件和触发效果.本文 ...
- 深入浅出node(2) 模块机制
这部分主要总结深入浅出Node.js的第二章 一)CommonJs 1.1CommonJs模块定义 二)Node的模块实现 2.1模块分类 2.2 路径分析和文件定位 2.2.1 路径分析 2.2.2 ...
- IOS 网络-深入浅出(一 )-> 三方SDWebImage
首要我们以最为常用的UIImageView为例介绍实现原理: 1)UIImageView+WebCache: setImageWithURL:placeholderImage:options: 先显 ...
随机推荐
- olivehc--百度开源的cdn cache
github 地址:http://git.baidu.com/olivehc/olivehc 主要是为了方便管理,百度cdn承载了全百度40%的流量,但是cdn团队只有几个人(一次培训中提到只有4个) ...
- Swap是个什么东东?
要明白这个首先要知道什么是保护模式和实模式.以前的操作系统是实模式,例如dos.每个时候只有一个进程在跑,这个进程使用全部的物理内存.后来发展到保护模式,分时多进程.一个CPU上跑多个进程, 但进程不 ...
- Java对文件的读取方式以及它们的优缺点
Java常用的对文件的读取方式基本包括: BufferedReader -> readLine(): 按行读取文件,直到读取内容==null FileInputStream -> read ...
- Ping服务
什么是Ping服务 ping是基于XML_RPC标准协议的更新通告服务,用于博客把内容更新快速通知给百度,以便百度及时进行抓取和更新. Ping服务使用方法 你可以采取手动通知和自动通知两种方式使用p ...
- Window窗口布局 --- DecorView浅析
开发中,通常都是在onCreate()中调用setContentView(R.layout.custom_layout)来实现想要的页面布局,我们知道,页面都是依附在窗口之上的,而DecorView即 ...
- 【php】php与mysql初体验
第一次体验在web站点上使用MySQL数据库,遇到了很多问题,总结如下: 1.安装XAMPP软件后,将文件放到hotdocs文件夹下,要访问其中的文件,使用localhost/XXX/XXX ,路径要 ...
- openssl生成iis需要的pfx格式的证书
合成.pfx证书 将私钥文件(server.key)和服务器crt证书文件(server.crt ),放到openssl安装目录的bin目录下. 控制台也进到此目录下,然后执行下面指令. openss ...
- Redis(二)安全设置
redis是一款流行的内存数据库,默认是无密码登录,若出于安全考虑,需要为其加上控制权限,则可以通过以下2种方法来设置. 1. 采用绑定IP的方式来进行控制(比较安全的方法) 在redis.conf( ...
- 杂记(那些我还容易混淆的c和c++知识)
1: 定义一个对象时先调用基类的构造函数.然后调用派生类的构造函数:析构的时候恰好相反:先调用派生类的析构函数.然后调用基类的析构函数.2: 多态性具体体现在运行和编译两个方面:在程序运行时的多态性 ...
- 解决apache启动错误:Could not reliably determine the server's fully qualified domain name
启动apache遇到错误:httpd: Could not reliably determine the server's fully qualified domain name [root@serv ...