【问题标题】:C++ to C#, static_cast into enumC++ 到 C#,static_cast 到枚举
【发布时间】:2011-07-29 22:19:23
【问题描述】:

我正在尝试将一些 VC 6.0 C++ 代码转换为 C#。具体来说,我正在解析一个二进制 dat 文件,但在转换这段代码时遇到了问题:

ar.GetFile()->Read(buf,sizeof(int));  
memmove(&x,buf,4);

pEBMA->before_after = static_cast<enum EBMA_Reserve>(x);
pEBMA->method       = static_cast<enum EBMA_Method>(x >> 4);

这是一些相关的代码。

struct EBMA_Data *pEBMA = &EBMA_data;

typedef CArray<struct EBMA_Data,struct EBMA_Data&> EBMA_data;

enum EBMA_Reserve
   {EBMA_DONT_RESERVE,
    EBMA_BEFORE,
    EBMA_AFTER
   };

enum EBMA_Method
   {EBMA_CENTER,
    EBMA_ALL_MATERIAL,
    EBMA_FRACTION,
    EBMA_RESERVE
   };


struct EBMA_Data
   {double reserved;
    double fraction;
    enum EBMA_Method method : 4;
    enum EBMA_Reserve before_after : 4;
   };

我在这里Cast int to Enum in C# 阅读了这个帖子,但我的代码没有给我与旧程序相同的结果。

这是我的一些 C# 代码:

reserved = reader.ReadDouble();
fraction = reader.ReadDouble();
beforeAfter = (EBMAReserve)Enum.ToObject(typeof(EBMAReserve), x);
method = (EBMAMethod)Enum.ToObject(typeof(EBMAMethod), (x >> 4));

我确实有字节序问题,所以我正在反转字节序。

public override double ReadDouble()
        {
            byte[] b = this.ConvertByteArrayToBigEndian(base.ReadBytes(8));
            double d = BitConverter.ToDouble(b, 0);
            return d;
        }
 private byte[] ConvertByteArrayToBigEndian(byte[] b)
        {
            if (BitConverter.IsLittleEndian)
            {
                Array.Reverse(b);
            }

            return b;
        }

然后我想也许字节顺序问题仍然让我失望,所以这是另一个尝试:

byte[] test = reader.ReadBytes(8);
Array.Reverse(test);
int test1 = BitConverter.ToInt32(buffer, 0);
int test2 = BitConverter.ToInt32(buffer, 4);
beforeAfter = (EBMAReserve)test1;
method = (EBMAMethod)test2;

我希望我已经提供了足够的详细信息来说明我正在尝试做的事情。

编辑:

这就是我解决问题的方法,显然我需要的值存储在二进制文件中 4 字节段的第一个字节中。这是一个循环。

byte[] temp = reader.ReadBytes(4);
byte b = temp[0];

res = (EBMAReserve)(b & 0x0f);
meth = (EBMAMethod)(b >> 4);

【问题讨论】:

  • 两个平台都是小端的

标签: c# c++ enums binaryfiles static-cast


【解决方案1】:

编辑:实际上EBMA_Data 的结构大小看起来是 17 个字节。

struct EBMA_DATA
{
  double reserved; //(8 bytes)
  double fraction; //(8 bytes)
  enum EBMA_Method method : 4; //(this is packed to 4 bits, not bytes)
  enum EMBA_Reserve before_after : 4; //(this too, is packed to 4 bits)
}

所以你的阅读代码应该看起来更像这样:

 EBMA_Data data = new EBMA_Data;
 data.reserved = reader.ReadDouble();
 data.fraction = reader.ReadDouble();
 byte b = reader.ReadByte();
 data.method = (EBMAMethod)(b >> 4);
 data.before_after = (EBMAReserve)(b & 0x0f);

不是 100% 确定,但看起来执行移位 x &gt;&gt; 4 字节的代码可能是被忽视的潜在问题。如果EBMAReserve 是 x 的低 4 位,EBMAMethod 是前 4 位,也许这段代码可以工作?

 EBMAReserve res = (EBMAReserve)(x & 0x0f);
 EBMAMethod meth = (EBMAMethod)(x >> 4);

我认为这就是 : 4 在结构中枚举之后的含义,它将两个枚举作为单个字节而不是 2 个字节打包到结构中。

【讨论】:

  • 它没有用。我的 EBMAReserve 是 EBMA_DONT_RESERVE 和 EBMAMethod 是 33554432 当它应该是 EBMAReserve: 32 和 EBMAMethod: EBMA_FRACTION 根据 C++ 程序
  • 嘿,谢谢,我让它工作了。您正确的是它存储在 17 个字节中,但在我正在使用的二进制文件中,此信息使用 20 个字节来循环。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2011-11-02
  • 1970-01-01
  • 1970-01-01
  • 2011-07-31
  • 1970-01-01
相关资源
最近更新 更多