首先澄清一些事情。我不尝试将字节数组转换为单个字符串。我正在尝试将字节数组转换为字符串数组。
我使用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
课程。似乎没什么关系。
答案 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);