C# c++ 傳遞函數指針


C#和c++之間相互傳遞函數指針

 

在C++和C#之中都有很多callback method,可以相互調用嗎,怎么傳遞,是我表弟的問題.

1.定義c++ dll ,導出方法

 

// sort.cpp : 定義 DLL 應用程序的導出函數。
//

#include "stdafx.h"
#include "sort.h"
#include "stdlib.h"
#include "iostream"
// 這是導出函數的一個示例。
typedef  int (__cdecl *compare )(const void *elem1, const void *elem2 ) ;
void fnsort(int arr[],int size,compare fuccsharp)
{
	std::cout<<"\r\narray length:"<<size<<"\r\n";
	std::cout << "entry fnsort in cpp\r\n";
	for(int index=0; index<size;index++){
		std::cout << arr[index] << " ";
	}
	qsort(arr,size,sizeof(int),fuccsharp);
	std::cout<<"\r\n sort end\r\n";
	for(int index=0; index<size;index++){
		std::cout << arr[index] << " ";
	}
	return ;
}


定義導出文件sort.def

LIBRARY   BTREE

EXPORTS
   fnsort

現在我們期待c#能呼叫fnsort,並實現funccsharp來實現排序算法,在c#中LoadLibrary,GetProcAddress,FreeLibrary當然是必不可少的.另外定義接口實現compare ,傳入兩個const void *,傳出int

 

static int SortASC(IntPtr a, IntPtr b) {
    int va = System.Runtime.InteropServices.Marshal.ReadInt32(a);
    int vb = System.Runtime.InteropServices.Marshal.ReadInt32(b);
    return va - vb;
}
static int SortDESC(IntPtr a, IntPtr b)
{
    int va = System.Runtime.InteropServices.Marshal.ReadInt32(a);
    int vb = System.Runtime.InteropServices.Marshal.ReadInt32(b);
    return vb - va;
}

同時定義委托

 

[UnmanagedFunctionPointerAttribute(CallingConvention.Cdecl)]
public delegate int INTARRAYSORT(IntPtr a, IntPtr b);

 

 

現在我們認為我們實現了int (__cdecl *compare )(const void *elem1, const void *elem2 ) ;現在沒事了,一切順利了,Marshal.GetDelegateForFunctionPointer會將一個intptr轉換為一個委托 ,new delegate(pfunc)可以將一個csharp func轉換為一個函數指針傳繪cpp,依上例,完整實現

 

using System;
using System.Text;
using System.Runtime.InteropServices;


namespace callDLL
{
    class Program
    {
        [DllImport("kernel32")]
        public static extern IntPtr LoadLibrary(string lpFileName);
        [DllImport("Kernel32")]
        public static extern bool FreeLibrary(IntPtr handle);
        [DllImport("Kernel32")]
        public static extern IntPtr GetProcAddress(IntPtr handle, String funcname);


        [UnmanagedFunctionPointerAttribute(CallingConvention.Cdecl)]
        public delegate int INTARRAYSORT(IntPtr a, IntPtr b);
        [UnmanagedFunctionPointerAttribute(CallingConvention.Cdecl)]
        public delegate void CPPSORT(int[] arr, int size, INTARRAYSORT callback);




        static int SortASC(IntPtr a, IntPtr b) {
            int va = System.Runtime.InteropServices.Marshal.ReadInt32(a);
            int vb = System.Runtime.InteropServices.Marshal.ReadInt32(b);
            return va - vb;
        }
        static int SortDESC(IntPtr a, IntPtr b)
        {
            int va = System.Runtime.InteropServices.Marshal.ReadInt32(a);
            int vb = System.Runtime.InteropServices.Marshal.ReadInt32(b);
            return vb - va;
        }
        static void Main(string[] args)
        {
            IntPtr dll = LoadLibrary("sort.dll");
            IntPtr func=GetProcAddress(dll, "fnsort");
            CPPSORT cppsort = (CPPSORT)Marshal.GetDelegateForFunctionPointer(func, typeof(CPPSORT));
            int[] arr = new int[] { 1, 7, 4 };
            //回叫函數可以使用委托實現
            cppsort(arr,arr.Length,new INTARRAYSORT(SortASC));
            cppsort(arr, arr.Length, new INTARRAYSORT(SortDESC));
            FreeLibrary(dll);
            Console.WriteLine("\r\nend");
            Console.Read();
        }
    }
}

 



輸出如下

 

 
        



array length:3
entry fnsort in cpp
1 7 4
 sort end
1 4 7
array length:3
entry fnsort in cpp
1 4 7
 sort end
7 4 1
end

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM