3

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

我正在使用GetClipboardDataAPI 从剪贴板中获取一些数据,然后将内存中的数据作为字节数组复制。当您复制多个文件(因此是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课程。似乎没有任何相关性。

4

2 回答 2

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();

(顺便说一句:您的反序列化不起作用的原因是因为序列化 astring[]不只是将字符写为字节;它写出字符串数组的结构,包括 .NET 使用的其他内部位,如长度,和带有类型信息的二进制标头。您从剪贴板返回的内容不存在,因此无法反序列化。)

于 2012-07-25T19:33:54.093 回答
1

这个怎么样:

var strings = Encoding.Unicode
    .GetString(buffer)
    .Split(new[] { '\0' }, StringSplitOptions.RemoveEmptyEntries);
于 2012-07-25T19:55:52.100 回答