• P/Invoke继续谈


    上回说到封装rocketmq-client-cpp的时候会碰到内存损坏的问题,本以为能收到服务端的消息就万事大吉了,但是在深入测试的过程中,发现收到的消息会有乱码出现,

    之前和服务端的研发人员讨论过字符集的问题,知道服务器回传的是UTF8的字符串,所以在接收时用下面这样的方式可以在大部分情况下得到正确的内容:

    return Encoding.UTF8.GetString(Encoding.Default.GetBytes(Marshal.PtrToStringAnsi(GetMessageIdNative(message))));

    但是在少数情况下会有乱码内容,比较之后发现是中英文符号混杂的情况就会出现这种问题,经过反复测试和思考之后确认需要的是Marshal.PtrToStringUTF8,

    可惜目前的项目都是在.NET Framework下开发,PtrToStringUTF8在.NET Framework中完全没有支持,所以只能自己写一个了

    static string PtrToStringUTF8(IntPtr ptr)
    {
        var bytesCount = 0;
        byte b;
        do
        {
            b = Marshal.ReadByte(ptr, bytesCount);
            bytesCount++;
        }
        while (b != 0);
        var bytes = new byte[bytesCount - 1];
        Marshal.Copy(ptr, bytes, 0, bytesCount - 1);
        return Encoding.UTF8.GetString(bytes);
    }

    没过几天,又遇到个接入某省定制多功能读卡器的需求,在这里碰到了一个看起来很简单的函数,声明如下:

    long iReadXXX(char* pOutInfo)

    这里的pOutInfo是一个长度为4096的输出参数,所以虽然也是要接收一个字符串,但是通过输出参数的形式,而不是上次的返回值,平台调用的形式也就必然不同。

    这里采用的是接收字符数组的方式来进行调用,声明如下:

    public static extern int iReadXXX(IntPtr pOutInfo);

    调用代码如下:

    byte[] bytes = new byte[4096];
    GCHandle pinned = GCHandle.Alloc(bytes, GCHandleType.Pinned);
    var result = iReadXXX(pinned.AddrOfPinnedObject());
    pinned.Free();
    int count = bytes.ToList().FindIndex(b => b == 0);
    var text = Encoding.Default.GetString(bytes, 0, count);
  • 相关阅读:
    ubuntu18.04 安装 jdk
    MySQL集群(PXC)入门
    Synchronized 详解
    Java多线程之内存可见性
    MySQL8.0新特性
    vue配置域名访问
    debain8 安装mysql8
    idea 激活步骤
    spring boot 设置tomcat post参数限制
    人脸识别技术全面总结:从传统方法到深度学习[转载自机器之心]
  • 原文地址:https://www.cnblogs.com/s5689412/p/12773177.html
Copyright © 2020-2023  润新知