6

我有一个项目使用内存映射文件让两个应用程序相互共享数据。生产者应用程序是用 C# 编写的,消费者应用程序使用普通的旧 C。两者都使用 VS2010。

MSDN 说“BinaryWriter.Write Method(String)”在数据前面加上一个 UTF-7 编码的无符号整数,然后写入有效负载。这正是我卡住的地方。如果我写一个长度为 256 个字符的字符串,C 应用程序的调试器会显示这个字节序列:0x80 0x2 <256 倍有效负载字符>。将长度前缀转换为我可以在消费者应用程序中安全使用的东西的最佳方法是什么?

生产者应用程序:

using System;
using System.IO;
using System.IO.MemoryMappedFiles;
using System.Threading;
using System.Text;
using System.Linq;

class Program
{
    static void Main(string[] args)
    {
        using (MemoryMappedFile mmf_read = MemoryMappedFile.CreateNew("mappedview", 4096))
        {
            using (MemoryMappedViewStream stream = mmf_read.CreateViewStream())
            {
                string str;
                BinaryWriter writer = new BinaryWriter(stream);

                str = string.Join("", Enumerable.Repeat("x", 256));

                writer.Write(str);
            }
        }
    }
}

消费者应用:

#include <windows.h>
#include <stdio.h>
#include <conio.h>
#include <tchar.h>
#pragma comment(lib, "user32.lib")

#define BUF_SIZE 4096
TCHAR szName[]=TEXT("Global\\mappedview");


int _tmain()
{
    HANDLE hMapFile;
    LPCSTR pBuf;

    hMapFile = OpenFileMapping(
               FILE_MAP_ALL_ACCESS,         // read/write access
               FALSE,                       // do not inherit the name
               szName);                     // name of mapping object

    if (hMapFile == NULL)
    {
        _tprintf(TEXT("Could not open file mapping object (%d).\n"),
         GetLastError());
        return 1;
    }

    pBuf = (LPCSTR) MapViewOfFile(hMapFile,     // handle to map object
           FILE_MAP_ALL_ACCESS,             // read/write permission
           0,
           0,
           BUF_SIZE);

    if (pBuf == NULL)
    {
        _tprintf(TEXT("Could not map view of file (%d).\n"),
                GetLastError());

        CloseHandle(hMapFile);
        return 1;
    }

    printf("Proc1: %s\n\n", pBuf);              // print mapped data

    UnmapViewOfFile(pBuf);

    CloseHandle(hMapFile);

    return 0;
}

兄弟,克里斯

4

2 回答 2

8

尽管微软文档说了什么,

  1. 写入的前缀数字实际上是一个LEB128 编码的 count
  2. 这是字节数,而不是字符数。

我链接的 Wiki 页面为您提供了解码代码,但我会考虑使用我自己的方案。您可以使用手动将字符串转换为 UTF8Encoding.GetBytes()并将其写入 MMF,并在其前面加上一个普通的无符号短字符。这样你就可以完全控制一切。

于 2013-04-16T12:54:58.033 回答
8

虽然BinaryWriter.Write 上的 MSDN 文档声明它“首先将字符串的长度写入为 UTF-7 编码的无符号整数”,但这是错误的。首先,UTF-7 是一种字符串编码,不能使用 UTF-7对整数进行编码。文档的含义(和代码所做的)是它使用可变长度 7 位编码(有时称为LEB128 )写入长度。在您的特定情况下,数据字节80 02的含义如下:

1000 0000 0000 0010

Nbbb bbbb Eaaa aaaa

  • N设置为 1 表示这不是最后一个字节
  • E设置为零表示这是最后一个字节
  • aaaaaaa并且bbbbbbb是真实数据;结果是:

00000100000000

aaaaaaabbbbbbb

100000000二进制,即十进制的 256。

于 2013-04-16T12:58:10.917 回答