【C#】 使用Gsof.Native 動態調用 C動態庫

来源:https://www.cnblogs.com/gaoshang212/archive/2018/04/12/8809360.html
-Advertisement-
Play Games

【C#】 使用Gsof.Native 動態調用 C動態庫 一、背景 使用C# 開發客戶端時候,我們經常會調用一些標準的動態庫或是C的類庫。雖然C# 提供的PInvoke的方式,但因為使用的場景的多變,有些時候可能沒辦法,固定位置的調用,或是需要按需求調用不同的庫。 二、說明 支持根據路徑動態載入DL ...


【C#】 使用Gsof.Native 動態調用 C動態庫

一、背景

使用C# 開發客戶端時候,我們經常會調用一些標準的動態庫或是C的類庫。
雖然C# 提供的PInvoke的方式,但因為使用的場景的多變,有些時候可能沒辦法,固定位置的調用,或是需要按需求調用不同的庫。

二、說明

  1. 支持根據路徑動態載入DLL類庫
  2. 支持using 銷毀載入的類庫
  3. 調用方便

github: https://github.com/gaoshang212/gsof/tree/master/Gsof.Native

nuget: https://www.nuget.org/packages/Gsof.Native/

介面說明:

NativeFactory 創建INative調用對象:

/// <summary>
/// 創建INative 對象
/// </summary>
/// <param name="p_fileName">文件路徑</param>
/// <returns></returns>
public static INative Create(string p_fileName);

/// <summary>
/// 創建INative 對象
/// </summary>
/// <param name="p_fileName">文件路徑</param>
/// <param name="p_calling">調用轉換方式(同PInvoke CallingConvention)/param>
/// <returns></returns>
public static INative Create(string p_fileName, CallingConvention _calling);

/// <summary>
/// 銷毀INative, 也可以調用 Native的Dispose方法
/// </summary>
/// <param name="p_native"></param>
public static void Free(INative p_native);

INative:

public interface INative : IDisposable
{
    /// <summary>
    /// 獲取函數委托
    /// </summary>
    /// <typeparam name="TDelegate"></typeparam>
    /// <returns></returns>
    TDelegate GetFunction<TDelegate>();
    /// <summary>
    /// 函數委托調用方式
    /// </summary>
    /// <typeparam name="TResult">返回值類型</typeparam>
    /// <typeparam name="TDelegate">函數對應的委托類型</typeparam>
    /// <param name="p_params">函數傳參</param>
    /// <returns></returns>
    TResult Invoke<TResult, TDelegate>(params object[] p_params);
    /// <summary>
    /// 函數名調用
    /// </summary>
    /// <typeparam name="TResult">返回值類型</typeparam>
    /// <param name="p_funName">函數名</param>
    /// <param name="p_params">函數傳參</param>
    /// <returns></returns>
    TResult Invoke<TResult>(string p_funName, params object[] p_params);
    /// <summary>
    /// 函數名調用
    /// </summary>
    /// <typeparam name="TResult">返回值類型</typeparam>
    /// <param name="p_funName">函數名</param>
    /// <param name="p_calling">調用轉換方式(同PInvoke CallingConvention)</param>
    /// <param name="p_params">函數傳參</param>
    /// <returns></returns>
    TResult Invoke<TResult>(string p_funName, CallingConvention p_calling, params object[] p_params);
    /// <summary>
    /// 函數名調用(非泛型)
    /// </summary>
    /// <param name="p_funName">函數名</param>
    /// <param name="p_retrunType">返回值類型</param>
    /// <param name="p_params">函數傳參</param>
    /// <returns></returns>
    object Invoke(string p_funName, Type p_retrunType, params object[] p_params);
    /// <summary>
    /// 函數委托調用方式
    /// </summary>
    /// <typeparam name="TDelegate">函數對應的委托類型</typeparam>
    /// <param name="p_params">函數傳參</param>
    void Call<TDelegate>(params object[] p_params);
    /// <summary>
    /// 函數名調用
    /// </summary>
    /// <param name="p_funName">函數名</param>
    /// <param name="p_params">函數傳參</param>
    void Call(string p_funName, params object[] p_params);
    /// <summary>
    /// 函數名調用
    /// </summary>
    /// <param name="p_funName">函數名</param>
    /// <param name="p_calling">調用轉換方式(同PInvoke CallingConvention)</param>
    /// <param name="p_params">函數傳參</param>
    void Call(string p_funName, CallingConvention p_calling, params object[] p_params);
}

三、使用

libtest.dll 為 中包括一個test函數

 int test(int input)
 {
     return input;
 }

方法名調用


int input = 0;
int result = -1;
using (var native = NativeFactory.Create(@"../../libtest.dll"))
{
    result = native.Invoke<int>("test", input);
}

dynamic 方式調用

  • 優點:調用方便,簡單類型調用時,不用做過多的定義。
  • 缺點:4.0下性能不理想,4.5+性能好很多,但相較於委托的方式,還差些。
int input = 0;
int result = -1;
using (dynamic native = NativeFactory.Create(@"../../libtest.dll"))
{
    result = native.test<int>(input);
}

委托方式調用

  • 優化:效率高,沒有了第一次動態構造委托的消耗,可獲取到函數委托增加 重覆調用消耗
  • 缺點:如果函數較多,委托定義較為繁瑣

[NativeFuncton("test")]
[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
delegate int Test(int p_sleep);

public void DelegateFunction()
{
    int input = 0;
    int result = -1;
    using (var native = NativeFactory.Create(@"../../libtest.dll"))
    {
        // 直接調用
        var result1 = native1.Invoke<int, Test>(input);

        // 獲取函數委托調用 
        var test = native.GetFunction<Test>();
        result = test(input);
    }

    Assert.AreEqual(input, result);

}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 練習 4.12: 流行的web漫畫服務xkcd也提供了JSON介面。例如,一個 https://xkcd.com/571/info.0.json 請求將返回一個很多人喜愛的571編號的詳細描述。 下載每個鏈接(只下載一次)然後創建一個離線索引。編寫一個xkcd工具,使用這些離線索引,列印和命令行輸入 ...
  • 在Kotlin中 使用js 函數 ...
  • 在一對多或者多對多的時候。如果通過一的一方取獲得多的一方的數據。除了第一次查詢表的數據以外。每獲得一條多的一方的數據就查詢一次。 如:通過學生表的記錄查詢成績表的記錄。 一個學生就查詢一次,50個學生就查詢50次。 如果需要查詢50個學生的成績,需要查詢資料庫的次數為 第一次查詢學生的記錄+50次查 ...
  • 概述 UWP Community Toolkit Extensions 中有一個為可視元素提供的擴展 - VisualExtensions,本篇我們結合代碼詳細講解 VisualExtensions 的實現。 VisualExtensions 為可視元素提供了一種簡單的在 XAML 中修改通用屬性的 ...
  • 搭建Spring: 3、演示IOC操作,在Service中調用Dao中的方法 UserDao.java package com.zzb.www.dao public class UserDao{ public void add(){ System.out.println("dao........") ...
  • 我們只知道靜態成員的用法,比如調用一個靜態方法要用類名去點出來,比如Math.Max(),但是為什麼是這樣調用呢?為什麼靜態構造函數不能有訪問修飾符?為什麼靜態構造函數沒有參數?靜態函數的執行時間又是什麼時候?為什麼? 在面向對象中,萬物皆對象。其中就有一種對象叫作類型對象,類型對象就是我們創建一個 ...
  • 1 /// <summary> 2 /// 智能查詢類型 3 /// </summary> 4 5 public enum QueryType : byte 6 { 7 /// <summary> 8 /// 發貨方聯繫人信息 9 /// </summary> 10 [DataMappingAttr ...
  • 1 <html> 2 <head> 3 </head> 4 <body> 5 <form id="recordform" name="recordform" autocomplete="off"> 6 <table cellpadding="0" cellspacing="0" class="tab ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...