Qt之调用C#的动态库的解决方法

环境:VS2019+Qt5.12

1. CLR库安装

首先,如果你VS2019没有安装CLR库,那么操作步骤为:

  • 打开 Visual Studio Installer
  • 在已安装中点击修改
  • 将使用C++的桌面开发的对V142(14.25)生成工具的C++/CLI支持
  • 点击右下角的修改,安装完成后重启软件即可

2. 新建类库(.NET Framework)

注意:此处请确认选择用于创建C#类库(.dll)的项目

此时解决方案的视图为:

一个简单的测试直接在Class1.cs文件添加内容即可,此测试中只修改此文件内容

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace ClassLibrary1
{
    public class Class1
    {
        public Class1() { }
        public int myAdd(int a, int b)
        {
            int c = a + b + 5;
            return c;
        }
        public void mySub(int a, int b, ref int c)
        {
            c = a - b - 5;
        }
        public void mySetText(string text)
        {
            Console.WriteLine("ClassLibrary1的类名Class1下的mySetText: {0}", text);
        }
        public void myGetText(ref string text)
        {
            text = "ClassLibrary1的类名Class1下的myGetText";
        }
    }
}

编写好了之后生成dll,将生成的dll复制到CLR的文件路径下

3. 新建CLR类库(.NET Framework)

此时解决方案的视图为:

一个简单的测试对应的头文件ClassLibrary2.h修改即可,此测试中只修改此文件内容

#pragma once

using namespace System;
using namespace System::Reflection;
using namespace System::Runtime::InteropServices;

#using "./ClassLibrary1.dll"

using namespace ClassLibrary1;

extern "C" __declspec(dllexport) int MyAdd(int a, int b)
{
	ClassLibrary1::Class1 obj;
	return obj.myAdd(a, b);
}
extern "C" __declspec(dllexport) void MySub(int a, int b,int *c)
{
	ClassLibrary1::Class1 obj;
	return obj.mySub(a, b, *c);
}

extern "C" __declspec(dllexport) void MySetText(char* text)
{
	ClassLibrary1::Class1 obj;
	String^ str = gcnew String(text);
	obj.mySetText(str);
}

extern "C" __declspec(dllexport) void MyGetText(char** text)
{
	ClassLibrary1::Class1 obj;
	String^ str = gcnew String("");
	obj.myGetText(str);
	*text = (char*)(void*)Marshal::StringToHGlobalAnsi(str);
}

这里编写完成后生成dll,然后非常重要的一步来了,将ClassLibrary1.dll、ClassLibrary2.dll、ClassLibrary2.lib准备复制到运行的Qt执行目录下,如果没有在同一个目录下,在ClassLibrary2调用ClassLibrary1时会找不到ClassLibrary1.dll文件而报错

4. Qt调用

4.1. 调用方法1

#include <QCoreApplication>
#include <windows.h>
#include <QDebug>
#include <QLibrary>
#include <QFile>

typedef int (*ADD)(int,int);
typedef void (*SUB)(int,int,int *);
typedef void (*SHOW)(QString);
int main(int argc, char *argv[])
{
    QLibrary mylib("ClassLibrary2.dll");  //声明dll文件
    if (mylib.load())  //判断加载是否成功
    {
        qDebug() << "DLL  loaded!";
        ADD add = (ADD)mylib.resolve("MyAdd"); //链接到add函数
        qDebug() << "add status: " << add;
        if (add){
            qDebug()<< "Link to add Function is OK!" << add(3,2) ;
        }
        SUB sub = (SUB)mylib.resolve("MySub");
        qDebug() << "sub status: " << sub;
        if (sub){
            int c = 10;
            sub(3,2,&c);
            qDebug()<< "Link to sub Function is OK!" << c;
        }
        SHOW show = (SHOW)mylib.resolve("MySetText");
        qDebug() << "show status: " << show;
        if (show){
            qDebug()<< "Link to show Function is OK!" ;
            const char *buf = "helloworld";
            show(buf);
        }
        qDebug()<< "DLL unload " << mylib.unload ();
    }

    else
    {
        qDebug()<< "DLL is not loaded!" ;
    }

    return 0;
}

4.2. 调用方法2

  • 右键项目->选择添加库->选择外部库
  • 在库文件中找到刚才生成的ClassLibrary2.lib
  • 将平台下的linux、Mac取消勾选
  • 将Windows下的所有都取消勾选

此时界面如下

extern "C" __declspec(dllexport) void MyGetText(char **p);
extern "C" __declspec(dllexport) void MySetText(char *p);

int main(int argc, char *argv[])
{
    MySetText(QString("helloworld").toUtf8().data());

    char* change_t=nullptr;
    MyGetText(&change_t);
    qDebug() << QString(change_t);
}

到此这篇关于Qt之调用C#的动态库的文章就介绍到这了,更多相关Qt调用C#动态库内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Qt程序中调用C#编写的dll(推荐)

    1.打开Visual Studio,新建一个C#的Class Library项目(这里选择的是.Net Framework 4),项目名为CSharpDll. 2.由于默认没有引入Forms等UI库,先在reference中添加引用System.Windows.Forms以便可以在测试中使用MessageBox等. 3.最终C#编写的dll的源代码如下图所示,命名空间为CSharpDll,公共类为CSharpClass. using System; using System.Collection

  • C# Mqtt 断线重连的实现代码

    在通过 MqttClient 客户端连接之后,在服务端服务重启时,客户端如果没有重连机制,则无法再接收到订阅的消息. 使用的 Mqtt 组件为:M2Mqtt.Net.dll 一些特性发现 (1)如果提供的服务端地址是不可解析的,会引发异常无法实例化 MqttClient 对象. (2)Connect 无法连接时会引发异常,IsConnected 为 false. (3)服务端断开会触发客户端的 ConnectionClosed 事件,IsConnected 为 false. (4)重新 Conn

  • C#调用非托管动态库中的函数方法

    C#如何调用一个非托管动态库中的函数呢,比如用VC6写的动态库,总之C#调用动态库的过程是比Java调用DLL动态库方便快捷多了,下面举例说明这个过程. 1.创建一个非托管动态库 代码如下: 复制代码 代码如下: //这一句是声明动态库输出一个可供外不调用的函数原型.     extern   "C"  __declspec(dllexport)  int  add( int ,  int ); int  add( int  a, int  b)      {          //实

  • Qt之调用C#的动态库的解决方法

    环境:VS2019+Qt5.12 1. CLR库安装 首先,如果你VS2019没有安装CLR库,那么操作步骤为: 打开 Visual Studio Installer 在已安装中点击修改 将使用C++的桌面开发的对V142(14.25)生成工具的C++/CLI支持 点击右下角的修改,安装完成后重启软件即可 2. 新建类库(.NET Framework) 注意:此处请确认选择用于创建C#类库(.dll)的项目 此时解决方案的视图为: 一个简单的测试直接在Class1.cs文件添加内容即可,此测试中

  • Rust应用调用C语言动态库的操作方法

    目录 外部功能接口FFI UDP套接字的读超时 Rust调用C语言动态库中的函数 避免重复造轮子,使用Rust官方C语言库 外部功能接口FFI 虽然高级(脚本)编程语言的功能丰富,表达能力强,但对底层的一些特殊操作的支持并不完善,就需要以其他编程语言来实现.调用其他编程语言的接口,被称为Foreign Function Interface,直译为外部功能接口.该接口通常是调用C语言实现的外部功能模块,因为C语言接近于全能,几乎任何功能都能够实现:正如同使用汇编语言也可以实现很多功能一样,但开发效

  • Java通过调用C/C++实现的DLL动态库——JNI的方法

    由于项目的需要,最近研究了java 调用DLL的方法,将如何调用的写于此,便于日后查阅: 采用的方法是JNI: Java Native Interface,简称JNI,是Java平台的一部分,可用于让Java和其他语言编写的代码进行交互. 下面是从网上摘取的JNI工作示意图: 总体说明:先在JAVA中建立一个类,通过javac生成.class,再由javah生成.h:然后将.h复制到VC下,由VC实现具体函, 并编译通过后生成DLL,将DLL放入JAVA工程中使用,完毕. 下面说说具体步骤(含实

  • android 调用JNI SO动态库的方法

    总结一下: android 调用JNI 分为静态调用与动态调用(不论动态还是静态前提都是NDK环境已经配置好的前提下) 一.静态主要就是将c(.c)或者c++(cpp)的源文件直接加到项目中进行调用, 然后在CMakeLists.txt中进行配置. 二.动态调用 1.动态调用使用已经编译好的动态库.so文件 2.android调用ndk类 生成后的so文件 public class SerialPort { p */ public static native int GetSOVer(Strin

  • golang 调用c语言动态库方式实现

    下面我们自己在 Linux 下做一个动态库(.so 文件 - Shared Object),然在用 Go 来使用它.本文所用的操作系统为 Ubuntu18.04, 以 gcc 作为编译器. 1.实现头文件,声明文件中函数.这里创建一个add.h文件. #ifndef __ADD_H__ #define __ADD_H__ char* Add(char* src, int n); #endif 2.实现add主体函数add.c #include <string.h> #include <s

  • Pycharm无法显示动态图片的解决方法

    最近在学习的时候遇到了一个问题始终没有解决,这个博客写的也不是完全解决了这个问题.指示换了一种可行的思路而已. 在运行一些显示动态的图片时,Pycharm只显示一帧,也没有找到什么解决办法,试着把项目在cmd指令下运行,发现竟然可以解决.所以原问题如果没有找到什么解决办法可以考虑这么做,如果找到了解决办法再更一下.实现的是python调用matplotlib画电视没有台的情形. import numpy as np from matplotlib import pyplot as plt fro

  • 关于JS中setTimeout()无法调用带参函数问题的解决方法

    本文实例分析了JS中setTimeout()无法调用带参函数问题的解决方法.分享给大家供大家参考,具体如下: 解决方法:重写setTimeout() 方法,需要用到闭包函数.如下: var _st = window.setTimeout; window.setTimeout = function(fRef, mDelay){ if (typeof fRef == 'function') { var argu = Array.prototype.slice.call(arguments, 2);

  • sql server定时作业调用Kettle job出错的快速解决方法

    错误信息: Unable to list jar files in plugin folder 'C:\Windows\system32\config\systemprofile\.kettle\plugins' Unable to get VFS File object for filename 'C:\Windows\system32\config\systemprofile\.kettle\plugins' : Could not find file with URI "C:\Window

  • 浅谈c++调用python链接的问题及解决方法

    python 版本3.3 系统:windows 问题:链接时报告 1>pythonIniti.obj : error LNK2019: 无法解析的外部符号 __imp___Py_NegativeRefcount,该符号在函数 "public: __thiscall boost::python::api::object_base::~object_base(void)" (??1object_base@api@python@boost@@QAE@XZ) 中被引用 1>pyth

  • Qt串口通信开发之QSerialPort模块Qt串口通信接收数据不完整的解决方法

    在使用串口接收数据时,当数据量大的时候会出现数据接收不完整的情况. 因为串口数据获取函数readAll()由readyRead()信号触发,但readyRead()信号在串口读到起始标志时立即发送,并不保证一定是当前所发数据的起始部分. 因此串口通信双方在通信前应制定好通信协议,规定好数据的起始和结束标志,串口当读到完整的起始和结束标志之后,才认定读完一条完整的数据. 本例中用串口定时发送当前时间,用"#"表示数据的结尾,定时时间为0毫秒,即能发多快就发多快. 发送 void Widg

随机推荐