- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我刚刚开始通过各种 Google 搜索学习套接字,但我在弄清楚如何在 C# 中正确使用套接字时遇到了一些问题,我需要一些帮助。
我有一个测试应用程序(Windows 窗体)和一个不同的类(实际上在它自己的 .dll 中,但这无关紧要)我有我的套接字代码的所有服务器/客户端代码。
问题 1)
在我的测试应用程序中,在服务器部分,用户可以单击“开始监听”按钮,我的套接字应用程序的服务器部分应该开始监听指定地址和端口上的连接,到目前为止一切顺利。
但是,应用程序将被阻止,在有人连接到服务器之前我无法做任何事情。如果没有人连接怎么办?我该如何处理?我可以指定一个接收超时,但那又如何呢?它抛出一个异常,我能用它做什么?我想要的是在主应用程序上进行某种事件,以便用户知道应用程序没有卡住并且正在等待连接。但是,如果没有建立连接,它应该会超时并关闭所有内容。
也许我应该使用异步调用来发送/接收方法,但它们看起来很困惑,我无法让它工作,只能同步工作(我将在下面发布我当前的代码)。
问题 2)
当某些发送/接收调用超时时,我是否需要关闭任何东西。正如您将在我当前的代码中看到的那样,我在套接字上有一堆关闭,但不知何故这感觉不对。但当操作超时并且我没有关闭套接字时,感觉也不对。
作为我的两个问题的结论……我想要一个不会阻塞的应用程序,这样用户就知道服务器正在等待连接(例如带有一个小选取框动画)。如果一段时间后还没有建立连接,我想关闭所有应该关闭的东西。当建立连接或一段时间后没有建立连接时,我想将结果通知主应用程序。
这是我的一些代码,其余类似。 Packet
类是一个自定义类,表示我的自定义数据单元,目前它只是一堆基于 enums
的属性,具有将它们转换为字节并返回到字节的方法属性。
开始监听连接的函数是这样的:
public void StartListening(string address, int port) {
try {
byte[] bufferBytes = new byte[32];
if(address.Equals("0.0.0.0")) {
udpSocket.Bind(new IPEndPoint(IPAddress.Any, port));
} else {
udpSocket.Bind(new IPEndPoint(IPAddress.Parse(address), port));
}
remoteEndPoint = new IPEndPoint(IPAddress.Any, 0);
int numBytesReceived = udpSocket.ReceiveFrom(bufferBytes, ref remoteEndPoint);
if(numBytesReceived == 0) {
udpSocket.Close();
return;
}
Packet syncPacket = new Packet(bufferBytes);
if(syncPacket.PacketType != PacketType.Control) {
udpSocket.Close();
return;
}
} catch {
if(udpSocket != null) {
udpSocket.Close();
}
}
}
我确信我有一堆不必要的代码,但我是新手,我不确定该怎么做,非常感谢任何修复我的代码以及如何解决上述问题的帮助。
编辑:
我可能应该说明我的要求是使用 UDP 并在应用层自己实现这些东西。您可以将其视为家庭作业,但我没有这样标记,因为代码无关紧要,不会成为我成绩的一部分,而且我的问题(我的问题)是“如何编码”,因为我的 Sockets 经验很少而且不是受教了。
但是我必须说我现在解决了我的问题我想...我在演示应用程序上使用线程,这给我带来了一些问题,现在我在协议(protocol)连接中使用它,更有意义而且我可以轻松更改我的自定义协议(protocol)类属性并从演示应用程序中读取这些属性。
我已经指定了一个超时并在达到超时时抛出一个 SocketException。每当捕获到这样的异常时,套接字连接就会关闭。我只是在谈论连接握手,仅此而已。如果没有捕获到异常,则代码可能运行顺利并建立了连接。
请相应地调整您的答案。现在将其中任何一个标记为已接受的答案对我来说都没有意义,希望你能理解。
最佳答案
你搞错了。
首先,UDP是无连接的。您不连接或断开连接。您所做的就是发送和接收(每次都必须指定目的地)。您还应该知道,UDP 唯一 promise 的是每次读取都会收到完整的消息。 UDP 不保证您的消息以正确的顺序到达或完全到达。
另一方面,TCP 是基于连接的。您连接、发送/接收并最终断开连接。 TCP 是基于流的(而 UDP 是基于消息的),这意味着您可以在第一次读取时获得一半消息,在第二次读取时获得另一半消息。 TCP 向您保证一切都会以正确的顺序到达(或者会死于尝试;)。因此,使用 TCP 意味着您应该有某种逻辑来了解完整消息何时到达以及用于构建完整消息的缓冲区。
下一个大问题是关于阻塞。由于您是新手,我建议您使用线程来处理套接字。将监听器套接字放在一个线程中,将每个连接套接字放在一个单独的线程中(5 个连接的客户端 = 5 个线程)。
我还建议您使用 TCP,因为构建完整的消息比排序消息和构建交易系统(如果您想确保所有消息到达/来自客户端则需要)更容易。
你还是弄错了UDP。 Close 除了清理系统资源外不做任何事情。你应该做这样的事情:
public void MySimpleServer(string address, int port)
{
try
{
byte[] bufferBytes = new byte[32];
if(address.Equals("0.0.0.0")) {
udpSocket.Bind(new IPEndPoint(IPAddress.Any, port));
} else {
udpSocket.Bind(new IPEndPoint(IPAddress.Parse(address), port));
}
remoteEndPoint = new IPEndPoint(IPAddress.Any, 0);
while (serverCanRun)
{
int numBytesReceived = udpSocket.ReceiveFrom(bufferBytes, ref remoteEndPoint);
// just means that one of the clients closed the connection using Shutdown.
// doesnt mean that we cant continue to receive.
if(numBytesReceived == 0)
continue;
// same here, loop to receive from another client.
Packet syncPacket = new Packet(bufferBytes);
if (syncPacket.PacketType != PacketType.Control)
continue;
HandlePacket(packet, endPoint);
}
} catch {
if(udpSocket != null) {
udpSocket.Close();
}
}
}
看到了吗?因为没有连接,所以关闭一个 UDP 套接字以开始从另一个套接字开始监听只是浪费时间。同一个套接字可以从所有知道正确端口和地址的 udp 客户端接收。这就是 remoteEndPoint
的用途。它告诉发送消息的客户端。
小更新以总结我的所有评论。
UDP 是无连接的。您永远无法检测到连接是建立还是断开。 UDP 套接字上的 Close 方法只会释放系统资源。对 client.Close()
的调用不会通知服务器套接字(与 TCP 一样)。
检查连接是否打开的最佳方法是创建 ping/pong 样式的数据包。即客户端发送 PING 消息,服务器以 PONG 响应。请记住,如果消息未到达,UDP 将不会尝试重新发送消息。因此,在假设服务器已关闭(如果您没有收到 PONG)之前,您需要重新发送几次 PING。
至于客户端关闭,您需要向服务器发送您自己的消息,告诉它客户端将停止与服务器对话。为了可靠性,同样的事情发生在这里,继续重新发送 BYE 消息,直到您收到回复。
恕我直言,如果您想要可靠性,则必须为 UDP 实现事务系统。 SIP (google rfc3261) 是使用 UDP 交易的协议(protocol)示例。
关于c# - 对 C# 中使用 UDP 协议(protocol)的套接字感到困惑,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5415171/
我正在维护一些 Java 代码,我目前正在将它们转换为 C#。 Java 代码是这样做的: sendString(somedata + '\000'); 在 C# 中,我正在尝试做同样的事情: sen
如何确定函数中传递的参数是字符串还是字符(不确定如何正确调用它)文字? 我的函数(不正确): void check(const char* str) { // some code here }
我真的不知道如何准确地提出这个问题,但我希望标题已经说明了这一点。 我正在寻找一种方法(一个框架/库),它提供了执行 String.contains() 函数的能力,该函数告诉我给定的字符串是否与搜索
我正在尝试编写一些读取 Lambda 表达式并输出 beta 缩减版本的东西。 Lambda 的类型如下:\variable -> expression,应用程序的形式为 (表达式) (表达式)。因此
StackOverflow 上的第 1 篇文章,如果我没能把它做好,我深表歉意。我陷入了一个愚蠢的练习,我需要制作一个“刽子手游戏”,我尝试从“.txt”文件中读取单词,然后我得到了我的加密函数,它将
我想在 Groovy 中测试我的 Java 自定义注释,但由于字符问题而未能成功。 Groovyc: Expected 'a' to be an inline constant of type cha
当我尝试在单击按钮期间运行 javascript location.href 时,出现以下错误“字 rune 字中的字符过多”。 最佳答案 这应该使用 OnClientClick相反? 您可能还想停
我想要类似的东西: let a = ["v".utf8[0], 1, 2] 我想到的最接近的是: let a = [0x76, 1, 2] 和 "v".data(using: String.Encod
有没有办法在 MySQL 中指定 Unicode 字 rune 字? 我想用 Ascii 字符替换 Unicode 字符,如下所示: Update MyTbl Set MyFld = Replace(
阅读 PNG 规范后,我有点惊讶。我读过字 rune 字应该用像 0x41 这样的二进制值进行硬编码,而不是在(程序员友好的)'A' 中。问题似乎是在具有不同底层字符集的不同系统上编译期间字 rune
考虑一个具有 UTF-8 执行字符集的 C++11 编译器(并且符合要求 char 类型为有符号 8 位字节的 x86-64 ABI) . 字母 Ä(元音变音)具有 0xC4 的 unicode 代码
为什么即使有 UTF-8 字符串文字,C11 或 C++11 中也没有 UTF-8 字 rune 字?我知道,一般来说,字 rune 字表示单个 ASCII 字符,它与单字节 UTF-8 代码点相同,
我怎样才能用 Jade 做到这一点? how would I do this 我几乎可以做任何事情,除了引入一个 span 中间句子。 最佳答案 h3.blur. how would I do t
这似乎是一个非常简单的问题,但我只是想澄清我的疑问。我正在查看其他开发人员编写的代码。有一些涉及 float 的计算。 示例:Float fNotAvlbl = new Float(-99); 他为什
我想知道第 3 行“if dec:”中的“dec”是什么意思 1 def dec2bin(dec): 2 result='' 3 if dec:
我试图在字符串中查找不包含任何“a”字符的单词。我写了下面的代码,但它不起作用。我怎么能对正则表达式说“不包括”?我不能用“^”符号表示“不是”吗? import re string2 = "asfd
这个问题在这里已经有了答案: Is floating point math broken? (31 个答案) Is floating point arbitrary precision availa
我正在创建一个时尚的文本应用程序,但在某些地方出现错误(“字 rune 字中的字符太多”)。我只写了一个字母,但是当我粘贴它时,它会转换成许多这样的字母:“\uD83C\uDD89”,原始字母是“🆉
我正在尝试检查用户是否在文本框中输入了一个数字值,是否接受了小数位。非常感谢任何帮助。 Private Sub textbox1_AfterUpdate() If IsNumeric(textbox1
我知道一个 Byte 是 8 位,但其他的代表什么?我正在参加一个使用摩托罗拉 68k 架构的汇编类(class),我对目前的词汇感到困惑。 最佳答案 如 operator's manual for
我是一名优秀的程序员,十分优秀!