亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

在C#項目中調(diào)用C++編寫的動態(tài)庫的三種方式

 更新時間:2024年01月11日 08:25:44   作者:小乖獸技術(shù)  
這篇文章給大家介紹了三種方式詳解如何在C#項目中調(diào)用C++編寫的動態(tài)庫,文中通過代碼示例給大家介紹的非常詳細,具有一定的參考價值,需要的朋友可以參考下

場景和優(yōu)點

在以下場景下,可能會使用C#調(diào)用C++編寫的dll:

  • C++庫已經(jīng)存在并且經(jīng)過了充分測試和驗證,需要被C#項目重復使用時;
  • C++編寫的庫中包含高性能計算、海量數(shù)據(jù)處理等需要使用底層語言實現(xiàn)的操作時,可以考慮將這些操作封裝為動態(tài)鏈接庫供C#調(diào)用;
  • 在跨平臺開發(fā)時,C++可在多個平臺上運行,通過封裝為dll,可以讓C#項目也能夠在多個平臺上運行;
  • 需要將不同的功能模塊拆分成獨立的組件,C++編寫的dll可以作為一個獨立的組件,供C#項目或其他語言的項目調(diào)用。

此外,使用C#調(diào)用C++編寫的dll還有以下優(yōu)點:

  • C#具有較高的開發(fā)效率和易用性,通過調(diào)用C++編寫的dll可以兼顧高性能和高開發(fā)效率。
  • C#可以使用.NET Framework提供的強大工具和庫,如LINQ、異步編程等等,這些工具和庫可以提高開發(fā)效率,同時也可以利用C++的性能優(yōu)勢。
  • C#可以與其他語言,如Java、Python等配合使用,借助各種技術(shù),如SOAP、WCF、gRPC等實現(xiàn)多語言之間的互操作。
  • C++作為一種系統(tǒng)級編程語言,可以訪問系統(tǒng)底層資源,如內(nèi)存、磁盤、網(wǎng)絡等,C#調(diào)用C++編寫的dll可以實現(xiàn)訪問這些底層資源的功能,從而提供更多的功能。

C#調(diào)用C++編寫的動態(tài)庫的方式

在C#中調(diào)用C++編寫的動態(tài)庫有以下幾種方式:

1. 使用DllImport特性

使用DllImport特性可以直接引入動態(tài)鏈接庫中的C++函數(shù),并在C#中進行調(diào)用。

下面是一個簡單的示例:

首先,我們在C++中編寫一個簡單的dll,里面包含一個計算兩數(shù)之和的函數(shù)addition:

c++Copy Code// file: mylib.cpp
#include "pch.h"
#include "mylib.h"

int addition(int a, int b) {
    return a + b;
}

然后,我們在C++中將其封裝為一個dll,并導出addition函數(shù):

// file: mylib.h

#ifdef MYLIB_EXPORTS
#define MYLIB_API __declspec(dllexport)
#else
#define MYLIB_API __declspec(dllimport)
#endif

extern "C" MYLIB_API int addition(int a, int b);  // export the function

接著,在C#項目中使用DllImport特性導入這個dll,并調(diào)用其中的函數(shù):

using System.Runtime.InteropServices;

class Program
{
    [DllImport("MyLib.dll", CallingConvention = CallingConvention.Cdecl)]
    public static extern int addition(int a, int b);

    static void Main(string[] args)
    {
        int result = addition(1, 2);
        Console.WriteLine("The sum is: " + result);
    }
}

在上述示例中,我們使用DllImport特性聲明了一個addition方法,將其與C++中的addition函數(shù)進行綁定。在Main函數(shù)中,我們調(diào)用了這個方法,并輸出計算結(jié)果。

需要注意的是,在使用DllImport特性時,需要指定正確的dll名稱和函數(shù)調(diào)用規(guī)約,否則可能會出現(xiàn)運行時錯誤。

2. 使用C++/CLI

另一種實現(xiàn)方式是使用C++/CLI(C++/Common Language Infrastructure)。

C++/CLI是一種結(jié)合了C++和CLR(Common Language Runtime)的語言,它可以編寫針對.NET Framework/CLR的代碼,同時也可以訪問C++的底層資源。因此,我們可以使用C++/CLI來封裝C++庫,并將其作為dll供C#調(diào)用。

下面是一個簡單的示例:

首先,在C++/CLI中編寫一個類LibraryWrapper,里面包含一個使用C++庫計算兩數(shù)之和的方法Addition:

// file: LibraryWrapper.h
#pragma once

namespace MyLibrary {

    public ref class LibraryWrapper
    {
    private:
        Library* lib;  // the C++ object we want to wrap
    public:
        LibraryWrapper();  // constructor
        ~LibraryWrapper();  // destructor
        int Addition(int a, int b);  // method used to add two numbers
    };
}

其中,Library是我們需要封裝的C++庫中的一個類。

然后,在實現(xiàn)文件LibraryWrapper.cpp中實現(xiàn)類的構(gòu)造函數(shù)、析構(gòu)函數(shù)和Addition方法:

// file: LibraryWrapper.cpp
#include "pch.h"
#include "LibraryWrapper.h"
#include "Library.h"

using namespace MyLibrary;

LibraryWrapper::LibraryWrapper()
{
    lib = new Library();  // create a new Library object
}

LibraryWrapper::~LibraryWrapper() {
    delete lib;  // release the memory
}

int LibraryWrapper::Addition(int a, int b)
{
    return lib->addition(a, b);  // call the addition method in C++ library
}

這里我們實例化了一個C++庫中的對象,然后在Addition方法中調(diào)用了它的addition方法。

最后,在C++/CLI項目中發(fā)布dll,并在C#項目中引用。在C#項目中,我們可以創(chuàng)建一個LibraryWrapper對象,并調(diào)用其中的Addition方法:

using System;
using System.Runtime.InteropServices;

namespace CppCLILibraryTest
{
    class Program
    {
        static void Main(string[] args)
        {
            MyLibrary.LibraryWrapper wrapper = new MyLibrary.LibraryWrapper();
            int result = wrapper.Addition(1, 2);
            Console.WriteLine("The sum is: " + result);
        }
    }
}

需要注意的是,當使用C++/CLI封裝C++庫時,我們需要確保兩者所使用的Runtime是相同的。比如,如果C++庫是使用靜態(tài)連接的方式與CRT(C Runtime)鏈接的,那么我們需要在C++/CLI項目的屬性中設置“/MT”選項,以保證代碼使用相同的CRT版本。

3. 使用COM組件

另一種實現(xiàn)方式是使用COM組件。COM是微軟推出的一種二進制接口標準,它可以讓不同的應用程序之間以二進制碼互相通信。

下面是一個簡單的示例:

首先,在C++中編寫一個簡單的dll,里面包含一個計算兩數(shù)之和的函數(shù)addition:

// file: MyLibrary.h
#pragma once

#ifdef MYLIBRARY_EXPORTS
#define MYLIBRARY_API __declspec(dllexport)
#else
#define MYLIBRARY_API __declspec(dllimport)
#endif

namespace MyLibrary {
    class MyMath {
    public:
        static int Addition(int a, int b);
    };
}

然后,我們將這個dll封裝為一個COM組件。我們需要創(chuàng)建一個類,其中包含COM接口和類工廠:

// file: MathCOM.h
#pragma once

#include "MyLibrary.h"

class MathCOM : public IUnknown {
private:
    ULONG m_cRef;
public:
    MathCOM();
    ~MathCOM();

    // IUnknown methods
    STDMETHODIMP QueryInterface(REFIID riid, void** ppv);
    STDMETHODIMP_(ULONG) AddRef(void);
    STDMETHODIMP_(ULONG) Release(void);

    // COM interface method
    STDMETHODIMP Addition(int a, int b, int* result);
};

class MathClassFactory : public IClassFactory {
private:
    ULONG m_cRef;
public:
    MathClassFactory();
    ~MathClassFactory();

    // IUnknown methods
    STDMETHODIMP QueryInterface(REFIID riid, void** ppv);
    STDMETHODIMP_(ULONG) AddRef(void);
    STDMETHODIMP_(ULONG) Release(void);

    // IClassFactory methods
    STDMETHODIMP CreateInstance(IUnknown* pUnknownOuter, REFIID riid, void** ppvObject);
    STDMETHODIMP LockServer(BOOL fLock);
};

在實現(xiàn)文件MathCOM.cpp中,我們需要為這些接口方法提供具體的實現(xiàn):

// file: MathCOM.cpp
#include "stdafx.h"
#include "MathCOM.h"

MathCOM::MathCOM() {
    m_cRef = 1;
}

MathCOM::~MathCOM() {}

STDMETHODIMP MathCOM::QueryInterface(REFIID riid, void** ppv) {
    *ppv = NULL;

    if (riid == IID_IUnknown || riid == IID_IDispatch)
        *ppv = this;

    if (*ppv != NULL) {
        ((LPUNKNOWN)*ppv)->AddRef();
        return S_OK;
    }

    return E_NOINTERFACE;
}

STDMETHODIMP_(ULONG) MathCOM::AddRef() {
    return InterlockedIncrement((LONG*)&m_cRef);
}

STDMETHODIMP_(ULONG) MathCOM::Release() {
    ULONG cRef = InterlockedDecrement((LONG*)&m_cRef);
    if (cRef == 0) delete this;
    return cRef;
}

STDMETHODIMP MathCOM::Addition(int a, int b, int* result) {
    *result = MyLibrary::MyMath::Addition(a, b);
    return S_OK;
}

MathClassFactory::MathClassFactory() {
    m_cRef = 1;
}

MathClassFactory::~MathClassFactory() {}

STDMETHODIMP MathClassFactory::QueryInterface(REFIID riid, void** ppv) {
    *ppv = NULL;

    if (riid == IID_IUnknown || riid == IID_IClassFactory)
        *ppv = this;

    if (*ppv != NULL) {
        ((LPUNKNOWN)*ppv)->AddRef();
        return S_OK;
    }

    return E_NOINTERFACE;
}

STDMETHODIMP_(ULONG) MathClassFactory::AddRef() {
    return InterlockedIncrement((LONG*)&m_cRef);
}

STDMETHODIMP_(ULONG) MathClassFactory::Release() {
    ULONG cRef = InterlockedDecrement((LONG*)&m_cRef);
    if (cRef == 0) delete this;
    return cRef;
}

STDMETHODIMP MathClassFactory::CreateInstance(IUnknown* pUnknownOuter, REFIID riid, void** ppvObject) {
    if (pUnknownOuter) return CLASS_E_NOAGGREGATION;

    MathCOM* pMathCOM = new MathCOM();
    if (!pMathCOM) return E_OUTOFMEMORY;

    HRESULT hResult = pMathCOM->QueryInterface(riid, ppvObject);
    pMathCOM->Release();
    return hResult;
}

STDMETHODIMP MathClassFactory::LockServer(BOOL fLock) {
    return S_OK;
}

在項目中使用C++編譯器生成COM組件dll之后,在C#項目中使用COM互操作性來調(diào)用這個COM組件,代碼如下:

using System.Runtime.InteropServices;

namespace COMTest
{
    [ComImport, Guid("B9D43B8A-61F3-4668-AB30-C2BE194AD0AA")]
    [InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
    public interface IMathCOM {
        [PreserveSig]
        int Addition(int a, int b, out int result);
    }

    [ComImport, Guid("8CFD0B22-24A3-4490-9127-9DB3FD53E15F")]
    class MathCOM { }

    class Program
    {
        static void Main(string[] args)
        {
            IMathCOM mathCOM = (IMathCOM)new MathCOM();
            int result = 0;
            mathCOM.Addition(1, 2, out result);
            Console.WriteLine("The sum is: " + result);
        }
    }
}

在這個示例中,我們聲明了一個用來調(diào)用COM組件的接口IMathCOM,然后實例化MathCOM類并把它轉(zhuǎn)換為IMathCOM類型,就可以調(diào)用其中的Addition方法了。

以上三種方式都可用于調(diào)用C++編寫的動態(tài)庫,選擇使用哪種方式應該根據(jù)具體的場景和需求來決定。

到此這篇關于在C#項目中調(diào)用C++編寫的動態(tài)庫的三種方式的文章就介紹到這了,更多相關C#調(diào)用C++動態(tài)庫內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • Unity TextMeshPro實現(xiàn)富文本超鏈接默認字體追加字體

    Unity TextMeshPro實現(xiàn)富文本超鏈接默認字體追加字體

    這篇文章主要為大家介紹了Unity TextMeshPro實現(xiàn)富文本超鏈接默認字體追加字體示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-01-01
  • C#使用doggleReport生成pdf報表的方法

    C#使用doggleReport生成pdf報表的方法

    這篇文章主要介紹了C#使用doggleReport生成pdf報表的方法,結(jié)合實例形式分析了C# doggleReport安裝及使用具體操作技巧,需要的朋友可以參考下
    2017-06-06
  • C#使用FileSystemWatcher控件實現(xiàn)的文件監(jiān)控功能示例

    C#使用FileSystemWatcher控件實現(xiàn)的文件監(jiān)控功能示例

    這篇文章主要介紹了C#使用FileSystemWatcher控件實現(xiàn)的文件監(jiān)控功能,結(jié)合實例形式分析了C# FileSystemWatcher組件的功能及監(jiān)控文件更改情況的具體使用技巧,需要的朋友可以參考下
    2017-08-08
  • C#中String類常用方法匯總

    C#中String類常用方法匯總

    這篇文章主要介紹了C#中String類常用方法,較為詳細的匯總了String類中的常用方法,對于深入掌握C#字符串操作有著很好的學習借鑒價值,需要的朋友可以參考下
    2014-11-11
  • Unity實現(xiàn)場景漫游相機

    Unity實現(xiàn)場景漫游相機

    這篇文章主要為大家詳細介紹了Unity實現(xiàn)場景漫游相機,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-10-10
  • c#中返回文章發(fā)表的時間差的示例

    c#中返回文章發(fā)表的時間差的示例

    現(xiàn)在是2012-12-04 11:29:59,發(fā)表時間是:2012-12-02 21:29:59,傳統(tǒng)的ts.Days因為值為1天14小時0分0秒,會返回“昨天”,而這個會返回“前天”
    2012-12-12
  • C#調(diào)用存儲過程詳解(帶返回值、參數(shù)輸入輸出等)

    C#調(diào)用存儲過程詳解(帶返回值、參數(shù)輸入輸出等)

    這篇文章主要介紹了C#調(diào)用存儲過程的方法,結(jié)合實例形式詳細分析了各種常用的存儲過程調(diào)用方法,包括帶返回值、參數(shù)輸入輸出等,需要的朋友可以參考下
    2016-06-06
  • 訪問修飾符(C# 編程指南)

    訪問修飾符(C# 編程指南)

    所有類型和類型成員都具有可訪問性級別,用來控制是否可以在您程序集的其他代碼中或其他程序集中使用它們。您在聲明類型或成員時使用以下訪問修飾符之一來指定其可訪問性
    2011-02-02
  • C#下使用XmlDocument操作XML詳解

    C#下使用XmlDocument操作XML詳解

    本文詳細講解了C#使用XmlDocument操作XML的方法,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-06-06
  • C#實現(xiàn)單例模式的幾種方法總結(jié)

    C#實現(xiàn)單例模式的幾種方法總結(jié)

    這篇文章主要介紹了C#實現(xiàn)單例模式的幾種方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-01-01

最新評論