JNA调用dll(c++)附带解析xml——JNA-JNI(四)

JNA调用dll(c++)附带解析xml——JNA-JNI(四)

系列文章:
Java通过JNI调用C++动态链接库dll,并打在jar包内 ——JNA-JNI(一)
Java使用JNA调用C++动态链接库——JNA-JNI(二)
Mac M1 Xcode创建动态链接库dylib(c++)——JNA-JNI(三)
JNA调用dll(c++)附带解析xml——JNA-JNI(四)
JNA参数类型转换(含接收、发送结构体)——JNA-JNI(五)

使用tinyxml2

使用tinyxml2:https://github.com/leethomason/tinyxml2,下载zip解压即可

tinyxml2.cpptinyxml2.h 拷贝至项目目录,使用时包含头文件和引入名字空间

#include "tinyxml2.h"
using namespace tinyxml2

c++解析xml源码

xxx.cpp文件,生成dll

#include <stdlib.h>
#include <string.h>
#include <iostream>
#include <stdio.h>
#include <string>
#include "xxx.h"
#include "tinyxml2.h"   

using namespace std;
using namespace tinyxml2;

//function:create a xml file
//param:xmlPath:xml文件路径
//return:0,成功;非0,失败
int createXML(const char* xmlPath)
{
    const char* declaration = "<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"no\"?>";
    tinyxml2::XMLDocument doc;
    doc.Parse(declaration);//会覆盖xml所有内容

    //添加申明可以使用如下两行
    //XMLDeclaration* declaration=doc.NewDeclaration();
    //doc.InsertFirstChild(declaration);

    XMLElement* root = doc.NewElement("DBUSER");
    doc.InsertEndChild(root);

    return doc.SaveFile(xmlPath);
}


//获取xml文件所有内容
void getXMLContent(const char* xmlPath, string& content)
{
    tinyxml2::XMLDocument doc;
    if (doc.LoadFile(xmlPath) != XML_SUCCESS)
    {
        cout << "load xml file failed" << endl;
        return;
    }
    XMLPrinter printer;
    doc.Print(&printer);
    content = printer.CStr();
}

//查询xml文件的指定节点
XMLElement* queryUserNodeByName(XMLElement* root, const string& userName)
{
    XMLElement* userNode = root->FirstChildElement("User");
    while (userNode != NULL)
    {
        if (userNode->Attribute("Name") == userName)
        {
            break;
        }
        userNode = userNode->NextSiblingElement();//下一个兄弟节点
    }
    return userNode;
}

void xmltest() {

    //获取xml文件的所有信息
    string content;
    getXMLContent("C:/Users/haonan/source/repos/example.xml", content);
    cout << content << endl;

    //获取文件内容之后,解析字符串
    tinyxml2::XMLDocument doc;
    doc.Parse(content.c_str());
    XMLElement* rootElement = doc.RootElement();
    // const char* rootName = rootElement->Value();

    XMLElement* userNode;
    userNode = queryUserNodeByName(rootElement, "lvlv");
    struct Student s1;//定义结构体类型变量s1
    if (userNode != NULL)
    {
        string gender = userNode->FirstChildElement("Gender")->GetText();
        string mobile = userNode->FirstChildElement("Mobile")->GetText();
        string email = userNode->FirstChildElement("Email")->GetText();

        s1.Gender = atoi(gender.c_str());//结构体变量的引用
        s1.Mobile = (char*)mobile.c_str();
        s1.Email = atoi(email.c_str());
        s1.column = &email;

        cout << s1.Gender << endl;
        cout << s1.Mobile << endl;
        cout << s1.Email << endl;
        cout << s1.column << endl;
    }
}

xxx.h文件

#ifndef SIMPLE_H_INCLUDED
#define SIMPLE_H_INCLUDED

#include <string>
#include <windows.h>
using namespace std;

/*声明结构体*/
typedef struct Student
{
 int Gender;
 char *Mobile;
 int Email;
 void *column;
};

#ifdef __cplusplus
#define EXPORT extern "C" __declspec (dllexport)
#else
#define EXPORT __declspec (dllexport)
#endif // __cplusplus

EXPORT  void xmltest();

#endif // SIMPLE_H_INCLUDED

可能的报错: IntelliSense: “XMLDocument” 不明确

原因: 由于所引用的某些头文件中也使用了 XMLDocument 这个名称,造成命名冲突。如 “windows.h” 也包含了XMLDocument

解决办法:指明命名空间,使用tinyxml2::XMLDocument

JNA调用

@SpringBootApplication
public class DemoApplication {

    public static void main(String[] args) {
        SpringApplication.run(DemoApplication.class);

        DemoApplication.CLibrary.INSTANCE.xmltest();
    }

    public interface CLibrary extends Library {
        // 通过INSTANCE这个常量,就可以获得这个接口的实例
        DemoApplication.CLibrary INSTANCE = (DemoApplication.CLibrary) Native.loadLibrary("Projectxx", DemoApplication.CLibrary.class);

        public void xmltest();
    }

运行结果

使用的example.xml如下

<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<DBUSER>
    <User Name="lvlv" Password ="123456">
        <Gender>1</Gender>
        <Mobile>111</Mobile>
        <Email>222</Email>
    </User>
</DBUSER>

java运行截图

参考链接

https://blog.csdn.net/qq_24127015/article/details/88792741
https://blog.csdn.net/K346K346/article/details/48750417

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
JNA(Java Native Access)是一个用于在Java程序中调用本地代码(如C、C++等)的库。它通过简化调用本地函数的过程,提供了一种简单而又直观的方式来将Java代码与底层的dll文件进行交互。 下面是一个简单的JNA调用dll的示例: 首先,需要下载并导入JNA库。可以从官方网站(https://github.com/java-native-access/jna)上下载最新版本的JNA库。将下载的jar文件导入到你的Java项目中。 接下来,我们先创建一个Java接口,用于定义我们要调用的本地函数,示例代码如下: ```java import com.sun.jna.Library; import com.sun.jna.Native; public interface MyDll extends Library { MyDll INSTANCE = Native.loadLibrary("mydll", MyDll.class); void helloWorld(); } ``` 在上面的代码中,我们定义了一个MyDll接口,它继承了JNA的Library接口。然后,我们使用Native.loadLibrary方法加载我们的dll文件,这里假设我们的dll文件名为"mydll"。 接下来,我们在Java代码中调用dll的函数,示例如下: ```java public class Main { public static void main(String[] args) { MyDll.INSTANCE.helloWorld(); // 调用dll中的helloWorld函数 } } ``` 在上面的代码中,我们通过MyDll.INSTANCE对象来调用dll中定义的helloWorld函数。这样,我们就能够在Java中成功调用dll函数了。 需要注意的是,在实际使用过程中,需要根据dll文件中函数的参数及返回值类型,在接口中定义对应的方法。 通过以上步骤,我们可以成功地使用JNA调用dll。这种方式非常方便,使得Java程序与本地代码的集成更加简单和高效。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

anjushi_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值