.net调用C++代码需要注意的一些问题

.net调用C++代码需要注意的一些问题

先写问题和总结

1、第一个问题是在编译时平台选择一致,比如同为X86或者X64。实验环境为64位机,我第一次C代码进去默认x86,C#则默认ANY CPU,调用时报错,检查后发现该问题,修改后正常;
2、C#环境下使用System.Runtime.InteropServices中的DLLImport调用的标准本机C代码生成的dll;
3、C代码中没有选择使用指针作为返回值,而是在函数参数中用指针返回,原因是C#中DLLImport声明方法中返回值不支持用引用ref返回,函数参数则允许使用;
4、实验中具体调用的C语言函数选择了常用的数组作为参数,在C中数组参数做参数其实就是指针,而C#中应尽量避免用指针这种不安全代码,使用ref,引用数组第一个元素,即为ref a[0],与C中传入数组第一个元素做函数参数类似;
5、C#中的引用ref 也可以调用C中的引用类型,比如void Swap(int& a,int& b),对应的C#则是void Swap(ref int a,ref int b)。

代码说明

C#源码:

using System;
using System.Runtime.InteropServices;

namespace DllPinvoke1
{
    class Program
    {
        //若调用的dll有多个接口,必须分开写,C中指针类型对应C#中ref
        [DllImport("Dll1.dll")]
        public static extern void Add(ref int a, ref int b, int len);

        [DllImport("Dll1.dll")]
        public static extern void Minus(ref int a, ref int b, int len);

        [DllImport("Dll1.dll")]
        public static extern int Changed(ref int a, ref int b, int len);

        static void Main(string[] args)
        {
            var a = new int[3] { 9, 8, 7 };
            var b = new int[3] { 3, 2, 1 };
            Add(ref a[0], ref b[0], a.Length);//调用时传入数组第一个值作函数参数
            Console.WriteLine("两数组相加的值是:{0},{1},{2}。", a[0], a[1], a[2]);

            var c = new int[3] { 9, 8, 7 };
            var d = new int[3] { 3, 2, 1 };
            Minus(ref c[0], ref d[0], c.Length);
            Console.WriteLine("两数组相减的值是:{0},{1},{2}。", c[0], c[1], c[2]);

            var e = new int[3] { 9, 8, 7 };
            var f = new int[3] { 3, 2, 1 };
            Console.WriteLine("-e[0]的值是{0}。",Changed(ref e[0],ref f[0],e.Length));

            Console.ReadKey();
        }
    }
}

C++源码:

#include "stdafx.h"

EXTERN_C _declspec(dllexport) void Add(int* a,int* b,int len) {
    for (int i = 0; i < len; i++) {
        a[i] = a[i] + b[i];
    }
}
EXTERN_C _declspec(dllexport) void Minus(int* a, int* b, int len) {
    for (int i = 0; i < len; i++) {
        a[i] = a[i] - b[i];
    }
}
EXTERN_C _declspec(dllexport) int Changed(int* a, int *b,int len) {
    Add(a, b, len);
    Minus(b, a, len);//说明这里的函数可以相互调用
    return b[0];
}

上面C代码中添加EXTERN_C _declspec(dllexport)是为了在C#中调用时能找到入口点,如果不添加该语句,则应该在源文件中增加同名.def文件说明入口点,格式如下:

LIBRARY "Dll1"

EXPORTS

Add @ 1
Minus @ 2
Changed @ 3

上面的Dll1是dll的名称,Add @ 1则表示dll中的第一个函数,以此类推。
此时的C#代码不用变,也能成功调用,C代码如下:

#include "stdafx.h"

void Add(int* a,int* b,int len) {
    for (int i = 0; i < len; i++) {
        a[i] = a[i] + b[i];
    }
}
void Minus(int* a, int* b, int len) {
    for (int i = 0; i < len; i++) {
        a[i] = a[i] - b[i];
    }
}
int Changed(int* a, int *b, int len) {
    Add(a, b, len);
    Minus(b, a, len);//说明这里的函数可以相互调用
    return b[0];
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值