网络通信中,经常会有收发数据的缓冲区内没有填充满,字节数组的后半段全是空白\0 符号.
如果收发的是可字符编码的数据,可以对内容进行转字符串,然后去除末尾的\0符号即可,可以直接用字符串进行后续处理.
例如:
string recv = Encoding.Default.GetString(bytes);
recv = recv.TrimEnd('\0');
如果收到的不是可字符编码的数据,而是需要特定的方式解码的数据,由于字符集编码不能覆盖全部所有,
将字节数组转成某些超出字符编码范围的不可读取的字符串,用字符串处理方式去除末尾\0,再从字符串转回去的时候,不可避免的会产生数据的丢失,所以很多时候需要直接处理byte数组,而不是转成字符串后去处理,不过BitConverter是一个例外,放最后面再讲.
C# byte数组去除尾部0x00空白区域的方式是很多的,
不去除前面和中间的00,是因为在网络通信中那些有可能是数据包含的,移除后会破坏数据.
而末尾的00是由两部分组成的,一是缓冲区未填满的区域,二是字符串结束符\0.
代码如下:
/// <summary>
/// 去除byte[]数组缓冲区内的尾部空白区;从末尾向前判断;
/// </summary>
/// <param name="bytes"></param>
/// <returns></returns>
public byte[] bytesTrimEnd(byte[] bytes)
{
List<byte> list = bytes.ToList();
for (int i = bytes.Length - 1; i >= 0; i--)
{
if(bytes[i]==0x00)
{
list.RemoveAt(i);
}
else
{
break;
}
}
return list.ToArray();
}
另外还有其它方法,比如另一个博主的代码:
//原文:https://blog.csdn.net/yuwen1658/article/details/103855259
//在网上找了半天 C# byte数组尾部0x00,去除方法,没找到,自己写了一个,测试使用没问题,发了上来
public byte[] CutTail(byte[] byList)
{
int j = 0;
byte[] tempb = null;
for (int i = byList.Length-1; i >= 0; i--)
{
if (byList[i] != 0x00&j==0)
{
j = i;
if (tempb == null)
{
tempb = new byte[j + 1];
}
tempb[j] = byList[i];
j--;
}
else
{
if (tempb != null)
{
tempb[j] = byList[i];
j--;
}
}
}
return tempb;
}
最后提一下特殊的BitConverter静态类.
BitConverter.ToString 将指定的字节数组的每个元素的数值转换为其等效的十六进制字符串表示形式。
BitConverter.ToString生成了字符串,但是字符串的内容是16进制码,不会产生任何数据损失.
转换后的字符串是以-分隔的,如图下:
去除空白的方法也简单:
recvMsg=BitConverter.ToString(bytes);
recvMsg = recvMsg.TrimEnd('-','0','0');
最终得到
BitConverter.ToString的处理方式与如下将字节数据转字符串的方式类似,只是中间多了一个分隔符而已.
for (int i = 0; i < bytes.Length; i++)
{
str+= bytes[i].ToString("X2");
}
十六进制字符串转回字节数组的方式,是循环每个以'-'分隔的字符串元素s,且声明是16进制字符串转过去.
buffer[i] = Convert.ToByte(s,16);
方式多种多样,可以是List<byte> 添加完最后ToArray() . 也可以直接是byte[]数组,长度就是Split('-')的长度.
就这么多吧,不续了.