majie007
捷于晨
2020-03-09 09:31
采纳率: 50%
浏览 426

C#调用C++ Dll 返回信息与C++返回结构体中信息不一致?

各位大佬,小弟遇到一个很难理解的问题,情况如下:
C++ 开发的Dll和硬件通讯,在C++中使用返回正常的数值,而在C#中通过DllImport方式调用,发现返回的数值不一致,具体代码如下:

** C++ 提供的结构体:**

typedef struct _SXPSERVER { 
    BYTE bNodeaddr[6];  
    DWORD dwIp;  
    char  szMachineType[16]; 
    char  szHostName[16]; 
} SXPSERVER, *LPSXPSERVER; 


图片说明

**C# 自己写的DllImport 的结构体**
     [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
        public struct SXPSERVER
        {
            public uint dwIp;
            [MarshalAs(UnmanagedType.ByValArray, SizeConst = 6, ArraySubType = UnmanagedType.I1)]
            public byte[] bNodeaddr;
            [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 16)]
            public string szMachineType;
            [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 16)]
            public string szHostName;
        }

图片说明

调用全部都一切正常,结构体中也能够返回值,现在就是这个DWORD dwIp; 转换到C#中的 public uint dwIp;返回不一样,其他的都没有问题,请问各位大佬,我这个转换不对吗?查了半天看到的对应转换关系都是DWORD -》uint,我改成了int同样也是不对,这个咋整,这个值还会是后面的函数中的关键参数,卡到这里进行不下去了。

PS:看到有大神留言说这个参数位置写反了,我最开始的时候就是按照这个顺序(

    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
        public struct SXPSERVER
        {
            [MarshalAs(UnmanagedType.ByValArray, SizeConst = 6, ArraySubType = UnmanagedType.I1)]
            public byte[] bNodeaddr;
                      public uint dwIp;
            [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 16)]
            public string szMachineType;
            [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 16)]
            public string szHostName;
        }

)来的,出来的结果和这个是一样的

问题已经解决了,解决的方式是这样的:
我在C++的结构体与C#定义的结构体的顺序是不一样的,但这个不是主要的原因。而是我调用方式有问题。我把c#实现的这个结构体单独放到了一个DLL中,然后再通过别的工程去调用这个DLL,这样的话。不管结构体中的顺序是什么,它都能够调用成功,但是他成功之后,里面的数值是不对的,这个是我通过反编译看到的,所有的结构体变量约束都没有了,所以这个值肯定不对啦。而我把这个调用C++的这个结构体直接放到了我现在用的工程当中,按照错误的顺序,这个方法都不会调用成功,而按照正确的顺序后就ok啦!

  • 点赞
  • 写回答
  • 关注问题
  • 收藏
  • 邀请回答

4条回答 默认 最新

  • caozhy
    已采纳
    BYTE bNodeaddr[6];  
    DWORD dwIp;  
    

    这两个写反了
    而且考虑到数据对齐,bNodeaddr可能占用的是8个字节,具体你调试的时候看下内存

    点赞 评论
  • FakeTaoZero
    世当珍惜 2020-03-09 09:37

    是不是C#结构体里面的成员生命顺序和C++不一样导致的,声明顺序决定了成员的内存排布

    点赞 评论
  • d2262272d
    窝米逗佛~ 2020-03-09 10:09

    我想,c#跟c++交叉的话,首先他不会用返回值为结构体吧!~
    然后是以参数形式的,会不会用的是 SXPSERVER&?
    这种的话我记得是要用SXPSERVER*& 具体是什么原因有点记不得了!~
    另外还有一个也是忘了原因的,就是给c#的dll都用__stdcall,不适用另外两种,跟你同事试试?

    点赞 评论
  • slevenxulianjie
    slevenxulianjie 2020-03-09 17:25

    感觉应该这么写:

        [StructLayout(LayoutKind.Sequential, Pack = 4,CharSet = CharSet.Ansi)]
        public class SXPSERVER
        {
            [MarshalAs(UnmanagedType.ByValArray, SizeConst = 6)]
            public byte[] bNodeaddr = new byte[6];
    
            public UInt32 dwIp;
    
            [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 16)]
            public string szMachineType = new string(' ',16);
    
            [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 16)]
            public string szHostName = new string(' ',16);
        }
    

    主要的不同是*字节对齐*与*初始化*,看看能否帮到你

    点赞 评论

相关推荐