`
hugh-lin
  • 浏览: 70371 次
  • 性别: Icon_minigender_1
  • 来自: 厦门
社区版块
存档分类
最新评论

[转]使用C#进行点对点通讯和文件传输(通讯基类部分)

阅读更多

最近一个项目要用到点对点文件传输,俺就到处找资料写程序,最后终于完成了,为了让别人少走些弯路,俺决定将俺程序中最重要的部分贡献出来,希望对大家有所帮助。



俺的程序分三部分,包括发送部分、接受部分和一个两者共享的通讯基类,这个基类才是俺心血的结晶:)



一、通讯基类

using System;

using System.Net.Sockets;

using System.Net ;

using System.IO ;

using System.Windows.Forms;

using System.Text;



namespace BaseClass

{

/// <summary>

///
传送信息的格式为 给定长度的命令部分+给定长度的命令注释部分+可变长度的长度信息+可变长度的信息部分

/// </summary>

public class CommunClass

{

public CommunClass()

{

//

// TODO:
在此处添加构造函数逻辑

//

}

/// <summary>

///
命令部分的长度

/// </summary>

private static readonly int CMDLEN = 50 ;

/// <summary>

///
命令注释部分的长度

/// </summary>

private static readonly int DESCLEN = 100 ;

/// <summary>

///
可变长度的长度信息部分所占的字节数

/// </summary>

private static readonly int DYNAMICLENGTHLEN = 10 ;

/// <summary>

///
每次处理可变信息部分的长度

/// </summary>

private static readonly int DEALLEN = 1024 ;

/// <summary>

/// /
应答的最大长度

/// </summary>

private static readonly int RESPONLEN = 20 ;

/// <summary>

///
用于填充命令或注释不足长度部分的字符

/// </summary>

private static readonly char FILLCHAR = '^' ;



/// <summary>

///
成功发送一部分数据后的回调方法(也可以认为是触发的事件,但严格来说还不是)

/// </summary>

public delegate void OnSend(int iTotal,int iSending) ;



/// <summary>

///
根据给定的服务器和端口号建立连接

/// </summary>

/// <param name="strHost">
服务器名</param>

/// <param name="iPort">
端口号</param>

/// <returns></returns>

public static Socket ConnectToServer(string strHost,int iPort)

{

try

{

IPAddress ipAddress = Dns.Resolve(strHost).AddressList[0];

IPEndPoint ipPoint = new IPEndPoint(ipAddress,iPort) ;



Socket s = new Socket(AddressFamily.InterNetwork,SocketType.Stream,ProtocolType.Tcp) ;

s.Connect(ipPoint) ;

return s ;

}

catch (Exception e)

{

throw (new Exception("
建立到服务器的连接出错" + e.Message)) ;

}

}

/// <summary>

///
将文本写到Socket

/// </summary>

/// <param name="s">
要发送信息的Socket</param>

/// <param name="strInfo">
要发送的信息</param>

/// <returns>
是否成功</returns>

public static bool WriteTextToSocket(Socket s,string strInfo)

{

byte [] buf = Encoding.UTF8.GetBytes(strInfo) ;

try

{

s.Send(buf,0,buf.Length,SocketFlags.None) ;

return true ;

}

catch(Exception err)

{

MessageBox.Show("
发送文本失败!"+err.Message) ;

return false ;

}

}

/// <summary>

///
将命令文本写到Socket

/// </summary>

/// <param name="s">
要发送命令文本的Socket</param>

/// <param name="strInfo">
要发送的命令文本</param><!----><o:p></o:p>

/// <returns>是否成功</returns>

public static bool WriteCommandToSocket(Socket s,string strCmd)

{

if (strCmd == "")

strCmd = "NOP" ;

strCmd = strCmd.PadRight(CMDLEN,FILLCHAR) ;

return WriteTextToSocket(s,strCmd) ;

}

/// <summary>

///
将命令注释写到Socket

/// </summary>

/// <param name="s">
要发送命令注释的Socket</param>

/// <param name="strInfo">
要发送的命令注释</param>

/// <returns>
是否成功</returns>

public static bool WriteCommandDescToSocket(Socket s,string strDesc)

{

if (strDesc == "")

strDesc = "0" ;

strDesc = strDesc.PadRight(DESCLEN,FILLCHAR) ;

return WriteTextToSocket(s,strDesc) ;

}

/// <summary>

///
发送可变信息的字节数

/// </summary>

/// <param name="s">
要发送字节数的Socket</param>

/// <param name="iLen">
字节数</param>

/// <returns>
是否成功</returns>

public static bool WriteDynamicLenToSocket(Socket s,int iLen)

{

string strLen = iLen.ToString().PadRight(DYNAMICLENGTHLEN,FILLCHAR) ;

return WriteTextToSocket(s,strLen) ;

}

/// <summary>

///
将缓存的指定部分发送到Socket

/// </summary>

/// <param name="s">
要发送缓存的Socket</param>

/// <param name="buf">
要发送的缓存</param>

/// <param name="iStart">
要发送缓存的起始位置</param>

/// <param name="iCount">
要发送缓存的字节数</param>

/// <param name="iBlock">
每次发送的字节说</param>

/// <param name="SendSuccess">
每次发送成功后的回调函数</param>

/// <returns>
是否发送成功</returns>

public static bool WriteBufToSocket(Socket s,byte [] buf,int iStart,int iCount,int iBlock,OnSend SendSuccess)

{

int iSended = 0 ;

int iSending = 0 ;

while(iSended<iCount)

{

if (iSended + iBlock <= iCount)

iSending = iBlock ;

else

iSending = iCount - iSended ;

s.Send(buf,iStart+iSended,iSending,SocketFlags.None) ;

iSended += iSending ;

if (ReadResponsionFromSocket(s)=="OK")

if (SendSuccess != null)

SendSuccess(iCount,iSended) ;

else

return false;

}

return true ;

}

/// <summary>

///
将长度不固定文本发送到socket

/// </summary>

/// <param name="s">
要发送文本的Socket</param>

/// <param name="strText">
要发送的文本</param>

/// <param name="OnSendText">
成功发送一部分文本后的回调函数</param>

/// <param name="settextlen">
得到文本长度的回调函数</param>

/// <returns></returns>

public static bool WriteDynamicTextToSocket(Socket s,string strText,

OnSend OnSendText)

{

byte [] buf = Encoding.UTF8.GetBytes(strText) ;



int iLen = buf.Length ;

try

{

WriteDynamicLenToSocket(s,iLen) ;

return WriteBufToSocket(s,buf,0,iLen, DEALLEN,OnSendText) ;

}

catch(Exception err)

{

MessageBox.Show("
发送文本失败!"+err.Message) ;

return false ;

}

}

/// <summary>

///
将文件写到Socket

/// </summary>

/// <param name="s">
要发送文件的Socket</param>

/// <param name="strFile">
要发送的文件</param>

/// <returns>
是否成功</returns>

public static bool WriteFileToSocket(Socket s,string strFile,

OnSend OnSendFile)

{

FileStream fs = new FileStream(strFile,FileMode.Open,FileAccess.Read,FileShare.Read) ;

int iLen = (int)fs.Length ;

WriteDynamicLenToSocket(s,iLen) ;

byte [] buf = new byte[iLen] ;

try

{

fs.Read(buf,0,iLen) ;

return WriteBufToSocket(s,buf,0,iLen,DEALLEN,OnSendFile) ;

}

catch(Exception err)

{

MessageBox.Show("
发送文件失败!"+err.Message) ;

return false ;

}

finally

{

fs.Close() ;

}

}

/// <summary>

///
对方对自己消息的简单回应

/// </summary>

/// <param name="s"></param>

/// <returns></returns>

public static string ReadResponsionFromSocket( Socket s)

{

byte [] bufCmd = new byte[RESPONLEN] ;

int iCount = s.Receive(bufCmd) ;

string strRespon = Encoding.UTF8.GetString(bufCmd,0,iCount) ;

return strRespon ;

}<o:p></o:p>

/// <summary>

///
Socket读取命令

/// </summary>

/// <param name="s">
要读取命令的Socket</param>

/// <returns>
读取的命令</returns>

public static string ReadCommandFromSocket( Socket s)

{

byte [] bufCmd = new byte[CMDLEN] ;

int iCount = s.Receive(bufCmd,0,CMDLEN,SocketFlags.Partial) ;

string strCommand = Encoding.UTF8.GetString(bufCmd,0,CMDLEN) ;

return strCommand = strCommand.TrimEnd(FILLCHAR) ;

}

/// <summary>

///
读取命令注释

/// </summary>

/// <param name="s">
要读取命令注释的Socket</param>

/// <returns>
读取的命令注释</returns>

public static string ReadCommandDescFromSocket( Socket s)

{

byte [] bufCmd = new byte[DESCLEN] ;

int iCount = s.Receive(bufCmd,0,DESCLEN,SocketFlags.Partial) ;

string strCommand = Encoding.UTF8.GetString(bufCmd,0,DESCLEN) ;

return strCommand = strCommand.TrimEnd(FILLCHAR) ;

}

/// <summary>

///
读取可变部分的长度

/// </summary>

/// <param name="s">
要读取可变部分长度的Socket</param>

/// <returns>
读取的可变部分的长度</returns>

public static int ReadDynamicLenFromSocket( Socket s)

{

byte [] bufCmd = new byte[DYNAMICLENGTHLEN] ;

int iCount = s.Receive(bufCmd,0,DYNAMICLENGTHLEN,SocketFlags.Partial) ;

string strCommand = Encoding.UTF8.GetString(bufCmd,0,DYNAMICLENGTHLEN) ;

return int.Parse(strCommand.TrimEnd(FILLCHAR)) ;

}

/// <summary>

///
读取文本形式的可变信息

/// </summary><o:p></o:p>

/// <param name="s">要读取可变信息的Socket</param>

/// <returns>
读取的可变信息</returns>

public static string ReadDynamicTextFromSocket( Socket s)

{

int iLen = ReadDynamicLenFromSocket(s) ;



byte [] buf = new byte[iLen] ;

string strInfo = "" ;



int iReceiveded = 0 ;

int iReceiveing = 0 ;

while(iReceiveded<iLen)

{

if (iReceiveded + DEALLEN <= iLen)

iReceiveing = DEALLEN ;

else

iReceiveing = iLen - iReceiveded ;

s.Receive(buf,iReceiveded,iReceiveing,SocketFlags.None) ;

CommunClass.WriteTextToSocket(s,"OK") ;

iReceiveded+= iReceiveing ;

}



strInfo = Encoding.UTF8.GetString(buf,0,iLen) ;



return strInfo ;

}

/// <summary>

///
读取文件形式的可变信息

/// </summary>

/// <param name="s">
要读取可变信息的Socket</param>

/// <param name="strFile">
读出后的文件保存位置</param>

/// <returns>
是否读取成功</returns>

public static bool ReadDynamicFileFromSocket( Socket s,string strFile)

{

int iLen = ReadDynamicLenFromSocket(s) ;

byte [] buf = new byte[iLen] ;

FileStream fs = new FileStream(strFile,FileMode.Create,FileAccess.Write) ;



try

{

int iReceiveded = 0 ;

int iReceiveing = 0 ;

while(iReceiveded<iLen)

{if (iReceiveded + DEALLEN <= iLen)

iReceiveing = DEALLEN ;

else

iReceiveing = iLen - iReceiveded ;

s.Receive(buf,iReceiveded,iReceiveing,SocketFlags.None) ;

CommunClass.WriteTextToSocket(s,"OK") ;

iReceiveded+= iReceiveing ;

}

fs.Write(buf,0,iLen) ;

return true ;

}

catch(Exception err)

{

MessageBox.Show("
接收文件失败"+err.Message) ;

return false ;

}

finally

{

fs.Close() ;

}

}

}//end class

}//end namespace

上面是俺的通讯基础类,有了这个类,再进行发送接受还不是小菜一碟吗?

分享到:
评论

相关推荐

    使用C#进行点对点通讯和文件传输

    使用C#进行点对点通讯和文件传输(通讯基类部分)-.NET教程,C#语言 最近一个项目要用到点对点文件传输,俺就到处找资料写程式,最后终于完成了,为了让别人少走些弯路,俺决定将俺程式中最重要的部分贡献出来,...

    C#基类库(苏飞版)

    Assistant创建显示图像的标签和文件 OWCChart统计图的封装类 2.Cookie&Session&Cache缓存帮助类 CacheHelper C#操作缓存的帮助类,实现了怎么设置缓存,怎么取缓存,怎么清理缓存等方法,只需要调用方法就可以...

    C#全能速查宝典

    分别介绍了C#语言基础、Windows窗体及常用控件、Windows高级控件、控件公共属性、方法及事件、数据库开发、文件、数据流与注册表、GDI+绘图技术和C#高级编程,共包含562个C#编程中常用的属性、方法、类和各种技术,...

    vc++ 应用源码包_1

    点对点多线程断点续传软件《传圣》源代码 多任务多线程断点续传DLL 多线程文件下载 多线程查找文件(功能增强版) 自定义了一个文件搜索类。多线程实现。 多线程高速文件搜索程序源码 VC++视频聊天系统源代码 实例...

    vc++ 应用源码包_2

    点对点多线程断点续传软件《传圣》源代码 多任务多线程断点续传DLL 多线程文件下载 多线程查找文件(功能增强版) 自定义了一个文件搜索类。多线程实现。 多线程高速文件搜索程序源码 VC++视频聊天系统源代码 实例...

    vc++ 应用源码包_6

    点对点多线程断点续传软件《传圣》源代码 多任务多线程断点续传DLL 多线程文件下载 多线程查找文件(功能增强版) 自定义了一个文件搜索类。多线程实现。 多线程高速文件搜索程序源码 VC++视频聊天系统源代码 实例...

    vc++ 应用源码包_3

    点对点多线程断点续传软件《传圣》源代码 多任务多线程断点续传DLL 多线程文件下载 多线程查找文件(功能增强版) 自定义了一个文件搜索类。多线程实现。 多线程高速文件搜索程序源码 VC++视频聊天系统源代码 实例...

    vc++ 应用源码包_5

    点对点多线程断点续传软件《传圣》源代码 多任务多线程断点续传DLL 多线程文件下载 多线程查找文件(功能增强版) 自定义了一个文件搜索类。多线程实现。 多线程高速文件搜索程序源码 VC++视频聊天系统源代码 实例...

    asp.net知识库

    使用Relations建立表之间的关系并却使用PagedDataSource类对DataList进行分页 通过作业,定时同步两个数据库 SQLSERVER高级注入技巧 利用反射实现ASP.NET控件和数据实体之间的双向绑定,并且在客户端自动验证输入的...

    wpf源码大全 精通C#3.0图书源码 详细源码 学习好用

    QueryOrderElement 演示通过XElement对XML元素进行排序。 QuerySubElement 演示通过XElement查询XML元素的子元素。 LoadXMLTree 演示通过XElement加载XML树。 第7章(\C07) 示例描述:本章介绍了WPF的...

    vc++ 开发实例源码包

    服务器可以将收集到的信息以柱状图和文件列表以及其他方式呈现给用户,以便用户对局域网内的主机进行监测和管理。 CClockST_demo 电子钟的实现,自绘Button、Static的实现,其中自定了一个辅助主题风格类。 ...

    公网qq聊天软件

    对消息发送进行管理。 7、修改部分程序BUG. 目前还存在问题: 1、用户数过多时登陆socket返回的用户列表无法显示;要分步传输; 2、大数据量的包,虽然自动拆包,但没有定义序列,不能按照顺序组包,如果网络不好就...

    MVC_EF框架_Bootstrap示例

    Model基类:提供数据传输和底层的最基本的基类及接口 DAL底层:基于EF code first,提供Repository泛型方法及写历史日志 Untility:通用函数库,基本都全了 Web:复写MVC基类,及通用MVC控件 通用模块Core 核心...

    【。net 专业】 面试题

    12.概述.NET里对 remoting 和 webservice 两项技术的理解和实际中的应用。 远程逻辑调用,remoing接口只能用在.net中 13.什么是code-behind技术 aspx and cs 14.概述三层结构体系 web/business/dataaccess 15.asp...

    【09-异常处理】

    •程序在try块里打开了一些物理资源(例如数据库连接、网络连接和磁盘文件等),这些物理资源都 必须显式回收。 •为了保证一定能回收try块中打开的物理资源,异常处理机制提供了finally块。不管try块中的代码...

    Silverlight2.0功能展示Demo源码

    Silverlight 2.0 使用c#开发可脚本化的代码,Silverlight与宿主页面的DOM之间的交互,Silverlight与宿主页面的JavaScript之间的交互。 ScriptableMemberAttribute - 需要脚本化的属性、方法、事件要标记为此 ...

Global site tag (gtag.js) - Google Analytics