C# 利用socekt做到http监听,怎么样才能做到高性能
c#原始提供了http的监听的类HttpListener,实现了简单的http。文章地址《C# 控制台或者winform程序开启http的监听状态》
但是经过我测试,这个HttpListener提供的真的就只是简单的http监听功能,无法实现高并发处理。
不知道是我处理问题还是其他什么原因,无法实现,当上一个http请求连接尚未关闭的情况下,即便是把请求放到另外一个线程执行,都要等到处理结束,close了才能接受和处理下一次的连接请求。
也许你会说HttpListener不是提供了异步监听的嘛?异步不就可以类使用多线程实现嘛。但是经过我测试,确实没有得到我想要的实际效果。
所以另辟蹊径。http其实质就是socket的tcp封装实现的功能,单次请求,处理,关闭的socket功能。
所以这里找到了可以使用最原始的socket的来提供http监听,处理数据,关闭状态。
好了直接上代码,,一下代码部分来至于博客园,园友帖子提供,时间久远亦不知道是哪位仁兄的帖子,见谅。
internal class HttpServer
{
private IPEndPoint _IP;
private TcpListener _Listeners;
private volatile bool IsInit = false;
HashSet<string> Names;
/// <summary>
/// 初始化服务器
/// </summary>
public HttpServer(string ip, int port, HashSet<string> names)
{
IPEndPoint localEP = new IPEndPoint(IPAddress.Parse(ip), port);
this._IP = localEP;
Names = names;
if (Names == null)
{
Names = new HashSet<string>();
}
try
{
foreach (var item in names)
{
Console.WriteLine(string.Format(System.DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss:fff:") + "Start Listen Http Socket -> {0}:{1}{2} ", ip, port, item));
}
this._Listeners = new TcpListener(IPAddress.Parse(ip), port);
);
IsInit = true;
this.AcceptAsync();
}
catch (Exception ex)
{
Console.WriteLine(ex);
this.Dispose();
}
}
private void AcceptAsync()
{
try
{
this._Listeners.BeginAcceptTcpClient(new AsyncCallback(AcceptAsync_Async), null);
}
catch (Exception) { }
}
private void AcceptAsync_Async(IAsyncResult iar)
{
this.AcceptAsync();
try
{
TcpClient client = this._Listeners.EndAcceptTcpClient(iar);
var socket = new HttpClient(client);
Console.WriteLine(System.DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss:fff:") + "Create Http Socket Remote Socket LocalEndPoint:" + client.Client.LocalEndPoint + " RemoteEndPoint:" + client.Client.RemoteEndPoint.ToString());
foreach (var item in Names)
{
if (socket.http_url.StartsWith(item))
{
try
{
socket.process();
return;
}
catch { break; }
}
}
socket.WriteFailure();
socket.Close();
}
catch (Exception) { }
}
/// <summary>
/// 释放资源
/// </summary>
public void Dispose()
{
if (IsInit)
{
IsInit = false;
this.Dispose(true);
GC.SuppressFinalize(this);
}
}
/// <summary>
/// 释放所占用的资源
/// </summary>
/// <param name="flag1"></param>
protected virtual void Dispose([MarshalAs(UnmanagedType.U1)] bool flag1)
{
if (flag1)
{
if (_Listeners != null)
{
try
{
Console.WriteLine(string.Format("Stop Http Listener -> {0}:{1} ", this.IP.Address.ToString(), this.IP.Port));
_Listeners.Stop();
_Listeners = null;
}
catch { }
}
}
}
/// <summary>
/// 获取绑定终结点
/// </summary>
public IPEndPoint IP { get { return this._IP; } }
}
这个是实现socket监听状态
public class HttpClient
{
* * ; // 10MB
;
private Stream inputStream;
public StreamWriter OutputStream;
public String http_method;
public String http_url;
public String http_protocol_versionstring;
public Hashtable httpHeaders = new Hashtable();
internal TcpClient _Socket;
/// <summary>
/// 这个是服务器收到有效链接初始化
/// </summary>
internal HttpClient(TcpClient client)
{
this._Socket = client;
inputStream = new BufferedStream(_Socket.GetStream());
OutputStream = new StreamWriter(new BufferedStream(_Socket.GetStream()), UTF8Encoding.Default);
ParseRequest();
}
internal void process()
{
try
{
if (http_method.Equals("GET"))
{
Program.Pool.ActiveHttp(this, GetRequestExec());
}
else if (http_method.Equals("POST"))
{
Program.Pool.ActiveHttp(this, PostRequestExec());
}
}
catch (Exception e)
{
Console.WriteLine("Exception: " + e.ToString());
WriteFailure();
}
}
public void Close()
{
OutputStream.Flush();
inputStream.Dispose();
inputStream = null;
OutputStream.Dispose();
OutputStream = null; // bs = null;
this._Socket.Close();
}
#region 读取流的一行 private string ReadLine()
/// <summary>
/// 读取流的一行
/// </summary>
/// <returns></returns>
private string ReadLine()
{
int next_char;
string data = "";
while (true)
{
next_char = this.inputStream.ReadByte();
if (next_char == '\n') { break; }
if (next_char == '\r') { continue; }
) { Thread.Sleep(); continue; };
data += Convert.ToChar(next_char);
}
return data;
}
#endregion
#region 转化出 Request private void ParseRequest()
/// <summary>
/// 转化出 Request
/// </summary>
private void ParseRequest()
{
String request = ReadLine();
if (request != null)
{
string[] tokens = request.Split(' ');
)
{
throw new Exception("invalid http request line");
}
http_method = tokens[].ToUpper();
http_url = tokens[];
http_protocol_versionstring = tokens[];
}
String line;
while ((line = ReadLine()) != null)
{
if (line.Equals(""))
{
break;
}
int separator = line.IndexOf(':');
)
{
throw new Exception("invalid http header line: " + line);
}
String name = line.Substring(, separator);
;
while ((pos < line.Length) && (line[pos] == ' '))
{
pos++;//过滤键值对的空格
}
string value = line.Substring(pos, line.Length - pos);
httpHeaders[name] = value;
}
}
#endregion
#region 读取Get数据 private Dictionary<string, string> GetRequestExec()
/// <summary>
/// 读取Get数据
/// </summary>
/// <returns></returns>
private Dictionary<string, string> GetRequestExec()
{
Dictionary<string, string> datas = new Dictionary<string, string>();
);
)
{
);
datas = getData(data);
}
WriteSuccess();
return datas;
}
#endregion
#region 读取提交的数据 private void handlePOSTRequest()
/// <summary>
/// 读取提交的数据
/// </summary>
private Dictionary<string, string> PostRequestExec()
{
;
MemoryStream ms = new MemoryStream();
if (this.httpHeaders.ContainsKey("Content-Length"))
{
//内容的长度
content_len = Convert.ToInt32(this.httpHeaders["Content-Length"]);
if (content_len > MAX_POST_SIZE) { throw new Exception(String.Format("POST Content-Length({0}) 对于这个简单的服务器太大", content_len)); }
byte[] buf = new byte[BUF_SIZE];
int to_read = content_len;
)
{
, Math.Min(BUF_SIZE, to_read));
)
{
) { break; }
else { throw new Exception("client disconnected during post"); }
}
to_read -= numread;
ms.Write(buf, , numread);
}
ms.Seek(, SeekOrigin.Begin);
}
WriteSuccess();
StreamReader inputData = new StreamReader(ms);
string data = inputData.ReadToEnd();
return getData(data);
}
#endregion
#region 输出状态
/// <summary>
/// 输出200状态
/// </summary>
public void WriteSuccess()
{
OutputStream.WriteLine("HTTP/1.0 200 OK");
OutputStream.WriteLine("Content-Type: text/html");
OutputStream.WriteLine("Connection: close");
OutputStream.WriteLine("");
}
/// <summary>
/// 输出状态404
/// </summary>
public void WriteFailure()
{
OutputStream.WriteLine("HTTP/1.0 404 File not found");
OutputStream.WriteLine("Content-Type: text/html");
OutputStream.WriteLine("Connection: close");
OutputStream.WriteLine("");
}
#endregion
/// <summary>
/// 分析http提交数据分割
/// </summary>
/// <param name="rawData"></param>
/// <returns></returns>
private static Dictionary<string, string> getData(string rawData)
{
var rets = new Dictionary<string, string>();
string[] rawParams = rawData.Split('&');
foreach (string param in rawParams)
{
string[] kvPair = param.Split('=');
];
]);
rets[key] = value;
}
return rets;
}
}
实现了对http数据请求处理
public interface ISocketPool
{
/// <summary>
///
/// </summary>
/// <param name="client"></param>
void ActiveHttp(Fly.Network.SocketPool.Http.HttpClient client, Dictionary<string, string> parms);
}
public class Program
{
public static MessagePool Pool = new MessagePool();
static void Main(string[] args)
{
HttpServer https = , new HashSet<string>() {"/test/","/flie/" });
Console.ReadLine();
}
}
class MessagePool : ISocketPool
{
public void ActiveHttp(HttpClient client, Dictionary<string, string> parms)
{
Thread.Sleep(, ));
foreach (var item in parms)
{
Console.WriteLine(DateTime.Now.NowString() + "item.Key:" + item.Key + "; item.Value:" + item.Value);
}
string strHtml = @"
<html><head></head>
<body>
<div> </div>
<div> </div>
<div> </div>
<div> </div>
<div> </div>
{0}
</body>
</html>
";
client.OutputStream.WriteLine(string.Format(strHtml, DateTime.Now.NowString() + "xxxxxxxxxxx"));
client.Close();
}
}
程序启动过后,看到输出
-- ::::Start Listen Http Socket -> /test/ -- ::::Start Listen Http Socket -> /flie/
接下来我们在浏览器输入 127.0.0.1/test/

正常收到请求,输出程序
127.0.0.1/test/

这里test1这个并不是我们监听饿目录,根本不会处理,
接下来我们再看看这个效果 get提交的参数 127.0.0.1/test/?bb=test

输出了get提交过来的参数信息。可能你会奇怪,为什么一次请求会收到两次连接请求。这里我查看过了其中一次请求是浏览器自带的请求页面标签的icon连接请求,
如果你拷贝了程序,你现在可以实现跑来程序,然后输入网址,按着F5不放,看看服务器的承受能力,当然这里忽律了逻辑方面对cpu内存消耗和时间消耗问题。
测试看看吧。
C# 利用socekt做到http监听,怎么样才能做到高性能的更多相关文章
- 利用Node的chokidar 监听文件改变的文件。
最近维护一个项目.每次改完东西,都要上传到服务器.然后有时候就忘记一些东西,于是就想有没有可以方法能监听文件的改变.然后我再利用程序把更改的文件一键上传到服务器. 于是就找到了nodejs 的chok ...
- 利用spring的ApplicationListener监听某一类事件的发生
1.ApplicationListener在使用过程中可以监听某一事件的发生,可以做出相应的处理,这个方式不常用,但是在特殊情况下面还是有用的. 2.导包pom.xml <project xml ...
- android菜鸟学习笔记23----ContentProvider(三)利用内置ContentProvider监听短信及查看联系人
要使用一个ContentProvider,必须要知道的是它所能匹配的Uri及其数据存储的表的结构. 首先想办法找到访问短信及联系人数据的ContentProvider能接受的Uri: 到github上 ...
- 利用input事件来监听移动端的输入
今天遇到一个新需求,经理要求评论功能需要限制字数,就像微博那样限制最多输入150字,这里就需要实时提醒用户还能输入多少字了. 在最开始的时候,想到的是监听keyup事件,然后计算用户输入的字数,但是有 ...
- 利用bootstrap制作滚动监听
滚动监听(Scrollspy)插件,即自动更新导航插件,会根据滚动条的位置自动更新对应的导航目标. 在这里,需要引入三个文件:bootstrap.min.css jquery-3.3.1.j ...
- 20180530利用Maxwell组件实时监听Mysql的binlog日志
转自:https://blog.csdn.net/qq_30921461/article/details/78320750 http://kafka.apache.org/quickstart htt ...
- 利用input event 实时监听input输入的内容
<div id="addNumber"> <p>How many people would you like to invite?</p> &l ...
- 利用原生JS实时监听input框输入值
传送门https://www.cnblogs.com/lantinggumo/p/7636715.html 传送门https://www.cnblogs.com/nailc/p/8572226.htm ...
- Android监听手机网络变化
Android监听手机网络变化 手机网络状态发生变化会发送广播,利用广播接收者,监听手机网络变化 效果图 注册广播接收者 <?xml version="1.0" encodi ...
随机推荐
- Python基本数据类型——str
字符串常用操作 移除空白 分割 长度 索引 切片 class str(basestring): """ str(object='') -> string Retur ...
- LOMA280保险原理读书笔记
LOMA是国际金融保险管理学院(Life Office Management Association)的英文简称.国际金融保险管理学院是一个保险和金融服务机构的国际组织,它的创建目的是为了促进信息交流 ...
- 在C#代码中应用Log4Net(四)在Winform和Web中捕获全局异常
毕竟人不是神,谁写的程序都会有bug,有了bug不可怕,可怕的是出错了,你却不知道错误在哪里.所以我们需要将应用程序中抛出的所有异常都记录起来,不然出了错,找问题就能要了你的命.下面我们主要讨论的是如 ...
- .NET 的 Debug 和 Release build 对执行速度的影响
这篇文章发布于我的 github 博客:原文 在真正开始讨论之前先定义一下 Scope. 本文讨论的范围限于执行速度,内存占用什么的不在评估的范围之内. 本文不讨论算法:编译器带来的优化基本上属于底层 ...
- 【VC++技术杂谈002】打印技术之获取及设置系统默认打印机
本文主要介绍如何获取以及设置系统的默认打印机. 1.获取系统中的所有打印机 获取系统中的所有打印机可以使用EnumPrinters()函数,该函数可以枚举全部的本地.网络打印机信息.其函数原型为: B ...
- Java 浅析三大特性之一多态
Java 浅析三大特性之一多态 之前我们的文章讲了Java的封装和继承,封装讲的时候,并没有体现出来封装的强大之处,反而还要慎用封装.因为这时的封装还没有和多态联系到一起,还无法看出向上转型的厉害之处 ...
- JS面向对象(3) -- Object类,静态属性,闭包,私有属性, call和apply的使用,继承的三种实现方法
相关链接: JS面向对象(1) -- 简介,入门,系统常用类,自定义类,constructor,typeof,instanceof,对象在内存中的表现形式 JS面向对象(2) -- this的使用,对 ...
- bootstrap与Select2使用小结
这个select2组件的功能确实很强大,可以将图片放入到select里面随着文字一起显示. 组件的下载地址以及API说明地址: 1.Select2使用示例地址:https://select2.gith ...
- sizzle分析记录: 自定义伪类选择器
可见性 :hidden :visible 隐藏对象没有宽高,前提是用display:none处理的 jQuery.expr.filters.hidden = function( elem ) { // ...
- NodeJS POST Request Over JSON-RPC
1.npm install art-template2.npm install request3.在app.js中加入以下代码转html: var template = require('art-t ...