C#学习教程之Socket的简单使用

前言

在开始介绍socket前先补充补充基础知识,在此基础上理解网络通信才会顺理成章,当然有基础的可以跳过去了。都是废话,进入正题。

TCP/IP:Transmission Control Protocol/Internet Protocol,传输控制协议/因特网互联协议,又名网络通讯协议。简单来说:TCP控制传输数据,负责发现传输的问题,一旦有问题就发出信号,要求重新传输,直到所有数据安全正确地传输到目的地,而IP是负责给因特网中的每一台电脑定义一个地址,以便传输。从协议分层模型方面来讲:TCP/IP由:网络接口层(链路层)、网络层、传输层、应用层。它和OSI的七层结构以及对于协议族不同,下图简单表示:

注:第一张图:TCP/IP的四层结构对应OSI七层结构。

第三张图:TCP/IP协议族在OSI七层中的位置及对应的功能。

第二张图:TCP/IP协议模块关系图。

现阶段socket通信使用TCP、UDP协议,相对应UDP来说,TCP则是比较安全稳定的协议了。

Socket是一种通信TCP/IP的通讯接口,也就是HTTP的抽象层,就是Socket在Http之上,Socket也就是发动机。实际上,传输层的TCP是基于网络层的IP协议的,而应用层的HTTP协议又是基于传输层的TCP协议的,而Socket本身不算是协议,就像上面所说,它只是提供了一个针对TCP或者UDP编程的接口。

在C#中可以非常方便的使用Socket进行数据传输。

Socket对象是C#使用它的重要对象在Socket的构造函数中,我们可以设置它的地址,Socket的类,支持的协议等等,其定义如下:

public Socket(AddressFamily addressFamily, SocketType socketType, ProtocolType protocolType);

我们想要使用Socket,那么就必须创建Socket的对象,创建这个对象,就必须需要IPEndPoint对象来绑定到套接词字中,有如下定义

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

然后再通过Socket的Bind来进行绑定。

socketWatch.Bind(endPoint);

因为我们时刻会被内网中的其他ip和端口进行连接,那么我们就需要创建线程来进行观察,有如下定义

// 设置监听队列的长度;
   socketWatch.Listen(10);
   // 创建负责监听的线程;
   threadWatch = new Thread(WatchConnecting);
   threadWatch.IsBackground = true;
   threadWatch.Start();
   ShowMsg("服务器启动监听成功!");

其检测方法如下,其中就是不断的去检测客户端的连接请求,通过Accept()方法可以获取一个套接字,然后通过Socket对象的RemoteEndPoint()可以获取一个IP。

void WatchConnecting()
  {
   while (true) // 持续不断的监听客户端的连接请求;
   {
    // 开始监听客户端连接请求,Accept方法会阻断当前的线程;
    Socket sokConnection = socketWatch.Accept(); // 一旦监听到一个客户端的请求,就返回一个与该客户端通信的 套接字;
    // 想列表控件中添加客户端的IP信息;
    Online.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); // 将新建的线程 添加 到线程的集合中去。
   }
  }

最后我们开启一个线程去执行RecMsg方法,然后我们不停的去监听客户端给我们的数据发送。

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
     Online.Items.Remove(sokClient.RemoteEndPoint.ToString());
     break;
    }
    catch (Exception e)
    {
     ShowMsg("异常:" + e.Message);
     // 从 通信套接字 集合中删除被中断连接的通信套接字;
     dict.Remove(sokClient.RemoteEndPoint.ToString());
     // 从通信线程集合中删除被中断连接的通信线程对象;
     dictThread.Remove(sokClient.RemoteEndPoint.ToString());
     // 从列表中移除被中断的连接IP
     Online.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);
      }
     }
    }
   }
  }

我们在方法中获得了一个Object类型的对象,将这个Object对象转换成了Socket,然后我们通过Socket的方法Receive()方法接收返回的数据,其中里面有它的属性,可以获取ip还有一些数据等等。服务器向客户端发送数据也是非常简单。通过Send方法就可以了,如以下定义:

string strMsg = "服务器" + "\r\n" + " -->" + richTextBox1.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 = Online.Text.Trim();
   if (string.IsNullOrEmpty(strKey)) // 判断是不是选择了发送的对象;
   {
    MessageBox.Show("请选择你要发送的好友!!!");
   }
   else
   {
    dict[strKey].Send(arrSendMsg);// 解决了 sokConnection是局部变量,不能再本函数中引用的问题;
    ShowMsg(strMsg);
    richTextBox1.Clear();
   }

最后需要注意的是,如果你的文件较大,有的时候这个缓冲区达不到你的文件字节那么大,那么就会截断,所以与的时候,先将文件转换为Byte是正确的做法。只要在客户端进行逆转就可以了!

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • C# [ImportDll()] 知识小结

    这周在做公司的一个C#项目中,要写一个webservice提供一个下载方法,之前公司有过,但是要整改,于是这种鸟屎摊子又交给了我,其中一个密文流的下载中要应用我们小组另一伙人用C++写的四个dll, 这些DLL是由我们自己编写的,非.NET 托管,这是一个麻烦事: 如果用 IDE往工程里add reference时,它会提示这几个dll是没有注册的,不是.NET托管的.好吧,顺便补充一下.NET托管和非托管. 具体官方术语是: 托管DLL就是能够在公共语言运行库(Common Language

  • C#实现Json转DataTable并导出Excel的方法示例

    本文实例讲述了C#实现Json转DataTable并导出Excel的方法.分享给大家供大家参考,具体如下: 需求:有一个log文件,需要整理成Excel,日志文件里面的数据都是json字符串 思路是,把Json字符串转换成DataTable,然后导出到Excel 在网上找了一些资料,整理了以下三种类型的Json 一.Json转换DataTable 1.处理简单Json: [{"mac":"20:f1:7c:c5:cd:80","rssi":&qu

  • C#使用base64对字符串进行编码和解码的测试

    需要引入命名空间: using System; using System.Text; 解码: public static string UnBase64String(string value) { if (value == null || value == "") { return ""; } byte[] bytes = Convert.FromBase64String(value); return Encoding.UTF8.GetString(bytes);

  • C#基于HttpWebRequest实现发送HTTP请求的方法分析

    本文实例讲述了C#基于HttpWebRequest实现发送HTTP请求的方法.分享给大家供大家参考,具体如下: 调用第三方API的时候要用到HttpWebRequest类发送HTTP请求,网上查阅一番后大致了解了该类的用法,现记录如下. 首先引入HttpWebRequest类,System.IO类 using HttpWebRequest using System.IO GET请求 /// <summary> /// 发送GET请求 /// </summary> /// <p

  • C#将Excel转成PDF的方法

    PS:公司的业务中有个超级大的作业就是把OFFICE文档转成PDF,我猜之前没程序猿们,公司那些人应该是一个个手动转.强烈为猿们感叹,帮你们做了这么多事,还在那抱怨....无法满足你们的需求啊: 微软net平台提供了对Office文档非常好的支持:其中有com组件直接集成到了VS中.利用这些API可以快速的免去N多繁琐的工作: 以下代码是翻阅了公司的代码,一个个敲出来的:奉上代码: using System; using System.Collections.Generic; using Sys

  • C#如何给枚举类型增加一个描述特性详解

    前言 相信很多人对枚举并不陌生,枚举可以很方便和直观的管理一组特定值.如果我们在页面上直接输出我们希望匹配的汉语意思或则其他满足我们需求的语句就更好了,当然,通常小伙伴们都会再页面上if(enum==1) "我是一个枚举"或者switch(enum)这种方式解决. 枚举的优点: <1>枚举可以使代码更易于维护,有助于确保给变量指定合法的.期望的值. <2>枚举使代码更清晰,允许用描述性的名称表示整数值,而不是用含义模糊的数来表示. <3>枚举使代码更

  • C#实现Word转为PDF的方法

    这里主要提供一种将word文档转成PDF文档的实现: 具体实现看c#代码:要引入Microsoft.Office.Interop.Word;版本12.0.0.0 using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Text; using System.Runtime.InteropServices; using Microsoft.Office.Inter

  • C# Winform实现石头剪刀布游戏

    本文实例为大家分享了Winform实现石头剪刀布游戏的具体代码,供大家参考,具体内容如下 新建一个windows窗体程序,用数字1代表石头,用数字2代表剪刀,用数字3代表布,结果取玩家和电脑出拳之差,有三种结果 玩家赢: -1,2 平手: 0 玩家输: 其它值 新建3个类: 1)Computer.cs 电脑随机出拳 using System; using System.Collections.Generic; using System.Linq; using System.Text; using

  • C#中Hash table的一些操作方法讲解

    散列表(Hash table,也叫哈希表),是根据关键码值(Key value)而直接进行访问的数据结构.也就是说,它通过把关键码值映射到表中一个位置来访问记录,以加快查找的速度.这个映射函数叫做散列函数,存放记录的数组叫做散列表. 今天难得有了半天时间,鸣炮!任务都做完了,我终于有点时间去分析公司这个项目的代码,看到了其中有好多的对Hash table的应用.有好多的BLL层的代码的方法传入的是hashtable.其实我所做的模块中,如果有超过三个变量以上的参数传入我会将这些封装成一个个对象,

  • c#操作mongodb插入数据效率

    mongodb的数据插入速度是其一个亮点,同样的10000条数据,插入的速度要比Mysql和sqlserver都要快,当然这也是要看使用者怎么个使用法,你代码如果10000次写入使用10000次连接,那也是比不过其他数据库使用事务一次性提交的速度的. 同样,mongo也提供的一次性插入巨量数据的方法,因为mongodb没有事务这回事,所以在在C#驱动里,具体方法是InsertManyAsync()一次性插入多个文档.与之对应的是InsertOneAsync,这个是一次插入一个文档: Insert

随机推荐