C# Socket系列二 简单的创建 socket 通信
看了系列一 我们开启了对socket tcp的监听状态,那么这一章我们来讲解怎么创建socket的通信代码
我新建一个类 TSocketBase
public abstract class TSocketBase
{
//封装socket
internal Socket _Socket;
//回调
private AsyncCallback aCallback;
//接受数据的缓冲区
private byte[] Buffers;
//标识是否已经释放
private volatile bool IsDispose;
//10K的缓冲区空间
* ;
//收取消息状态码
private SocketError ReceiveError;
//发送消息的状态码
private SocketError SenderError;
//每一次接受到的字节数
;
//接受空消息次数
;
public abstract void Receive(byte[] rbuff);
public void SetSocket()
{
this.aCallback = new AsyncCallback(this.ReceiveCallback);
this.IsDispose = false;
this._Socket.ReceiveBufferSize = this.BufferSize;
this._Socket.SendBufferSize = this.BufferSize;
this.Buffers = new byte[this.BufferSize];
}
/// <summary>
/// 关闭并释放资源
/// </summary>
/// <param name="msg"></param>
public void Close(string msg)
{
if (!this.IsDispose)
{
this.IsDispose = true;
try
{
try { this._Socket.Close(); }
catch { }
IDisposable disposable = this._Socket;
if (disposable != null) { disposable.Dispose(); }
this.Buffers = null;
GC.SuppressFinalize(this);
}
catch (Exception) { }
}
}
/// <summary>
/// 递归接收消息方法
/// </summary>
internal void ReceiveAsync()
{
try
{
if (!this.IsDispose && this._Socket.Connected)
{
, this.BufferSize, SocketFlags.None, out SenderError, this.aCallback, this);
CheckSocketError(ReceiveError);
}
}
catch (System.Net.Sockets.SocketException) { this.Close("链接已经被关闭"); }
catch (System.ObjectDisposedException) { this.Close("链接已经被关闭"); }
}
/// <summary>
/// 接收消息回调函数
/// </summary>
/// <param name="iar"></param>
private void ReceiveCallback(IAsyncResult iar)
{
if (!this.IsDispose)
{
try
{
//接受消息
ReceiveSize = _Socket.EndReceive(iar, out ReceiveError);
//检查状态码
if (!CheckSocketError(ReceiveError) && SocketError.Success == ReceiveError)
{
//判断接受的字节数
)
{
byte[] rbuff = new byte[ReceiveSize];
Array.Copy(this.Buffers, rbuff, ReceiveSize);
this.Receive(rbuff);
//重置连续收到空字节数
ZeroCount = ;
//继续开始异步接受消息
ReceiveAsync();
}
else
{
ZeroCount++;
) { this.Close("错误链接"); }
}
}
}
catch (System.Net.Sockets.SocketException) { this.Close("链接已经被关闭"); }
catch (System.ObjectDisposedException) { this.Close("链接已经被关闭"); }
}
}
/// <summary>
/// 错误判断
/// </summary>
/// <param name="socketError"></param>
/// <returns></returns>
bool CheckSocketError(SocketError socketError)
{
switch ((socketError))
{
case SocketError.SocketError:
case SocketError.VersionNotSupported:
case SocketError.TryAgain:
case SocketError.ProtocolFamilyNotSupported:
case SocketError.ConnectionAborted:
case SocketError.ConnectionRefused:
case SocketError.ConnectionReset:
case SocketError.Disconnecting:
case SocketError.HostDown:
case SocketError.HostNotFound:
case SocketError.HostUnreachable:
case SocketError.NetworkDown:
case SocketError.NetworkReset:
case SocketError.NetworkUnreachable:
case SocketError.NoData:
case SocketError.OperationAborted:
case SocketError.Shutdown:
case SocketError.SystemNotReady:
case SocketError.TooManyOpenSockets:
this.Close(socketError.ToString());
return true;
}
return false;
}
/// <summary>
/// 发送消息方法
/// </summary>
internal int SendMsg(byte[] buffer)
{
;
try
{
if (!this.IsDispose)
{
size = , buffer.Length, SocketFlags.None, out SenderError);
CheckSocketError(SenderError);
}
}
catch (System.ObjectDisposedException) { this.Close("链接已经被关闭"); }
catch (System.Net.Sockets.SocketException) { this.Close("链接已经被关闭"); }
buffer = null;
return size;
}
}
上面我们事先了socket的异步接受消息,和同步发送消息已经关闭释放资源代码
接受消息net底层提供的接受消息的方法有很多,为什么我们要选择上面所写的呢?那是为了兼容U3D,silverlight, wpf, wp, wf,等程序可执行,不在重复做相同工作。
现在我们来创建一个实现类 TSocketClient
public class TSocketClient : TSocketBase
{
/// <summary>
/// 是否是服务器端的资源
/// </summary>
bool isServer = false;
/// <summary>
/// 客户端主动请求服务器
/// </summary>
/// <param name="ip"></param>
/// <param name="port"></param>
)
{
isServer = false;
this._Socket = new System.Net.Sockets.Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
this._Socket.Connect(ip, port);
this.SetSocket();
this.ReceiveAsync();
}
/// <summary>
/// 这个是服务器收到有效链接初始化
/// </summary>
/// <param name="socket"></param>
public TSocketClient(Socket socket)
{
isServer = true;
this._Socket = socket;
this.SetSocket();
this.ReceiveAsync();
}
/// <summary>
/// 收到消息后
/// </summary>
/// <param name="rbuff"></param>
public override void Receive(byte[] rbuff)
{
Console.WriteLine("Receive Msg:" + System.Text.UTF8Encoding.Default.GetString(rbuff));
if (isServer)
{
this.SendMsg(System.Text.UTF8Encoding.Default.GetBytes("Holle Client!"));
}
}
}
因为是测试示例,所以我把服务器和客户端实现类写成了,只是用来不同的构造函数来区分,是客户端还是服务器的标识
接下来我们测试一下代码
class Program
{
static void Main(string[] args)
{
TCPListener tcp = new TCPListener();
TSocketClient client = new TSocketClient();
client.SendMsg(System.Text.UTF8Encoding.Default.GetBytes("Holle Server!"));
Console.ReadLine();
}
}

运行结果看出,我们连接成功并且发送消息成功。
C# Socket系列二 简单的创建 socket 通信的更多相关文章
- C# Socket系列一 简单的创建socket的监听
socket的应用场景,在快速,稳定,保持长连接的数据传输代码.Http也是socket封装出来的,基于一次请求一次回复,然后断开的socket连接封装. 比如我们常见的游戏服务器,目前的很火的物联网 ...
- LINQ to Sql系列二 简单查询和联接查询
这一篇文章主要总结LINQ to sql的简单查询(单表查询)和联接查询(多表查询) 单表查询 需求是我们要输出TClass表中的结果.使用了from-in-select语句,代码如下: public ...
- 简单的同步Socket程序服务端
首先,Socket是.Net提供的 System.Net.Sockets命名空间的Scoket类为网络通信提供了一套丰富的方法和属性 服务器按照Socket的基本流程 先创建Socket 在用Bind ...
- python 实现一个简单tcp epoll socket
python 实现一个epoll server #!/usr/bin/env python #-*- coding:utf-8 -*- import socket import select impo ...
- import socket模块二
---恢复内容开始--- 优化两个小脚本实现不间断聊天: server.py: import socket sk = socket.socket() # 创建socket addess = ('127 ...
- Socket学习总结系列(二) -- CocoaAsyncSocket
这是系列的第二篇 这是这个系列文章的第二篇,要是没有看第一篇的还是建议看看第一篇,以为这个是接着第一篇梳理的 先大概的总结一下在上篇的文章中说的些内容: 1. 整理了一下做IM我们有那些途径,以及我们 ...
- C# Socket系列三 socket通信的封包和拆包
通过系列二 我们已经实现了socket的简单通信 接下来我们测试一下,在时间应用的场景下,我们会快速且大量的传输数据的情况! class Program { static void Main(stri ...
- socket编程——一个简单的样例
从一个简单的使用TCP样例開始socket编程,其基本过程例如以下: server client ++ ...
- socket计划——一个简单的例子
从一个简单易用TCP样品开始socket计划,的基本过程例如下列: server client +++ ...
随机推荐
- mono的远程调试
mono可以让.net程序运行在linux平台上.于是.net程序员有了mono之后就转身跨平台了.但开放环境往往还是在windows下,于是有了这样的需求,是否可以用windows下的源码来实机调试 ...
- Amazon AWS EC2开启Web服务器配置
在Amazon AWS EC2申请了一年的免费使用权,安装了CentOS + Mono + Jexus环境做一个Web Server使用. 在上述系统安装好之后,把TCP 80端口开启(iptable ...
- JQuery图片轮播滚动效果(网页效果--每日一更)
今天,带来的是一个图片的轮播滚动效果! 先来看一下效果展示:亲,请点击这里 原理很简单,设置一个定时器,使图片列表在每隔一段时间后滚动一次.而循环效果,就是在每一滚动的时候,将第一张图片放到最后一张的 ...
- 浅谈Excel开发:八 Excel 项目的安装部署
前面几篇文章讲解了Excel开发的几个比较主要的也是比较重要的方面,比如菜单系统,Excel对象模型,自定义函数,RTD函数,异步自定义函数,用户自定义任务面板等,在实际开发中我们还会遇到各种“千奇百 ...
- Preserving Remote IP/Host while proxying
因为这个文章用一般手段看不到,所以摘录下来备用 (From http://kasunh.wordpress.com/2011/10/11/preserving-remote-iphost-while- ...
- ASP.NET MVC随想录——创建自定义的Middleware中间件
经过前2篇文章的介绍,相信大家已经对OWIN和Katana有了基本的了解,那么这篇文章我将继续OWIN和Katana之旅——创建自定义的Middleware中间件. 何为Middleware中间件 M ...
- 细说.NET 中的多线程 (一 概念)
为什么使用多线程 使用户界面能够随时相应用户输入 当某个应用程序在进行大量运算时候,为了保证应用程序能够随时相应客户的输入,这个时候我们往往需要让大量运算和相应用户输入这两个行为在不同的线程中进行. ...
- 异步编程之Javascript Promises 规范介绍
什么是 Promises Promises是一种关于异步编程的规范,目的是将异步处理对象和处理规则进行规范化,为异步编程提供统一接口. 传统的回调函数 说到JavaScript的异步编程处理,通常我们 ...
- ssh/openssh
http://www.cnblogs.com/wwufengg/articles/ssh-openssh-detail.html http://www.cnblogs.com/jjkv3/archiv ...
- Atitit.eclipse 4.3 4.4 4.5 4.6新特性
Atitit intellij idea的使用总结attilax 1. ideaIC-2016.2.4.exe1 1.1. Ij vs eclipse市场份额1 1.2. Ij的优点(方便的支持gro ...