Marshal C ++浮动到C#浮点精度问题

时间:2011-02-23 18:16:51

标签: c# c++ floating-point dbase

我有一个DBase IV数据库。每行都有一个带有ASCII编码字符串的备注字段,该字符串包含两个序列化的borland c ++结构。我能够使用OleDb提取数据,使用ASCIIEncoding类将其重新编码为ascii,使用BinaryReader将其转换为字节,然后使用Marshal.PtrToStructure将其转换为我的C#结构。我得到的数据是正确的,但是当它被转换为c#时,数据库中任何大的浮动都是完全错误的。例如,值为1149.00会转换为764.9844,但是像64.00这样的值会被罚款。我可以发布一些代码和结构,但我想我一开始就试着保持简短。我知道浮点数只能精确到7位数,但我很困惑,为什么我看到这个因为价值低于这个限制。

编辑:

struct cplusplusstruct // from the c++ code
{
  int   Number;
  float P;
  float ZP;
  float Hours;
  int   Month;
  int   Day;
  int   Year;
  int   Hour;
  int   Minute;
  int   Second;
  ULONG UPCTime;
  int   B;
  char  Name[21];
  float L;
  float H;
  float S;
}


[StructLayout(LayoutKind.Sequential, Pack = 1)]
public struct csharpstruct  //The C# struct I created
{
     public int Number;
     public float Pr;
     public float ZP;
     public float Hours;
     public int Month;
     public int Day;
     public int Year;
     public int Hour;
     public int Minute;
     public int Second;
     public UInt32 UPCTime;
     public int B;
 [MarshalAsAttribute(UnmanagedType.ByValTStr, SizeConst = 21)]
     public string Name;
     public float L;
     public float H;
     public float S;
 }


//OLE DB Connection and query ...

//Casting data to struct
ASCIIEncoding encoding = new ASCIIEncoding();
byte[] blob = encoding.GetBytes(memoString);
MemoryStream memoryStream = new MemoryStream(blob);
BinaryReader binaryReader = new BinaryReader(memoryStream);

int dataSize = Marshal.SizeOf(typeof(csharpstruct));
GCHandle handle = GCHandle.Alloc(binaryReader.ReadBytes(dataSize), GCHandleType.Pinned);
csharpstruct data = (csharpstruct) Marshal.PtrToStructure(handle.AddrOfPinnedObject(), typeof(csharpstruct));

编辑:以下是java代码,它可以很好地读取数据,但不使用任何强制转换。

org.xBaseJ.DBF dbf = new org.xBaseJ.DBF(dbPath);
org.xBaseJ.DBF dbf = new org.xBaseJ.DBF(dbPath);
MemoField m = (MemoField) dbf.getField("MEMOFIELD");

Charset charset = Charset.forName("US-ASCII");
CharsetDecoder decoder = charset.newDecoder();
ByteBuffer trendBytes = ByteBuffer.wrap(m.getBytes());
trendBytes.order(ByteOrder.LITTLE_ENDIAN);
trendBytes.getInt();
trendBytes.getFloat();

2 个答案:

答案 0 :(得分:3)

您的C#结构中有Pack = 1,但尚未说明您的C ++结构是否已打包。因为你的浮动之前有一个奇数大小的字段(21个字符的字符串)可能会导致麻烦,并且意味着你的浮动被读错了。之前的所有内容都有4个字节,所以打包不太可能导致问题。在继续之前,我会确保包装在C#和C ++中都匹配。

答案 1 :(得分:0)

我无法直接解决问题。问题似乎来自我使用的OLE数据提供程序。从数据库中检索到的数据与xBaseJ提供的数据略有不同。我最终使用IKVM.NET将xBaseJ转换为CLI字节码。这允许我用xBaseJ阅读器替换OLE数据提供程序。我的其余代码保持不变。