C# TCP实现多个客户端与服务端 数据 与 文件的传输
C#菜鸟做这个东东竟然花了快三天的时间了,真是菜,菜,菜~~~
下面是我用C#写的 一个简单的TCP通信,主要的功能有:
(1) 多个客户端与服务器间的数据交流
(2)可以实现群发的功能
(3)客户端与服务端可以进行文件的传输
主要用到的知识: TCP里的 socket 、、、 多线程 Thread 、、、
下面的是界面:


下面分别是服务端和客户端的代码,如若借用,请标明出处~~~
服务端代码:
- 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,端口等;
- 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 = 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.Tcp);
- // 获得文本框中的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);
- return;
- }
- // 设置监听队列的长度;
- socketWatch.Listen(10);
- // 创建负责监听的线程;
- threadWatch = new Thread(WatchConnecting);
- threadWatch.IsBackground = true;
- threadWatch.Start();
- ShowMsg("服务器启动监听成功!");
- //}
- }
- /// <summary>
- /// 监听客户端请求的方法;
- /// </summary>
- void WatchConnecting()
- {
- while (true) // 持续不断的监听客户端的连接请求;
- {
- // 开始监听客户端连接请求,Accept方法会阻断当前的线程;
- Socket sokConnection = socketWatch.Accept(); // 一旦监听到一个客户端的请求,就返回一个与该客户端通信的 套接字;
- // 想列表控件中添加客户端的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(sokConnection.RemoteEndPoint.ToString(), thr); // 将新建的线程 添加 到线程的集合中去。
- }
- }
- void RecMsg(object sokConnectionparn)
- {
- Socket sokClient = sokConnectionparn as Socket;
- while (true)
- {
- // 定义一个2M的缓存区;
- byte[] arrMsgRec = new byte[1024 * 1024 * 2];
- // 将接受到的数据存入到输入 arrMsgRec中;
- int length = -1;
- try
- {
- length = sokClient.Receive(arrMsgRec); // 接收数据,并返回数据的长度;
- }
- catch (SocketException se)
- {
- ShowMsg("异常:" + se.Message);
- // 从 通信套接字 集合中删除被中断连接的通信套接字;
- dict.Remove(sokClient.RemoteEndPoint.ToString());
- // 从通信线程集合中删除被中断连接的通信线程对象;
- dictThread.Remove(sokClient.RemoteEndPoint.ToString());
- // 从列表中移除被中断的连接IP
- lbOnline.Items.Remove(sokClient.RemoteEndPoint.ToString());
- break;
- }
- catch (Exception e)
- {
- ShowMsg("异常:" + e.Message);
- // 从 通信套接字 集合中删除被中断连接的通信套接字;
- dict.Remove(sokClient.RemoteEndPoint.ToString());
- // 从通信线程集合中删除被中断连接的通信线程对象;
- dictThread.Remove(sokClient.RemoteEndPoint.ToString());
- // 从列表中移除被中断的连接IP
- lbOnline.Items.Remove(sokClient.RemoteEndPoint.ToString());
- break;
- }
- if (arrMsgRec[0] == 0) // 表示接收到的是数据;
- {
- string strMsg = System.Text.Encoding.UTF8.GetString(arrMsgRec,1, length-1);// 将接受到的字节数据转化成字符串;
- ShowMsg(strMsg);
- }
- if (arrMsgRec[0] == 1) // 表示接收到的是文件;
- {
- SaveFileDialog sfd = new SaveFileDialog();
- if (sfd.ShowDialog(this) == System.Windows.Forms.DialogResult.OK)
- {// 在上边的 sfd.ShowDialog() 的括号里边一定要加上 this 否则就不会弹出 另存为 的对话框,而弹出的是本类的其他窗口,,这个一定要注意!!!【解释:加了this的sfd.ShowDialog(this),“另存为”窗口的指针才能被SaveFileDialog的对象调用,若不加thisSaveFileDialog 的对象调用的是本类的其他窗口了,当然不弹出“另存为”窗口。】
- 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(str + "\r\n");
- }
- // 发送消息
- private void btnSend_Click(object sender, EventArgs e)
- {
- string strMsg = "服务器" + "\r\n" + " -->" + txtMsgSend.Text.Trim() + "\r\n";
- byte[] arrMsg = System.Text.Encoding.UTF8.GetBytes(strMsg); // 将要发送的字符串转换成Utf-8字节数组;
- byte[] arrSendMsg=new byte[arrMsg.Length+1];
- arrSendMsg[0] = 0; // 表示发送的是消息数据
- Buffer.BlockCopy(arrMsg, 0, arrSendMsg, 1, arrMsg.Length);
- string strKey = "";
- strKey = lbOnline.Text.Trim();
- if (string.IsNullOrEmpty(strKey)) // 判断是不是选择了发送的对象;
- {
- MessageBox.Show("请选择你要发送的好友!!!");
- }
- else
- {
- dict[strKey].Send(arrSendMsg);// 解决了 sokConnection是局部变量,不能再本函数中引用的问题;
- ShowMsg(strMsg);
- txtMsgSend.Clear();
- }
- }
- /// <summary>
- /// 群发消息
- /// </summary>
- /// <param name="sender"></param>
- /// <param name="e">消息</param>
- private void btnSendToAll_Click(object sender, EventArgs e)
- {
- string strMsg = "服务器" + "\r\n" + " -->" + txtMsgSend.Text.Trim() + "\r\n";
- byte[] arrMsg = System.Text.Encoding.UTF8.GetBytes(strMsg); // 将要发送的字符串转换成Utf-8字节数组;
- <span style="font-size:14px;"> byte[] arrSendMsg = new byte[arrMsg.Length + 1]; // 上次写的时候把这一段给弄掉了,实在是抱歉哈~ 用来标识发送是数据而不是文件,如果没有这一段的客户端就接收不到消息了~~~
- arrSendMsg[0] = 0; // 表示发送的是消息数据
- Buffer.BlockCopy(arrMsg, 0, arrSendMsg, 1, arrMsg.Length);</span>
- 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 = "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 (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 = "我给你发送的文件为: "+fileName+fileExtension+"\r\n";
- byte[] arrMsg = System.Text.Encoding.UTF8.GetBytes(strMsg); // 将要发送的字符串转换成Utf-8字节数组;
- byte[] arrSendMsg = new byte[arrMsg.Length + 1];
- arrSendMsg[0] = 0; // 表示发送的是消息数据
- Buffer.BlockCopy(arrMsg, 0, arrSendMsg, 1, arrMsg.Length);
- bool fff = true;
- string strKey = "";
- strKey = lbOnline.Text.Trim();
- if (string.IsNullOrEmpty(strKey)) // 判断是不是选择了发送的对象;
- {
- MessageBox.Show("请选择你要发送的好友!!!");
- }
- else
- {
- dict[strKey].Send(arrSendMsg);// 解决了 sokConnection是局部变量,不能再本函数中引用的问题;
- byte[] arrFile = new byte[1024 * 1024 * 2];
- int length = fs.Read(arrFile, 0, arrFile.Length); // 将文件中的数据读到arrFile数组中;
- byte[] arrFileSend = new byte[length + 1];
- arrFileSend[0] = 1; // 用来表示发送的是文件数据;
- Buffer.BlockCopy(arrFile, 0, arrFileSend, 1, length);
- // 还有一个 CopyTo的方法,但是在这里不适合; 当然还可以用for循环自己转化;
- // sockClient.Send(arrFileSend);// 发送数据到服务端;
- dict[strKey].Send(arrFileSend);// 解决了 sokConnection是局部变量,不能再本函数中引用的问题;
- txtSelectFile.Clear();
- }
- }
- }
- txtSelectFile.Clear();
- }
- }
- }
客户端代码:
- 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; // 创建用于接收服务端消息的 线程;
- Socket sockClient = null;
- private void btnConnect_Click(object sender, EventArgs e)
- {
- IPAddress ip = IPAddress.Parse(txtIp.Text.Trim());
- IPEndPoint endPoint=new IPEndPoint (ip,int.Parse(txtPort.Text.Trim()));
- sockClient = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
- try
- {
- ShowMsg("与服务器连接中……");
- sockClient.Connect(endPoint);
- }
- catch (SocketException se)
- {
- MessageBox.Show(se.Message);
- return;
- //this.Close();
- }
- ShowMsg("与服务器连接成功!!!");
- threadClient = new Thread(RecMsg);
- threadClient.IsBackground = true;
- threadClient.Start();
- }
- void RecMsg()
- {
- while (true)
- {
- // 定义一个2M的缓存区;
- byte[] arrMsgRec = new byte[1024 * 1024 * 2];
- // 将接受到的数据存入到输入 arrMsgRec中;
- int length = -1;
- try
- {
- length = sockClient.Receive(arrMsgRec); // 接收数据,并返回数据的长度;
- }
- catch (SocketException se)
- {
- ShowMsg("异常;" + se.Message);
- return;
- }
- catch (Exception e)
- {
- ShowMsg("异常:"+e.Message);
- return;
- }
- if (arrMsgRec[0] == 0) // 表示接收到的是消息数据;
- {
- string strMsg = System.Text.Encoding.UTF8.GetString(arrMsgRec, 1, length-1);// 将接受到的字节数据转化成字符串;
- ShowMsg(strMsg);
- }
- if (arrMsgRec[0] == 1) // 表示接收到的是文件数据;
- {
- try
- {
- SaveFileDialog sfd = new SaveFileDialog();
- if (sfd.ShowDialog(this) == System.Windows.Forms.DialogResult.OK)
- {// 在上边的 sfd.ShowDialog() 的括号里边一定要加上 this 否则就不会弹出 另存为 的对话框,而弹出的是本类的其他窗口,,这个一定要注意!!!【解释:加了this的sfd.ShowDialog(this),“另存为”窗口的指针才能被SaveFileDialog的对象调用,若不加thisSaveFileDialog 的对象调用的是本类的其他窗口了,当然不弹出“另存为”窗口。】
- string fileSavePath = sfd.FileName;// 获得文件保存的路径;
- // 创建文件流,然后根据路径创建文件;
- using (FileStream fs = new FileStream(fileSavePath, FileMode.Create))
- {
- fs.Write(arrMsgRec, 1, length - 1);
- ShowMsg("文件保存成功:" + fileSavePath);
- }
- }
- }
- catch (Exception aaa)
- {
- MessageBox.Show(aaa.Message);
- }
- }
- }
- }
- void ShowMsg(string str)
- {
- txtMsg.AppendText(str + "\r\n");
- }
- // 发送消息;
- private void btnSendMsg_Click(object sender, EventArgs e)
- {
- string strMsg = txtName.Text.Trim()+"\r\n"+" -->"+ txtSendMsg.Text.Trim()+ "\r\n";
- byte[] arrMsg = System.Text.Encoding.UTF8.GetBytes(strMsg);
- byte[] arrSendMsg = new byte[arrMsg.Length + 1];
- arrSendMsg[0] = 0; // 用来表示发送的是消息数据
- Buffer.BlockCopy(arrMsg, 0, arrSendMsg, 1, arrMsg.Length);
- sockClient.Send(arrSendMsg); // 发送消息;
- ShowMsg(strMsg);
- txtSendMsg.Clear();
- }
- // 选择要发送的文件;
- private void btnSelectFile_Click(object sender, EventArgs e)
- {
- OpenFileDialog ofd = new OpenFileDialog();
- ofd.InitialDirectory = "D:\\";
- if (ofd.ShowDialog() == System.Windows.Forms.DialogResult.OK)
- {
- txtSelectFile.Text = ofd.FileName;
- }
- }
- //向服务器端发送文件
- private void btnSendFile_Click(object sender, EventArgs e)
- {
- if (string.IsNullOrEmpty(txtSelectFile.Text))
- {
- MessageBox.Show("请选择要发送的文件!!!");
- }
- 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 = "我给你发送的文件为: " + fileName + "\r\n";
- byte[] arrMsg = System.Text.Encoding.UTF8.GetBytes(strMsg);
- byte[] arrSendMsg = new byte[arrMsg.Length + 1];
- arrSendMsg[0] = 0; // 用来表示发送的是消息数据
- Buffer.BlockCopy(arrMsg, 0, arrSendMsg, 1, arrMsg.Length);
- sockClient.Send(arrSendMsg); // 发送消息;
- byte[] arrFile = new byte[1024 * 1024 * 2];
- int length = fs.Read(arrFile, 0, arrFile.Length); // 将文件中的数据读到arrFile数组中;
- byte[] arrFileSend = new byte[length + 1];
- arrFileSend[0] = 1; // 用来表示发送的是文件数据;
- Buffer.BlockCopy(arrFile, 0, arrFileSend, 1, length);
- // 还有一个 CopyTo的方法,但是在这里不适合; 当然还可以用for循环自己转化;
- sockClient.Send(arrFileSend);// 发送数据到服务端;
- txtSelectFile.Clear();
- }
- }
- }
- }
- }
- 转自http://blog.csdn.net/chwei_cson/article/details/7737766
C# TCP实现多个客户端与服务端 数据 与 文件的传输的更多相关文章
- [通信] C# TCP实现多个客户端与服务端 数据 与 文件的传输
说明: http://download.csdn.net/detail/chwei_cson/4423874 源码: http://download.csdn.net/download/meicanj ...
- 编写Java程序,实现客户端向服务端上传文件的功能
查看本章节 查看作业目录 需求说明: 实现客户端向服务端上传文件的功能 当启动服务端后,运行客户端程序,系统提示客户在客户端输入上传文件的完整路径.当客户在客户端输入完成后,服务端实现文件上传 实现思 ...
- TCP学习之五:客户端、服务端异步传输字符串
参考学习张子阳大神的博客:http://www.cnblogs.com/JimmyZhang/category/101698.html 消息发送接口: 消息接收接口: 客户端: 服务端: 消息发送类: ...
- TCP学习之三:客户端、服务端同步传输字符串
参考学习张子阳大神的博客:http://www.cnblogs.com/JimmyZhang/category/101698.html 一个客户端.发送一条消息 客户端: 服务端: 注意:Networ ...
- TCP Socket 通讯(客户端与服务端)
/*----------------------------编译环境:VS2015---------------------------------------*/ /*--------------- ...
- C#客户端和服务端数据的同步传输 (转载)
客户端: using System;using System.Collections.Generic;using System.ComponentModel;using System.Data;u ...
- Django服务端读取excel文件并且传输到接口
path_name = "opboss_download_" + str(int(time.time())) + ".csv" print(path_name) ...
- linux epoll机制对TCP 客户端和服务端的监听C代码通用框架实现
1 TCP简介 tcp是一种基于流的应用层协议,其“可靠的数据传输”实现的原理就是,“拥塞控制”的滑动窗口机制,该机制包含的算法主要有“慢启动”,“拥塞避免”,“快速重传”. 2 TCP socket ...
- Java实现TCP之Echo客户端和服务端
Java实现TCP之Echo客户端和服务端 代码内容 采用TCP协议编写服务器端代码(端口任意) 编写客户机的代码访问该端口 客户机按行输入 服务器将收到的字符流和接收到的时间输出在服务器consol ...
随机推荐
- 每一个程序员需要了解的10个Linux命令
作为一个程序员,在软件开发职业生涯中或多或少会用到Linux系统,并且可能会使用Linux命令来检索需要的信息.本文将为各位开发者分享10个有用的Linux命令,希望对你会有所帮助. 以下就是今天我们 ...
- 【MRPT】【icp-slam-live】Vs2013+ cmake3.6.1 + mrpt1.4.0+opencv2.9.4+wxWidget3.0.2环境配置
Win10下Vs2013 + cmake3.6.1 + mrpt1.4.0+opencv2.9.4+wxWidget3.1.0环境配置 所接触过的最令我崩溃的环境配置.之前没有考虑到vs2013 20 ...
- SyncServer obj
- ASP.NET Core文档中Work with Data章节的翻译目录
作为初学者看了相关的教程,遇到的问题有: 1. 教程不是针对初学者,往往在某一方面教的较深,但并不系统,不适合初学者: 2. 虽然翻译的很顺畅,但是谈了自己较多的开发体会,初学者看着困难,尤其是TOM ...
- hdu 1421
时隔多日,又回来啃dp... 题意:有n件物品,搬k次,每搬一个消耗的疲劳值为两件物品重量之差的平方,求最小的疲劳消耗 状态转移方程:dp[i][j] = min((dp[i-2][j-1]+(s[i ...
- DSP基础学习-ADC同步采样
DSP基础学习-ADC同步采样 彭会锋 2015-04-28 20:31:06 在DSP28027 LauchPad学习过程中,关于ADC同步采样和顺序采样的区别稍加研究了一下,发现里面还真有些门道, ...
- R语言自带数据包
向量 euro #欧元汇率,长度为11,每个元素都有命名 landmasses #48个陆地的面积,每个都有命名 precip #长度为70的命名向量 rivers #北美14 ...
- [saiku] 配置spring-security 允许 iframe加载saiku首页
最近提出了一个需求:在一个iframe中展现saiku首页 呵呵,这还不简单. 直接<iframe src="http://localhost:8080/saiku" /&g ...
- C# winform程序怎么打包成安装项目(图解)
1:新建安装部署项目 打开VS,点击新建项目,选择:其他项目类型->安装与部署->安装向导(安装项目也一样),然后点击确定.(详细见下图) 此主题相关图片如下: 2:安装向导 关闭后打开安 ...
- hdu 1811 Rank of Tetris (并查集+拓扑排序)
Rank of Tetris Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)To ...