繁体   English   中英

指定的参数超出有效值范围。 参数名称:大小和串口通讯

[英]Specified argument was out of the range of valid values. Parameter name: size & Serial Port Communication

我需要创建一个应用程序,该应用程序需要使用TCP / IP与现有软件进行通信,而我的和其他应用程序都将使用下面指定的端口号。

private void frmScan_Load(object sender, EventArgs e)
{
    clientSocket.Connect("100.100.100.30", 76545);
}

public void msg(string mesg)
{
    textBox1.Text = textBox1.Text + Environment.NewLine + " >> " + mesg;
}

private void cmdSCANok_Click(object sender, EventArgs e)
{

    msg("Client Started");
    NetworkStream serverStream = clientSocket.GetStream();
    byte[] outStream = Encoding.ASCII.GetBytes("PCK|SCAN|5025066840471");

    serverStream.Write(outStream, 0, outStream.Length);
    serverStream.Flush();

    byte[] inStream = new byte[10025];
    serverStream.Read(inStream, 0, (int)clientSocket.ReceiveBufferSize);
    string returndata = Encoding.ASCII.GetString(inStream, 0, inStream.Length);


    msg("Data from Server : " + returndata);
} 

发生的是,我正在与之通信的程序具有某种内置语言,它将可以理解我发送的代码,并根据我发送的代码返回数据。 因此,在上面的代码中,我发送了三位代码:( "PCK|SCAN|5025066840471" ),它将在数据库中找到特定的项目。 当它运行时,我在网上得到一个错误:

serverStream.Read(inStream, 0, (int)clientSocket.ReceiveBufferSize);

错误显示如下:“指定的参数超出有效值范围。参数名称:大小”

我遵循了在该网站上看到的教程: http : //csharp.net-informations.com/communications/csharp-client-socket.htm-但是我做的略有不同。 所以不要放

string returndata = Encoding.ASCII.GetString(inStream);

我写:

string returndata = Encoding.ASCII.GetString(inStream, 0, inStream.Length); 

我对为什么会遇到这些问题感到非常困惑,并且说实话,我对代码在做什么并不了解,我只是有一个大概的主意,但不足以解决此问题。 有人可以帮忙吗?

非常感激!

PS:我正在Visual Studio 2010上为Windows CE(便携式设备)编程。

您的代码很好地说明了如何不进行TCP通信。 我已经看过很多遍此代码了,很高兴为您指出一个关于TCP的好教程-太糟糕了,我还没看过:)

首先让我指出一些错误:

  • TCP不能保证数据包以一束字节的形式到达。 因此(理论上), Write操作可能会导致拆分,从而需要在另一侧进行两次读取。 通过TCP发送不带标头的数据是一个非常糟糕的主意-接收方不知道必须读取多少数据。 因此,您有两个选择-要么在数据本身之前写入整束数据的长度,要么使用控制字符结束“数据包”
  • 第一点还应该阐明您的阅读也是错误的。 读取整个“命令”可能需要多个读取操作,或者一次读取操作可能一次给您两个命令!
  • 您正在将ReceiveBufferSize字节读取到10025长的缓冲区中。 ReceiveBufferSize可能大于缓冲区。 不要那样做-读取inStream.Length的最大数量。 如果您使用C ++进行编码,这将是缓冲区溢出的一个很好的例子。
  • 当您将数据转换为字符串时,您期望整个缓冲区已满。 事实并非如此。 相反,您必须存储read调用的返回值-它告诉您实际读取了多少字节。 否则,您正在读取垃圾,并且基本上有另一个缓冲区溢出。

因此,更好的实现(尽管还远远不够完美)将是这样的:

NetworkStream serverStream = clientSocket.GetStream();
byte[] outStream = Encoding.ASCII.GetBytes("PCK|SCAN|5025066840471");

// It would be much nicer to send a terminator or data length first, 
// but if your server doesn't expect that, you're out of luck.
serverStream.Write(outStream, 0, outStream.Length);

// When using magic numbers, at least use nice ones :)
byte[] inStream = new byte[4096];

// This will read at most inStream.Length bytes - it can be less, and it
// doesn't tell us how much data there is left for reading.
int bytesRead = serverStream.Read(inStream, 0, inStream.Length);

// Only convert bytesRead bytes - the rest is garbage
string returndata = Encoding.ASCII.GetString(inStream, 0, bytesRead);

哦,我必须推荐这篇有关TCP协议设计的文章

它讨论了有关TCP的许多误解,最重要的是请参阅消息框架部分。

NetworkStream.Read方法内部进行以下检查:

if(size < 0 || size > (buffer.Length - offset))
   throw new ArgumentOutOfRanveException("size");

在您的情况下:

size = clientSocket.ReceiveBufferSize

offset = 0

buffer = inStream

您收到的错误意味着clientSocket.ReceiveBufferSize> inStream.Length 换句话说,您尝试读取的字节数多于可用字节数。 尝试使用以下代码:

...
var count = serverStream.Read(inStream, 0, inStream.Length);
string returndata = Encoding.ASCII.GetString(inStream, 0, count);

还看到一个例子在这里

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM