如何将字节数组转换为字符串数组?

时间:2012-07-25 19:07:26

标签: c# clipboard

首先澄清一些事情。我尝试将字节数组转换为单个字符串。我正在尝试将字节数组转换为字符串数组。

我使用GetClipboardData API从剪贴板中获取一些数据,然后我将数据作为字节数组从内存中复制。当您复制多个文件(因此为CF_HDROP剪贴板格式)时,我想将此字节数组转换为复制文件的字符串数组。

到目前为止,这是我的代码。

//Get pointer to clipboard data in the selected format
var clipboardDataPointer = GetClipboardData(format);

//Do a bunch of crap necessary to copy the data from the memory
//the above pointer points at to a place we can access it.
var length = GlobalSize(clipboardDataPointer);
var @lock = GlobalLock(clipboardDataPointer);

//Init a buffer which will contain the clipboard data
var buffer = new byte[(int)length];

//Copy clipboard data to buffer
Marshal.Copy(@lock, buffer, 0, (int)length);

GlobalUnlock(clipboardDataPointer);

snapshot.InsertData(format, buffer);

现在,这是我之后读取缓冲区数据的代码。

var formatter = new BinaryFormatter();
using (var serializedData = new MemoryStream(buffer))
{
    paths = (string[]) formatter.Deserialize(serializedData);
}

这不起作用,它会崩溃,并说明该流不包含二进制头。我想这是因为它不知道要反序列化的类型。

我试过通过查看Marshal课程。似乎没什么关系。

2 个答案:

答案 0 :(得分:2)

如果数据来自Win32 API,那么字符串数组将只是一个以空字符结尾的字符串序列,末尾有一个双重空终止符。 (请注意,字符串将是UTF-16,因此每个字符两个字节)。你基本上需要将一个字符串一次拉出一个数组。

您在这里寻找的方法是Marshal.PtrToStringUni,您应该使用该方法代替Marshal.Copy,因为它适用于IntPtr。它将从IntPtr中提取一个字符串,直到第一个空字符,然后将其复制到字符串中。

这个想法是不断提取单个字符串,然后将IntPtr超过空字节前进到下一个字符串的开头,直到用完缓冲区为止。我已经没有对此进行了测试,并且它可能会得到改进(特别是我认为有一种更智能的方法来检测缓冲区的结束)但基本的想法是:

var myptr = GetClipboardData(format);
var length = GlobalSize(myptr);

var result = new List<string>();

var pos = 0;
while ( pos < length )
{
    var str = Marshal.PtrToStringUni(myptr);
    var count = Encoding.Unicode.GetByteCount(str);

    myptr = IntPtr.Add(myptr, count + 1);
    pos += count + 1;

    result.Add(str);
}

return result.ToArray();

(顺便说一下:你的反序列化不起作用的原因是因为序列化string[]并不只是将字符写成字节;它会写出字符串数组的结构,包括.NET使用的其他内部位,如长度,以及带有类型信息的二进制头。你从剪贴板中获取的内容都不存在,因此无法反序列化。)

答案 1 :(得分:1)

这个怎么样:

var strings = Encoding.Unicode
    .GetString(buffer)
    .Split(new[] { '\0' }, StringSplitOptions.RemoveEmptyEntries);