百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 热门文章 > 正文

c#Socket通信实例

bigegpt 2024-08-23 11:51 2 浏览

一、Socket通信简介

1.按惯例先来介绍下socket

Windows中的很多东西都是从Unix领域借鉴过来的,Socket也是一样。在Unix中,socket代表了一种文件描述符(在Unix中一切都是以文件为单位),而这里这个描述符则是用于描述网络访问的。什么意思呢?就是程序员可以通过socket来发送和接收网络上的数据。你也可以理解成是一个API。有了它,你就不用直接去操作网卡了,而是通过这个接口,这样就省了很多复杂的操作。
在C#中,MS为我们提供了 System.Net.Sockets 命名空间,里面包含了Socket类。

2.有了socket,那就可以用它来访问网络了

不过你不要高兴得太早,要想访问网络,还得有些基本的条件(和编程无关的我就不提了):a. 要确定本机的IP和端口,socket只有与某一IP和端口绑定,才能发挥强大的威力。b. 得有协议吧(否则谁认得你这发送到网络上的是什么呀)。想要解决复杂的,我们可以自己来定协议。但是这个就不在这篇里提了,我这里介绍两种大家最熟悉不过的协议:TCP & UDP。(别说你不知道,不然...不然...我不告诉你)
如果具备了基本的条件,就可以开始用它们访问网络了。来看看步骤吧:
a. 建立一个套接字
b. 绑定本机的IP和端口
c. 如果是TCP,因为是面向连接的,所以要利用Listen()方法来监听网络上是否有人给自己发东西;如果是UDP,因为是无连接的,所以来者不拒。
d. TCP情况下,如果监听到一个连接,就可以使用accept来接收这个连接,然后就可以利用Send/Receive来执行操作了。而UDP,则不需要accept, 直接使用SendTo/ReceiveFrom来执行操作。(看清楚哦,和TCP的执行方法有区别,因为UDP不需要建立连接,所以在发送前并不知道对方的IP和端口,因此需要指定一个发送的节点才能进行正常的发送和接收)
e. 如果你不想继续发送和接收了,就不要浪费资源了。能close的就close吧。

面向连接的套接字系统调用时序 (TCP)


无连接的套接字系统调用时序(UDP)

二、TCP协议的Socket实例

服务端 后台代码:

  1 using System;
  2 using System.Collections.Generic;
  3 using System.ComponentModel;
  4 using System.Data;
  5 using System.Drawing;
  6 using System.IO;
  7 using System.Linq;
  8 using System.Net;
  9 using System.Net.Sockets;
 10 using System.Text;
 11 using System.Threading;
 12 using System.Threading.Tasks;
 13 using System.Windows.Forms;
 14 
 15 namespace Socket通信
 16 {
 17     public partial class Form1 : Form
 18     {
 19 
 20         public Form1()
 21         {
 22             InitializeComponent();
 23             TextBox.CheckForIllegalCrossThreadCalls = false;
 24 
 25         }
 26         Socket socketSend;
 27         Thread threadWatch = null; // 负责监听客户端连接请求的 线程;
 28         Socket socketWatch = null;
 29 
 30 
 31         /// <summary>
 32         /// 监听客户端请求的方法;
 33         /// </summary>
 34         void WatchConnecting()
 35         {
 36             try
 37             {
 38                 while (true)  // 持续不断的监听客户端的连接请求;
 39                 {
 40                     // 开始监听客户端连接请求,Accept方法会阻断当前的线程;
 41                     Socket sokConnection = socketWatch.Accept(); 
 42                     // 一旦监听到一个客户端的请求,就返回一个与该客户端通信的 套接字;
 43                     // 向列表控件中添加客户端的IP信息;
 44                     lbOnline.Items.Add(sokConnection.RemoteEndPoint.ToString());
 45                     ShowMsg("客户端连接成功!");
 46                     //开启一个新线程,执行接收消息方法
 47                     Thread r_thread = new Thread(Received);
 48                     r_thread.IsBackground = true;
 49                     r_thread.Start(sokConnection);
 50                 }
 51             }
 52             catch (Exception e)
 53             {
 54                 ShowMsg("异常:" + e.Message);
 55             }
 56 
 57         }
 58         /// <summary>
 59         /// 服务器端不停的接收客户端发来的消息
 60         /// </summary>
 61         /// <param name="o"></param>
 62         void Received(object o)
 63         {
 64             try
 65             {
 66                  socketSend = o as Socket;
 67                 while (true)
 68                 {
 69                     //客户端连接服务器成功后,服务器接收客户端发送的消息
 70                     // 定义一个3M的缓存区;
 71                     byte[] buffer = new byte[1024 * 1024 * 3];
 72                     //实际接收到的有效字节数
 73                     // 将接受到的数据存入到输入  buffer中;
 74                     int len = socketSend.Receive(buffer);
 75                     if (len == 0)
 76                     {
 77                         break;
 78                     }
 79                     string str = Encoding.UTF8.GetString(buffer, 0, len);
 80                     ShowMsg("接收到的客户端数据:" + socketSend.RemoteEndPoint + ":" + str);
 81                     Send("服务端接收成功(" + str + ")");
 82 
 83                 }
 84             }
 85             catch (Exception e)
 86             {
 87                 ShowMsg("异常:" + e.Message);
 88             }
 89         }
 90         /// <summary>
 91         /// 服务器向客户端发送消息
 92         /// </summary>
 93         /// <param name="str"></param>
 94         void Send(string str)
 95         {
 96             byte[] buffer = Encoding.UTF8.GetBytes(str);
 97             socketSend.Send(buffer);
 98 
 99         }
100         void ShowMsg(string str)
101         {
102             txtMsg.AppendText(str + "\r\n");
103         }
104         private void button1_Click_1(object sender, EventArgs e)
105         {
106             // 创建负责监听的套接字,注意其中的参数;
107             socketWatch = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
108             // 获得文本框中的IP对象;
109             IPAddress address = IPAddress.Parse(txtIp.Text.Trim());
110             // 创建包含ip和端口号的网络节点对象;
111             IPEndPoint endPoint = new IPEndPoint(address, int.Parse(txtPort.Text.Trim()));
112             try
113             {
114                 // 将负责监听的套接字绑定到唯一的ip和端口上;
115                 socketWatch.Bind(endPoint);
116             }
117             catch (SocketException se)
118             {
119                 MessageBox.Show("异常:" + se.Message);
120                 return;
121             }
122             // 设置监听队列的长度;
123             socketWatch.Listen(10);
124             // 创建负责监听的线程;
125             threadWatch = new Thread(WatchConnecting);
126             threadWatch.IsBackground = true;
127             threadWatch.Start();
128             ShowMsg("服务器启动监听成功!");
129         }
130     }
131 }

客户端 后台代码:

 1 using System;
 2 using System.Collections.Generic;
 3 using System.ComponentModel;
 4 using System.Data;
 5 using System.Drawing;
 6 using System.Linq;
 7 using System.Text;
 8 using System.Threading.Tasks;
 9 using System.Windows.Forms;
10 using System.Net;
11 using System.Net.Sockets;
12 using System.Threading;
13 
14 
15 namespace Socket通信客户端
16 {
17     public partial class Socket_client : Form
18     {
19         public Socket_client()
20         {
21             InitializeComponent();
22             CheckForIllegalCrossThreadCalls = false;
23         }
24         Socket socketSend;
25         void ShowMsg(string str)
26         {
27             txtMsg.AppendText(str + "\r\n");
28         }
29         /// <summary>
30         /// 接收服务端返回的消息
31         /// </summary>
32         void Received()
33         {
34             while (true)
35             {
36                 try
37                 {
38                     byte[] buffer = new byte[1024 * 1024 * 3];
39                     //实际接收到的有效字节数
40                     int len = socketSend.Receive(buffer);
41                     if (len == 0)
42                     {
43                         continue;
44                     }
45                     string str = Encoding.UTF8.GetString(buffer, 0, len);
46                     ShowMsg("接收到的服务端数据:" + socketSend.RemoteEndPoint + ":" + str);
47                 }
48                 catch
49                 {
50                     MessageBox.Show("接收失败,请检查服务端是否断开!");
51                     return;
52                 }
53             }
54         }
55 
56         private void btnDisconnect_Click(object sender, EventArgs e)
57         {
58             socketSend.Close();
59             ShowMsg("连接已经断开!");
60         }
61         void Send(string str)
62         {
63             byte[] buffer = Encoding.UTF8.GetBytes(str);
64             socketSend.Send(buffer);
65         }
66     }
67 }

三、UDP协议的Socket实例

using System;
using System.Collections.Generic;
using System.Text;
using System.Net;
using System.Net.Sockets;
namespace SimpleUdpSrvr
{
    class Program
    {
        static void Main(string[] args)
        {
            int recv;
            byte[] data = new byte[1024];
            IPEndPoint ipep = new IPEndPoint(IPAddress.Any, 9050);//定义一网络端点
            Socket newsock = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp);//定义一个Socket
            newsock.Bind(ipep);//Socket与本地的一个终结点相关联
            Console.WriteLine("Waiting for a client..");

            IPEndPoint sender = new IPEndPoint(IPAddress.Any, 0);//定义要发送的计算机的地址
            EndPoint Remote = (EndPoint)(sender);//
            recv = newsock.ReceiveFrom(data, ref Remote);//接受数据           
            Console.WriteLine("Message received from{0}:", Remote.ToString());
            Console.WriteLine(Encoding.ASCII.GetBytes(data,0,recv));

            string welcome = "Welcome to my test server!";
            data = Encoding.ASCII.GetBytes(welcome);
            newsock.SendTo(data, data.Length, SocketFlags.None, Remote);
            while (true)
            {
                data = new byte[1024];
                recv = newsock.ReceiveFrom(data, ref Remote);
                Console.WriteLine(Encoding.ASCII.GetString(data, 0, recv));
                newsock.SendTo(data, recv, SocketFlags.None, Remote);
            }
        }
    }
}
using System;
using System.Collections.Generic;
using System.Text;
using System.Net;
using System.Net.Sockets;
namespace SimpleUdpClient
{
    class Program
    {
        static void Main(string[] args)
        {
            byte[] data = new byte[1024];//定义一个数组用来做数据的缓冲区
            string input, stringData;
            IPEndPoint ipep = new IPEndPoint(IPAddress.Parse("127.0.0.1"), 9050);
            Socket server = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp);
            string welcome = "Hello,are you there?";
            data = Encoding.ASCII.GetBytes(welcome);
            server.SendTo(data, data.Length, SocketFlags.None, ipep);//将数据发送到指定的终结点

            IPEndPoint sender = new IPEndPoint(IPAddress.Any, 0);
            EndPoint Remote = (EndPoint)sender;
            data = new byte[1024];
            int recv = server.ReceiveFrom(data, ref Remote);//接受来自服务器的数据

            Console.WriteLine("Message received from{0}:", Remote.ToString());
            Console.WriteLine(Encoding.ASCII.GetString(data, 0, recv));
            while (true)//读取数据
            {
                input = Console.ReadLine();//从键盘读取数据
                if (input == "text")//结束标记
                {
                    break;
                }
                server.SendTo(Encoding.ASCII.GetBytes(input), Remote);//将数据发送到指定的终结点Remote
                data = new byte[1024];
                recv = server.ReceiveFrom(data, ref Remote);//从Remote接受数据
                stringData = Encoding.ASCII.GetString(data, 0, recv);
                Console.WriteLine(stringData);
            }
            Console.WriteLine("Stopping client");
            server.Close();
        }
    }
}     

上面的示例只是简单的应用了socket来实现通信,你也可以实现异步socket、IP组播 等等。

MS还为我们提供了几个助手类:TcpClient类、TcpListener类、UDPClient类。这几个类简化了一些操作,所以你也可以利用这几类来写上面的代码,但我个人还是比较习惯直接用socket来写。
既然快写完了,那我就再多啰嗦几句。在需要即时响应的软件中,我个人更倾向使用UDP来实现通信,因为相比TCP来说,UDP占用更少的资源,且响应速度快,延时低。至于UDP的可靠性,则可以通过在应用层加以控制来满足。当然如果可靠性要求高的环境下,还是建议使用TCP。

相关推荐

悠悠万事,吃饭为大(悠悠万事吃饭为大,什么意思)

新媒体编辑:杜岷赵蕾初审:程秀娟审核:汤小俊审签:周星...

高铁扒门事件升级版!婚宴上‘冲喜’老人团:我们抢的是社会资源

凌晨两点改方案时,突然收到婚庆团队发来的视频——胶东某酒店宴会厅,三个穿大红棉袄的中年妇女跟敢死队似的往前冲,眼瞅着就要扑到新娘的高额钻石项链上。要不是门口小伙及时阻拦,这婚礼造型团队熬了三个月的方案...

微服务架构实战:商家管理后台与sso设计,SSO客户端设计

SSO客户端设计下面通过模块merchant-security对SSO客户端安全认证部分的实现进行封装,以便各个接入SSO的客户端应用进行引用。安全认证的项目管理配置SSO客户端安全认证的项目管理使...

还在为 Spring Boot 配置类加载机制困惑?一文为你彻底解惑

在当今微服务架构盛行、项目复杂度不断攀升的开发环境下,SpringBoot作为Java后端开发的主流框架,无疑是我们手中的得力武器。然而,当我们在享受其自动配置带来的便捷时,是否曾被配置类加载...

Seata源码—6.Seata AT模式的数据源代理二

大纲1.Seata的Resource资源接口源码2.Seata数据源连接池代理的实现源码3.Client向Server发起注册RM的源码4.Client向Server注册RM时的交互源码5.数据源连接...

30分钟了解K8S(30分钟了解微积分)

微服务演进方向o面向分布式设计(Distribution):容器、微服务、API驱动的开发;o面向配置设计(Configuration):一个镜像,多个环境配置;o面向韧性设计(Resista...

SpringBoot条件化配置(@Conditional)全面解析与实战指南

一、条件化配置基础概念1.1什么是条件化配置条件化配置是Spring框架提供的一种基于特定条件来决定是否注册Bean或加载配置的机制。在SpringBoot中,这一机制通过@Conditional...

一招解决所有依赖冲突(克服依赖)

背景介绍最近遇到了这样一个问题,我们有一个jar包common-tool,作为基础工具包,被各个项目在引用。突然某一天发现日志很多报错。一看是NoSuchMethodError,意思是Dis...

你读过Mybatis的源码?说说它用到了几种设计模式

学习设计模式时,很多人都有类似的困扰——明明概念背得滚瓜烂熟,一到写代码就完全想不起来怎么用。就像学了一堆游泳技巧,却从没下过水实践,很难真正掌握。其实理解一个知识点,就像看立体模型,单角度观察总...

golang对接阿里云私有Bucket上传图片、授权访问图片

1、为什么要设置私有bucket公共读写:互联网上任何用户都可以对该Bucket内的文件进行访问,并且向该Bucket写入数据。这有可能造成您数据的外泄以及费用激增,若被人恶意写入违法信息还可...

spring中的资源的加载(spring加载原理)

最近在网上看到有人问@ContextConfiguration("classpath:/bean.xml")中除了classpath这种还有其他的写法么,看他的意思是想从本地文件...

Android资源使用(android资源文件)

Android资源管理机制在Android的开发中,需要使用到各式各样的资源,这些资源往往是一些静态资源,比如位图,颜色,布局定义,用户界面使用到的字符串,动画等。这些资源统统放在项目的res/独立子...

如何深度理解mybatis?(如何深度理解康乐服务质量管理的5个维度)

深度自定义mybatis回顾mybatis的操作的核心步骤编写核心类SqlSessionFacotryBuild进行解析配置文件深度分析解析SqlSessionFacotryBuild干的核心工作编写...

@Autowired与@Resource原理知识点详解

springIOCAOP的不多做赘述了,说下IOC:SpringIOC解决的是对象管理和对象依赖的问题,IOC容器可以理解为一个对象工厂,我们都把该对象交给工厂,工厂管理这些对象的创建以及依赖关系...

java的redis连接工具篇(java redis client)

在Java里,有不少用于连接Redis的工具,下面为你介绍一些主流的工具及其特点:JedisJedis是Redis官方推荐的Java连接工具,它提供了全面的Redis命令支持,且...