- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我有一个监听器套接字,它像 TCP 服务器通常那样接受、接收和发送。我在下面给出了我的接受和接收代码,它与 example on Microsoft's documentation 没有什么不同。 .主要区别是我的服务器在停止接收数据后不会终止连接(我不知道这是否是一个糟糕的设计?)。
private void on_accept(IAsyncResult xResult)
{
Socket listener = null;
Socket handler = null;
TStateObject state = null;
Task<int> consumer = null;
try
{
mxResetEvent.Set();
listener = (Socket)xResult.AsyncState;
handler = listener.EndAccept(xResult);
state = new TStateObject()
{
Socket = handler
};
consumer = async_input_consumer(state);
OnConnect?.Invoke(this, handler);
handler.BeginReceive(state.Buffer, 0, TStateObject.BufferSize, 0, new AsyncCallback(on_receive), state);
}
catch (SocketException se)
{
if (se.ErrorCode == 10054)
{
on_disconnect(state);
}
}
catch (ObjectDisposedException)
{
return;
}
catch (Exception ex)
{
System.Console.WriteLine("Exception in TCPServer::AcceptCallback, exception: " + ex.Message);
}
}
private void on_receive(IAsyncResult xResult)
{
Socket handler = null;
TStateObject state = null;
try
{
state = xResult.AsyncState as TStateObject;
handler = state.Socket;
int bytesRead = handler.EndReceive(xResult);
UInt16 id = TClientRegistry.GetIdBySocket(handler);
TContext context = TClientRegistry.GetContext(id);
if (bytesRead > 0)
{
var buffer_data = new byte[bytesRead];
Array.Copy(state.Buffer, buffer_data, bytesRead);
state.BufferBlock.Post(buffer_data);
}
Array.Clear(state.Buffer, 0, state.Buffer.Length);
handler.BeginReceive(state.Buffer, 0, TStateObject.BufferSize, 0, new AsyncCallback(on_receive), state);
}
catch (SocketException se)
{
if(se.ErrorCode == 10054)
{
on_disconnect(state);
}
}
catch (ObjectDisposedException)
{
return;
}
catch (Exception ex)
{
System.Console.WriteLine("Exception in TCPServer::ReadCallback, exception: " + ex.Message);
}
}
此代码用于连接到嵌入式设备并且(大部分)工作正常。我正在调查内存泄漏,并试图通过准确复制设备的功能来加快这个过程(我们的设备连接速度在大约 70kbps 的范围内,并且花了整个周末的压力测试来获取内存泄漏使服务器的内存占用量增加一倍)。
所以我写了一个 C# 程序来复制数据事务,但是我遇到了一个问题,当我断开测试程序时,服务器陷入了一个循环,它无休止地有它的 on_receive
调用回调。我的印象是 BeginReceive
在收到某些东西之前不会被触发,它似乎调用 on_receive
,像异步回调一样结束接收,处理数据,然后我希望连接等待更多数据,所以我再次调用 BeginReceive
。
我的测试程序出现问题的部分在这里:
private static void read_write_test()
{
mxConnection = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
mxConnection.Connect("12.12.12.18", 10);
if (mxConnection.Connected)
{
byte[] data = Encoding.ASCII.GetBytes("HANDSHAKESTRING"); //Connect string
int len = data.Length;
mxConnection.Send(data);
data = new byte[4];
len = mxConnection.Receive(data);
if (len == 0 || data[0] != '1')
{
mxConnection.Disconnect(false);
return;
}
}
//Meat of the test goes here but isn't relevant
mxConnection.Shutdown(SocketShutdown.Both);
mxConnection.Close();
}
直到 Shutdown(SocketShutdown.Both)
调用,一切都按预期工作。但是,当我进行该调用时,服务器似乎永远不会收到客户端已关闭套接字并陷入无休止尝试接收循环的通知。我已经完成作业,我认为我正在按照 this discussion 正确关闭我的连接.我已经弄乱了断开连接部分,只执行 mxConnection.Disconnect(false)
,但同样的事情发生了。
当设备与服务器断开连接时,我的服务器捕获到一个错误代码为 10054 的 SocketException
,该文档说:
Connection reset by peer.
An existing connection was forcibly closedby the remote host. This normally results if the peer application onthe remote host is suddenly stopped, the host is rebooted, the host orremote network interface is disabled, or the remote host uses a hardclose (see setsockopt for more information on the SO_LINGER option onthe remote socket). This error may also result if a connection wasbroken due to keep-alive activity detecting a failure while one ormore operations are in progress. Operations that were in progress failwith WSAENETRESET. Subsequent operations fail with WSAECONNRESET.
我用它来处理正在关闭的套接字,并且在大多数情况下都运行良好。但是,对于我的 C# 测试程序,它的工作方式似乎不同。
我是不是漏掉了什么?我很感激任何意见。谢谢。
最佳答案
The main difference is that my server doesn't kill a connection after it stops receiving data (I don't know if this is a bad design or not?).
当然是。
it seems like the server never gets notification that the client has closed the socket and gets stuck in a loop of endlessly trying to receive
服务器确实收到通知。只是你忽略了它。通知是您的接收操作返回 0
。当发生这种情况时,您只需再次调用 BeginReceive()
。这将启动一个新的读取操作。哪个……返回 0
!你只是一遍又一遍地这样做。
当接收操作返回 0
时,您应该完成正常关闭(调用 Shutdown()
和 Close()
) 远程端点启动。不要尝试再次接收。您只会不断得到相同的结果。
我强烈建议你多做功课。一个好的起点是 Winsock Programmer's FAQ .这是一个相当古老的资源,根本不涉及 .NET。但在大多数情况下,新手网络程序员在 .NET 中犯的错误与 20 年前新手 Winsock 程序员犯的错误是一样的。该文件在今天仍然和当时一样重要。
顺便说一句,您的客户端代码也有一些问题。首先,当 Connect()
方法成功返回时,套接字已连接。您不必检查 Connected
属性(事实上,永远 不应该检查该属性)。其次,Disconnect()
方法没有做任何有用的事情。当您想重新使用底层套接字句柄时使用它,但您应该在此处处理 Socket
对象。按照常用的套接字 API 习惯用法,只需使用 Shutdown()
和 Close()
。第三,任何从 TCP 套接字接收的代码必须在循环中执行此操作,并利用接收到的字节计数值来确定已读取哪些数据以及是否已读取足够的数据来执行任何操作有用。 TCP 可以在成功读取时返回任意正数的字节,并且您的程序的工作是识别已发送的任何特定数据 block 的开始和结束。
关于客户端关闭套接字后,C# tcp 异步监听器卡在我的 on_receive 回调中,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46795559/
当单击复选框(或选择所有复选框)时,我想向 CheckboxSelectionModel 添加一个监听器。 var smSensors = new xg.CheckboxSelectionModel(
我有一个简单的程序,允许 2 个客户端连接到服务器。 连接后,他们可以轮流点击空白卡片图像。 一旦 2 个客户中的任何一个点击空白卡片图片,卡片图片将变为 Ace 俱乐部图片。 更改将显示在客户端的两
我在这里看到了一个代码,该代码以字符串的形式检索鼠标的当前图标,但是此代码使用了TTimer来实现。 因此,我想知道是否存在某些事件(侦听器)来检测鼠标光标图标上的这些更改。 下面是使用TTimer的
我想在我的配置对象上获得一个 onload 事件。 以下工作,除非我创建一个 config.listeners={..} (我认为这就是我需要的?)替换 this.onload({...}); 我什至
通常,在 Java 中,当我有一个向其他对象提供某种通知的对象时,我将使用 Listener/Observer 模式。 有没有更类似于 Scala 的方式来做到这一点?我应该在 Scala 中使用这种
我有一个带有动画器的游戏对象和一些可以触发事件的动画(具有特定的回调函数)。 只要我将脚本添加到与动画器相同的游戏对象(包括接收器),一切都会正常工作: public class AnimatorEv
我有一个带有监听器的 DialogFragment,用于单击按钮以调用 fragment 中的函数。 我收到 lateinit property listener has not been initi
这个问题已经有答案了: Java ActionListener error: incompatible types (4 个回答) 已关闭 5 年前。 我最近刚刚开始学习 Java 代码。我在添加监听
我的代码遇到问题。我想知道是否有一种更简单的方法来使用监听器,而不是不断地这样做: example.addActionListener(new java.awt.event.ActionListene
有没有办法使用 .net 创建控制台应用程序。或通过某个端口监听 SMTP 消息的服务? 我需要创建一个中间层对象来捕获和处理 smtp 消息。也就是说,我希望该监听器发送和接收 smtp 消息,然后
我有一个带有动画器的游戏对象和一些可以触发事件的动画(具有特定的回调函数)。 只要我将脚本添加到与动画器相同的游戏对象(包括接收器),一切都会正常工作: public class AnimatorEv
我有许多向主事件生成服务注册的监听器。然而,我想告诉听众,事件流在某个时刻将会结束。您会通过简单地调用监听器上的方法(例如 finish())来完成此操作,还是有一个单独的事件方法 streamFin
我的代码有什么问题。 我创建了一个 JList,添加了项目并将其推到左侧(BorderLayout.WEST)。每次单击列表项时,我希望在列表右侧显示一个面板。但问题是,当选择列表项并运行监听器时,到
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 8 年前。 Improve th
这可能是一个简单的问题,但我没有看到它。 我有一个界面 public interface RenderableListener{ public void update(T element);
有人可以直接指出我的正确方向吗?当从组合框中选择适当的选项时,我希望小程序中的 Action 监听器显示从 html 文件检索的 jlabel 中的 3 个参数之一。 干杯 最佳答案 对于组合框,您需
我有一个网站,每个页面上都有许多 jQuery 事件处理程序,所有这些都在一个大型 .js 文件中。 这意味着对于任何页面,大多数事件处理程序都是针对不存在且根本不会使用的 HTML。 这会影响我的表
我有一些 jQuery 监听器设置,用于监听 type="text" 字段上的表单输入。但是,当用户从自动完成下拉框中选择一个选项(即他们之前输入的值已被记住以供将来使用)时,下面的监听器不会收集该值
关闭。这个问题需要多问focused 。目前不接受答案。 想要改进此问题吗?更新问题,使其仅关注一个问题 editing this post . 已关闭 9 年前。 Improve this ques
我正在用 unity (c#) 做一个类似国际象棋的游戏,但我在尝试进行向上转换以将信息从一个 child 发送到另一个抽象类时遇到了困难。 基本上,我有一个抽象类,它有一个带有函数的事件/委托(de
我是一名优秀的程序员,十分优秀!