2022-08-27 C#调用C++DLL获取char*的常用封装技巧

15 篇文章 0 订阅
9 篇文章 0 订阅

项目场景:

碰到一个小伙伴,c#调用c++ dll接收char*,不知道如何进行C++DLL的封装,以及C#的调用;
下面分享一种比较好的做法;实际上,不仅是char*,任何数据都可以返回;


问题描述

在C++中,可以使用char *GetValue();
直接返回一个字符串;但是在C#中想要获取C++封装的DLL中的数据,则一般需要进行封装;
否则会产生一些内存问题;


原因分析:

C#对象属于托管堆,有垃圾回收机制管理内存。C++的内存是非托管的;如果封装和调用时候,稍不注意,两者就会因为机制的问题,产生内存异常的问题;


解决方案:

分享一种封装方法

  1. C++的DLL里面这样写
#pragma once

//C++修改代码如下:
typedef struct  _ComMonDevices
{
    char data[1024];
}ComMonDevices;


extern "C" __declspec(dllexport)  void __stdcall getComMonDevices(ComMonDevices* lpDeviceInfo);
  1. C++具体实现:
// dllmain.cpp : 定义 DLL 应用程序的入口点。
#include "pch.h"
#include <string>
#include "export.h"
using namespace std;

//C++修改代码如下:


string getDevices()
{
    return string("hello world");
}


void __stdcall getComMonDevices(ComMonDevices* lpDeviceInfo)
{
    string com = getDevices();        //此处替换你自己的代码获取信息
    if (lpDeviceInfo)
    {
        memset(lpDeviceInfo, 0, sizeof(ComMonDevices));
        strcpy_s(lpDeviceInfo->data, com.c_str());    //注意一下comd的长度,需小于1024
    }
}


BOOL APIENTRY DllMain( HMODULE hModule,
                       DWORD  ul_reason_for_call,
                       LPVOID lpReserved
                     )
{
    switch (ul_reason_for_call)
    {
    case DLL_PROCESS_ATTACH:
    case DLL_THREAD_ATTACH:
    case DLL_THREAD_DETACH:
    case DLL_PROCESS_DETACH:
        break;
    }
    return TRUE;
}


  1. C#的主程序里面这样写
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;

namespace TestDLL1
{
    internal class Program
    {

        //C#修改代码如下:

        public struct ComMonDevices
        {
            [MarshalAsAttribute(UnmanagedType.ByValTStr, SizeConst = 1024)]
            public string strData;                         //数据区域,长度为1024
        }

        [DllImport("Dll1.dll", CallingConvention =CallingConvention.StdCall)]
        public static extern void getComMonDevices(ref ComMonDevices lpDeviceInfo);

        static void Main(string[] args)
        {
            ComMonDevices DeviceInfo = new ComMonDevices();
            //获取信息
            getComMonDevices(ref DeviceInfo);
            string steTemp = DeviceInfo.strData;
            Console.WriteLine(steTemp);
            while (true) { }
        }
    }
}

  1. 最终的运行效果
    在这里插入图片描述

  2. 想要源码工程的小伙伴,请看下面:

链接:链接: https://pan.baidu.com/s/1mTlCqTFuLAAKqFeDAHlrGw?pwd=pvtm 提取码: pvtm 复制这段内容后打开百度网盘手机App,操作更方便哦

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ShaYQ

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值