当前位置: 首页 > news >正文

C ++代码学习笔记(一)

1、GetStringUTFChars

用于将 Java 字符串(jstring)转换为 UTF-8 编码的 C 风格字符串(const char*)。

必须在使用完后调用 ReleaseStringUTFChars 释放内存,否则可能导致内存泄漏。

std::string data_converter::convertJavaStringToCpp(JNIEnv* env, jstring jStr) {const char* cStr = env->GetStringUTFChars(jStr, nullptr);std::string result(cStr);env->ReleaseStringUTFChars(jStr, cStr);return result;
}

2、NewStringUTF

用于将 C/C++ 的 UTF-8 字符串 转换为 Java 的 jstring 对象。

const char *cStr = "Hello from C++! 你好!";
jstring javaStr = env->NewStringUTF(cStr);

3、GetArrayLength

用于获取 Java 数组的长度。

JNIEXPORT jint JNICALL Java_Example_getArrayLength(JNIEnv *env, jobject obj, jintArray arr) {jsize length = env->GetArrayLength(arr);return length;
}

4、GetObjectArrayElement

用于从 Java 对象数组中获取指定索引处的元素。

对于基本类型数组(如 int[], float[] 等),不能直接使用 GetObjectArrayElement,而应该使用对应的函数如 GetIntArrayElements。

JNIEXPORT void JNICALL Java_Example_printArrayElements(JNIEnv *env, jobject obj, jobjectArray arr) {jsize length = env->GetArrayLength(arr);for (jsize i = 0; i < length; i++) {jobject element = env->GetObjectArrayElement(arr, i);// 对 element 进行操作...}
}

5、GetByteArrayRegion

用于将 Java 字节数组 (jbyteArray) 的一部分或全部复制到 C/C++ 的本地缓冲区中。

JNIEXPORT void JNICALL Java_Example_processByteArray(JNIEnv *env, jobject obj, jbyteArray javaArray) 
{// 获取数组长度jsize length = env->GetArrayLength(javaArray);// 分配本地缓冲区jbyte *buffer = new jbyte[length];// 复制整个数组到本地缓冲区env->GetByteArrayRegion(javaArray, 0, length, buffer);// 处理数据...for (jsize i = 0; i < length; i++) {// 处理 buffer[i]...}// 释放本地缓冲区delete[] buffer;
}

6、(void)env; 和 (void)instance;

这两行是用于 显式忽略未使用参数 的编程技巧。

static void saveDeviceID(JNIEnv* env, jobject instance, jstring devID)
{(void)env;      // 明确表示不使用 env 参数(void)instance; // 明确表示不使用 instance 参数// ...
}

7、GetObjectClass

用于在 C/C++ 代码中获取 Java 对象的类引用。它通常在编写本地方法(native methods)时使用。

#include <jni.h>JNIEXPORT void JNICALL Java_Example_printClassName(JNIEnv *env, jobject obj) {// 获取传入对象的类jclass cls = env->GetObjectClass(obj);// 获取类名jmethodID mid_getName = env->GetMethodID(cls, "getName", "()Ljava/lang/String;");jstring name = (jstring)env->CallObjectMethod(cls, mid_getName);const char *str = env->GetStringUTFChars(name, NULL);printf("Class name: %s\n", str);env->ReleaseStringUTFChars(name, str);
}

替代方案

如果你想获取某个已知类的引用(而不是通过对象实例),可以使用 FindClass() 函数:

jclass cls = env->FindClass("java/lang/String");

GetObjectClass() 主要用于当你已经有一个对象实例但需要访问其类信息时使用。

8、NewGlobalRef

用于创建一个全局引用(global reference)到 Java 对象。全局引用在本地代码中跨多个本地方法调用时保持有效,直到显式释放。

// 全局变量存储引用
jclass globalClassRef;JNIEXPORT void JNICALL Java_Example_init(JNIEnv *env, jobject obj) {// 获取局部引用jclass localClassRef = env->GetObjectClass(obj);// 创建全局引用globalClassRef = (jclass)env->NewGlobalRef(localClassRef);// 局部引用不再需要时可以删除env->DeleteLocalRef(localClassRef);
}JNIEXPORT void JNICALL Java_Example_cleanup(JNIEnv *env, jobject obj) {// 不再需要时释放全局引用if (globalClassRef != NULL) {env->DeleteGlobalRef(globalClassRef);globalClassRef = NULL;}
}

相关函数

  • DeleteGlobalRef(): 释放全局引用

  • NewLocalRef(): 创建局部引用

  • NewWeakGlobalRef(): 创建弱全局引用

9、

    NetworkManagerCallback() = default;virtual ~NetworkManagerCallback() = default;NetworkManagerCallback(const NetworkManagerCallback& other) = delete;NetworkManagerCallback& operator=(const NetworkManagerCallback& other) = delete;

这几行代码展示了一个类中常见的特殊成员函数的声明和定义方式。

  1. NetworkManagerCallback() = default;

    • 使用编译器生成的默认构造函数

    • 示例:当创建NetworkManagerCallback对象时,会调用这个默认构造函数

    NetworkManagerCallback callback; // 使用默认构造函数
  2. virtual ~NetworkManagerCallback() = default;

    • 声明虚析构函数并使用默认实现

    • 使得这个类可以作为基类被继承,且能正确调用派生类的析构函数

    • 示例:

    class DerivedCallback : public NetworkManagerCallback {~DerivedCallback() override { /* 清理资源 */ }
    };
  3. NetworkManagerCallback(const NetworkManagerCallback& other) = delete;

    • 禁用拷贝构造函数

    • 示例:以下代码会编译错误

    NetworkManagerCallback cb1;
    NetworkManagerCallback cb2(cb1); // 错误:拷贝构造函数被删除
  4. NetworkManagerCallback& operator=(const NetworkManagerCallback& other) = delete;

    • 禁用拷贝赋值运算符

    • 示例:以下代码会编译错误

    NetworkManagerCallback cb1;
    NetworkManagerCallback cb2;
    cb2 = cb1; // 错误:拷贝赋值运算符被删除

这种模式通常用于:

  • 需要作为基类但不需要拷贝功能的类(如回调接口)

  • 只允许移动不允许拷贝的类

  • 单例模式实现

http://www.xdnf.cn/news/1342027.html

相关文章:

  • TDengine IDMP 运维指南(常见问题)
  • 日语学习-日语知识点小记-构建基础-JLPT-N3阶段(18):文法+单词第6回1
  • 虚幻基础:曲线
  • 基于STM32单片机的二维码识别物联网OneNet云仓库系统
  • 图--常见面试问题
  • 从源码中学习Java面向对象的多态
  • 多级缓存一致性矩阵:ABP vNext 下的旁路 / 写穿 / 写回组合实战
  • MiniGPT-4
  • FPGA 在情绪识别领域的护理应用(三)
  • 机器学习1
  • 结合 Flutter 和 Rust 的跨平台开发方案
  • Vibe Coding v.s Prompt Engineering
  • 数据库面试常见问题
  • gsplat在windows本地部署
  • Dockerfile
  • Claude Code 已支持【团队版】和【企业版】订阅
  • Webpack的使用
  • 15. 多线程(进阶2) --- CAS 和 多线程常用的类
  • Mokker AI:一键更换照片背景的AI神器
  • 粗粮厂的基于flink的汽车实时数仓解决方案
  • selenium一些进阶方法如何使用
  • K8s快速上手-微服务篇
  • 机器学习中的聚类与集成算法:从基础到应用
  • 前端视频流处理从 0 到 “能跑”:可复制 Demo+WebGL/Worker 优化,覆盖会议 / 直播 / 监控场景
  • 【尝试】在macOS上安装cvat
  • 【51单片机】【protues仿真】基于51单片机水位监测系统
  • 鸿蒙开发中的List组件详解
  • 机器学习-集成算法
  • Django的生命周期
  • 项目1总结其三(图片上传功能)