C#-TCP实现多个客户端与服务端-数据-与-文件的传输_第1页
C#-TCP实现多个客户端与服务端-数据-与-文件的传输_第2页
C#-TCP实现多个客户端与服务端-数据-与-文件的传输_第3页
C#-TCP实现多个客户端与服务端-数据-与-文件的传输_第4页
C#-TCP实现多个客户端与服务端-数据-与-文件的传输_第5页
已阅读5页,还剩21页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、C#菜鸟做这个东东竟然花了快三天的时间了,真是菜,菜,菜下面是我用C#写的 一个简单的TCP通信,主要的功能有:(1) 多个客户端与服务器间的数据交流(2)可以实现群发的功能(3)客户端与服务端可以进行文件的传输主要用到的知识: TCP里的 socket 、 多线程 Thread 、下面的是界面:推荐精选下面分别是服务端和客户端的代码,如若借用,请标明出处服务端代码:csharp view plaincopyprint?using System; using System.Collections.Generic; using System.ComponentModel; using Syste

2、m.Data; using System.Drawing; using System.Linq; using System.Text; using System.Windows.Forms; using System.Net.Sockets; using System.Net; / IP,IPAddress, IPEndPoint,端口等; using System.Threading; using System.IO; namespace _11111 public partial class frm_server : Form public frm_server() 推荐精选 Initia

3、lizeComponent(); TextBox.CheckForIllegalCrossThreadCalls = false; Thread threadWatch = null; / 负责监听客户端连接请求的 线程; Socket socketWatch = null; Dictionary<string, Socket> dict = new Dictionary<string, Socket>(); Dictionary<string, Thread> dictThread = new Dictionary<string, Thread>

4、;(); private void btnBeginListen_Click(object sender, EventArgs e) / 创建负责监听的套接字,注意其中的参数; socketWatch = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp); / 获得文本框中的IP对象; IPAddress address = IPAddress.Parse(txtIp.Text.Trim(); / 创建包含ip和端口号的网络节点对象; IPEndPoint endPoint = new IPE

5、ndPoint(address, int.Parse(txtPort.Text.Trim(); try / 将负责监听的套接字绑定到唯一的ip和端口上; socketWatch.Bind(endPoint); catch (SocketException se) MessageBox.Show("异常:"+se.Message); return; / 设置监听队列的长度; socketWatch.Listen(10); / 创建负责监听的线程; threadWatch = new Thread(WatchConnecting); threadWatch.IsBackgrou

6、nd = true; threadWatch.Start(); ShowMsg("服务器启动监听成功!"); / / <summary> / 监听客户端请求的方法; / </summary> void WatchConnecting() 推荐精选 while (true) / 持续不断的监听客户端的连接请求; / 开始监听客户端连接请求,Accept方法会阻断当前的线程; Socket sokConnection = socketWatch.Accept(); / 一旦监听到一个客户端的请求,就返回一个与该客户端通信的 套接字; / 想列表控件中添加

7、客户端的IP信息; lbOnline.Items.Add(sokConnection.RemoteEndPoint.ToString(); / 将与客户端连接的 套接字 对象添加到集合中; dict.Add(sokConnection.RemoteEndPoint.ToString(), sokConnection); ShowMsg("客户端连接成功!"); Thread thr = new Thread(RecMsg); thr.IsBackground = true; thr.Start(sokConnection); dictThread.Add(sokConnec

8、tion.RemoteEndPoint.ToString(), thr); / 将新建的线程 添加 到线程的集合中去。 void RecMsg(object sokConnectionparn) Socket sokClient = sokConnectionparn as Socket; while (true) / 定义一个2M的缓存区; byte arrMsgRec = new byte1024 * 1024 * 2; / 将接受到的数据存入到输入 arrMsgRec中; int length = -1; try length = sokClient.Receive(arrMsgRec)

9、; / 接收数据,并返回数据的长度; catch (SocketException se) ShowMsg("异常:" + se.Message); / 从 通信套接字 集合中删除被中断连接的通信套接字; dict.Remove(sokClient.RemoteEndPoint.ToString(); / 从通信线程集合中删除被中断连接的通信线程对象; dictThread.Remove(sokClient.RemoteEndPoint.ToString(); / 从列表中移除被中断的连接IP lbOnline.Items.Remove(sokClient.RemoteEn

10、dPoint.ToString(); break; catch (Exception e) 推荐精选 ShowMsg("异常:" + e.Message); / 从 通信套接字 集合中删除被中断连接的通信套接字; dict.Remove(sokClient.RemoteEndPoint.ToString(); / 从通信线程集合中删除被中断连接的通信线程对象; dictThread.Remove(sokClient.RemoteEndPoint.ToString(); / 从列表中移除被中断的连接IP lbOnline.Items.Remove(sokClient.Remo

11、teEndPoint.ToString(); break; if (arrMsgRec0 = 0) / 表示接收到的是数据; string strMsg = System.Text.Encoding.UTF8.GetString(arrMsgRec,1, length-1);/ 将接受到的字节数据转化成字符串; ShowMsg(strMsg); if (arrMsgRec0 = 1) / 表示接收到的是文件; SaveFileDialog sfd = new SaveFileDialog(); if (sfd.ShowDialog(this) = System.Windows.Forms.Di

12、alogResult.OK) / 在上边的 sfd.ShowDialog() 的括号里边一定要加上 this 否则就不会弹出 另存为 的对话框,而弹出的是本类的其他窗口,这个一定要注意!【解释:加了this的sfd.ShowDialog(this),“另存为”窗口的指针才能被SaveFileDialog的对象调用,若不加thisSaveFileDialog 的对象调用的是本类的其他窗口了,当然不弹出“另存为”窗口。】 string fileSavePath = sfd.FileName;/ 获得文件保存的路径; / 创建文件流,然后根据路径创建文件; using (FileStream fs

13、= new FileStream(fileSavePath, FileMode.Create) fs.Write(arrMsgRec, 1, length - 1); ShowMsg("文件保存成功:" + fileSavePath); void ShowMsg(string str) txtMsg.AppendText(str + "rn"); 推荐精选/ 发送消息 private void btnSend_Click(object sender, EventArgs e) string strMsg = "服务器" + "

14、;rn" + " ->" + txtMsgSend.Text.Trim() + "rn" byte arrMsg = System.Text.Encoding.UTF8.GetBytes(strMsg); / 将要发送的字符串转换成Utf-8字节数组; byte arrSendMsg=new bytearrMsg.Length+1; arrSendMsg0 = 0; / 表示发送的是消息数据 Buffer.BlockCopy(arrMsg, 0, arrSendMsg, 1, arrMsg.Length); string strKey =

15、 "" strKey = lbOnline.Text.Trim(); if (string.IsNullOrEmpty(strKey) / 判断是不是选择了发送的对象; MessageBox.Show("请选择你要发送的好友!"); else dictstrKey.Send(arrSendMsg);/ 解决了 sokConnection是局部变量,不能再本函数中引用的问题; ShowMsg(strMsg); txtMsgSend.Clear(); / <summary> / 群发消息 / </summary> / <para

16、m name="sender"></param> / <param name="e">消息</param> private void btnSendToAll_Click(object sender, EventArgs e) string strMsg = "服务器" + "rn" + " ->" + txtMsgSend.Text.Trim() + "rn" byte arrMsg = System.Text.Encodin

17、g.UTF8.GetBytes(strMsg); / 将要发送的字符串转换成Utf-8字节数组; using System;using System.Collections.Generic;using System.ComponentModel;using System.Data;using System.Drawing;using System.Linq;using System.Text;using System.Windows.Forms;using System.Net.Sockets;using System.Net; / IP,IPAddress, IPEndPoint,端口等;推

18、荐精选using System.Threading;using System.IO;namespace _11111 public partial class frm_server : Form public frm_server() InitializeComponent(); TextBox.CheckForIllegalCrossThreadCalls = false; Thread threadWatch = null; / 负责监听客户端连接请求的 线程; Socket socketWatch = null; Dictionary<string, Socket> dict

19、 = new Dictionary<string, Socket>(); Dictionary<string, Thread> dictThread = new Dictionary<string, Thread>(); private void btnBeginListen_Click(object sender, EventArgs e) / 创建负责监听的套接字,注意其中的参数; socketWatch = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tc

20、p); / 获得文本框中的IP对象; IPAddress address = IPAddress.Parse(txtIp.Text.Trim(); / 创建包含ip和端口号的网络节点对象; IPEndPoint endPoint = new IPEndPoint(address, int.Parse(txtPort.Text.Trim(); try / 将负责监听的套接字绑定到唯一的ip和端口上; socketWatch.Bind(endPoint); catch (SocketException se) MessageBox.Show("异常:"+se.Message);

21、 return; / 设置监听队列的长度; socketWatch.Listen(10); / 创建负责监听的线程; threadWatch = new Thread(WatchConnecting); threadWatch.IsBackground = true;推荐精选 threadWatch.Start(); ShowMsg("服务器启动监听成功!"); / / <summary> / 监听客户端请求的方法; / </summary> void WatchConnecting() while (true) / 持续不断的监听客户端的连接请求;

22、 / 开始监听客户端连接请求,Accept方法会阻断当前的线程; Socket sokConnection = socketWatch.Accept(); / 一旦监听到一个客户端的请求,就返回一个与该客户端通信的 套接字; / 想列表控件中添加客户端的IP信息; lbOnline.Items.Add(sokConnection.RemoteEndPoint.ToString(); / 将与客户端连接的 套接字 对象添加到集合中; dict.Add(sokConnection.RemoteEndPoint.ToString(), sokConnection); ShowMsg("客户

23、端连接成功!"); Thread thr = new Thread(RecMsg); thr.IsBackground = true; thr.Start(sokConnection); dictThread.Add(sokConnection.RemoteEndPoint.ToString(), thr); / 将新建的线程 添加 到线程的集合中去。 void RecMsg(object sokConnectionparn) Socket sokClient = sokConnectionparn as Socket; while (true) / 定义一个2M的缓存区; byte

24、 arrMsgRec = new byte1024 * 1024 * 2; / 将接受到的数据存入到输入 arrMsgRec中; int length = -1; try length = sokClient.Receive(arrMsgRec); / 接收数据,并返回数据的长度; catch (SocketException se)推荐精选 ShowMsg("异常:" + se.Message); / 从 通信套接字 集合中删除被中断连接的通信套接字; dict.Remove(sokClient.RemoteEndPoint.ToString(); / 从通信线程集合中删

25、除被中断连接的通信线程对象; dictThread.Remove(sokClient.RemoteEndPoint.ToString(); / 从列表中移除被中断的连接IP lbOnline.Items.Remove(sokClient.RemoteEndPoint.ToString(); break; catch (Exception e) ShowMsg("异常:" + e.Message); / 从 通信套接字 集合中删除被中断连接的通信套接字; dict.Remove(sokClient.RemoteEndPoint.ToString(); / 从通信线程集合中删除

26、被中断连接的通信线程对象; dictThread.Remove(sokClient.RemoteEndPoint.ToString(); / 从列表中移除被中断的连接IP lbOnline.Items.Remove(sokClient.RemoteEndPoint.ToString(); break; if (arrMsgRec0 = 0) / 表示接收到的是数据; string strMsg = System.Text.Encoding.UTF8.GetString(arrMsgRec,1, length-1);/ 将接受到的字节数据转化成字符串; ShowMsg(strMsg); if (

27、arrMsgRec0 = 1) / 表示接收到的是文件; SaveFileDialog sfd = new SaveFileDialog(); if (sfd.ShowDialog(this) = System.Windows.Forms.DialogResult.OK) / 在上边的 sfd.ShowDialog() 的括号里边一定要加上 this 否则就不会弹出 另存为 的对话框,而弹出的是本类的其他窗口,这个一定要注意!【解释:加了this的sfd.ShowDialog(this),“另存为”窗口的指针才能被SaveFileDialog的对象调用,若不加thisSaveFileDialo

28、g 的对象调用的是本类的其他窗口了,当然不弹出“另存为”窗口。】 string fileSavePath = sfd.FileName;/ 获得文件保存的路径;推荐精选 / 创建文件流,然后根据路径创建文件; using (FileStream fs = new FileStream(fileSavePath, FileMode.Create) fs.Write(arrMsgRec, 1, length - 1); ShowMsg("文件保存成功:" + fileSavePath); void ShowMsg(string str) txtMsg.AppendText(st

29、r + "rn"); / 发送消息 private void btnSend_Click(object sender, EventArgs e) string strMsg = "服务器" + "rn" + " ->" + txtMsgSend.Text.Trim() + "rn" byte arrMsg = System.Text.Encoding.UTF8.GetBytes(strMsg); / 将要发送的字符串转换成Utf-8字节数组; byte arrSendMsg=new byt

30、earrMsg.Length+1; arrSendMsg0 = 0; / 表示发送的是消息数据 Buffer.BlockCopy(arrMsg, 0, arrSendMsg, 1, arrMsg.Length); string strKey = "" strKey = lbOnline.Text.Trim(); if (string.IsNullOrEmpty(strKey) / 判断是不是选择了发送的对象; MessageBox.Show("请选择你要发送的好友!"); else dictstrKey.Send(arrSendMsg);/ 解决了 so

31、kConnection是局部变量,不能再本函数中引用的问题; ShowMsg(strMsg); txtMsgSend.Clear(); / <summary> / 群发消息推荐精选 / </summary> / <param name="sender"></param> / <param name="e">消息</param> private void btnSendToAll_Click(object sender, EventArgs e) string strMsg = &qu

32、ot;服务器" + "rn" + " ->" + txtMsgSend.Text.Trim() + "rn" byte arrMsg = System.Text.Encoding.UTF8.GetBytes(strMsg); / 将要发送的字符串转换成Utf-8字节数组;csharp view plaincopyprint? csharp view plaincopyprint?<span style="font-size: 14px;"> byte arrSendMsg = new b

33、ytearrMsg.Length + 1; / 上次写的时候把这一段给弄掉了,实在是抱歉哈 用来标识发送是数据而不是文件,如果没有这一段的客户端就接收不到消息了 arrSendMsg0 = 0; / 表示发送的是消息数据 Buffer.BlockCopy(arrMsg, 0, arrSendMsg, 1, arrMsg.Length);</span> byte arrSendMsg = new bytearrMsg.Length + 1; / 上次写的时候把这一段给弄掉了,实在是抱歉哈 用来标识发送是数据而不是文件,如果没有这一段的客户端就接收不到消息了 arrSendMsg0 =

34、 0; / 表示发送的是消息数据 Buffer.BlockCopy(arrMsg, 0, arrSendMsg, 1, arrMsg.Length);csharp view plaincopyprint?foreach (Socket s in dict.Values) s.Send(arrMsg); ShowMsg(strMsg); txtMsgSend.Clear(); ShowMsg(" 群发完毕"); / 选择要发送的文件 private void btnSelectFile_Click_1(object sender, EventArgs e) OpenFileD

35、ialog ofd = new OpenFileDialog(); ofd.InitialDirectory = "D:" if (ofd.ShowDialog() = System.Windows.Forms.DialogResult.OK) txtSelectFile.Text = ofd.FileName; 推荐精选/ 文件的发送 private void btnSendFile_Click_1(object sender, EventArgs e) if (string.IsNullOrEmpty(txtSelectFile.Text) MessageBox.Sho

36、w("请选择你要发送的文件!"); else / 用文件流打开用户要发送的文件; using (FileStream fs = new FileStream(txtSelectFile.Text, FileMode.Open) string fileName=System.IO.Path.GetFileName(txtSelectFile.Text); string fileExtension=System.IO.Path.GetExtension(txtSelectFile.Text); string strMsg = "我给你发送的文件为: "+fi

37、leName+fileExtension+"rn" byte arrMsg = System.Text.Encoding.UTF8.GetBytes(strMsg); / 将要发送的字符串转换成Utf-8字节数组; byte arrSendMsg = new bytearrMsg.Length + 1; arrSendMsg0 = 0; / 表示发送的是消息数据 Buffer.BlockCopy(arrMsg, 0, arrSendMsg, 1, arrMsg.Length); bool fff = true; string strKey = "" st

38、rKey = lbOnline.Text.Trim(); if (string.IsNullOrEmpty(strKey) / 判断是不是选择了发送的对象; MessageBox.Show("请选择你要发送的好友!"); else dictstrKey.Send(arrSendMsg);/ 解决了 sokConnection是局部变量,不能再本函数中引用的问题; byte arrFile = new byte1024 * 1024 * 2; int length = fs.Read(arrFile, 0, arrFile.Length); / 将文件中的数据读到arrFil

39、e数组中; byte arrFileSend = new bytelength + 1; arrFileSend0 = 1; / 用来表示发送的是文件数据; Buffer.BlockCopy(arrFile, 0, arrFileSend, 1, length); / 还有一个 CopyTo的方法,但是在这里不适合; 当然还可以用for循环自己转化; / sockClient.Send(arrFileSend);/ 发送数据到服务端; dictstrKey.Send(arrFileSend);/ 解决了 sokConnection是局部变量,不能再本函数中引用的问题; txtSelectFil

40、e.Clear(); 推荐精选 txtSelectFile.Clear(); foreach (Socket s in dict.Values) s.Send(arrMsg); ShowMsg(strMsg); txtMsgSend.Clear(); ShowMsg(" 群发完毕"); / 选择要发送的文件 private void btnSelectFile_Click_1(object sender, EventArgs e) OpenFileDialog ofd = new OpenFileDialog(); ofd.InitialDirectory = "

41、D:" if (ofd.ShowDialog() = System.Windows.Forms.DialogResult.OK) txtSelectFile.Text = ofd.FileName; / 文件的发送 private void btnSendFile_Click_1(object sender, EventArgs e) if (string.IsNullOrEmpty(txtSelectFile.Text) MessageBox.Show("请选择你要发送的文件!"); else / 用文件流打开用户要发送的文件; using (FileStrea

42、m fs = new FileStream(txtSelectFile.Text, FileMode.Open) string fileName=System.IO.Path.GetFileName(txtSelectFile.Text); string fileExtension=System.IO.Path.GetExtension(txtSelectFile.Text);推荐精选 string strMsg = "我给你发送的文件为: "+fileName+fileExtension+"rn" byte arrMsg = System.Text.E

43、ncoding.UTF8.GetBytes(strMsg); / 将要发送的字符串转换成Utf-8字节数组; byte arrSendMsg = new bytearrMsg.Length + 1; arrSendMsg0 = 0; / 表示发送的是消息数据 Buffer.BlockCopy(arrMsg, 0, arrSendMsg, 1, arrMsg.Length); bool fff = true; string strKey = "" strKey = lbOnline.Text.Trim(); if (string.IsNullOrEmpty(strKey) /

44、 判断是不是选择了发送的对象; MessageBox.Show("请选择你要发送的好友!"); else dictstrKey.Send(arrSendMsg);/ 解决了 sokConnection是局部变量,不能再本函数中引用的问题; byte arrFile = new byte1024 * 1024 * 2; int length = fs.Read(arrFile, 0, arrFile.Length); / 将文件中的数据读到arrFile数组中; byte arrFileSend = new bytelength + 1; arrFileSend0 = 1;

45、/ 用来表示发送的是文件数据; Buffer.BlockCopy(arrFile, 0, arrFileSend, 1, length); / 还有一个 CopyTo的方法,但是在这里不适合; 当然还可以用for循环自己转化; / sockClient.Send(arrFileSend);/ 发送数据到服务端; dictstrKey.Send(arrFileSend);/ 解决了 sokConnection是局部变量,不能再本函数中引用的问题; txtSelectFile.Clear(); txtSelectFile.Clear(); 客户端代码:csharp view plaincopypr

46、int?using System; 推荐精选using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Text; using System.Windows.Forms; using System.Net; using System.Net.Sockets; using System.Threading; using System.IO; namespace _2222222 public partial class frmClient : Form public frmClient() InitializeComponent(); TextBox.CheckForIllegalCrossThreadCalls = false; Thread threadClient = null; / 创建用于接收服务端消息的 线程; Socke

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

最新文档

评论

0/150

提交评论