深入浅析jni中的java接口使用

JNI中的java接口使用

项目需求,需要在c++函数中监听相应的状态,并在java端进行一些列的处理。

这个需要在JNI中写一个subscriber,注册后在需要的地方进行引入使用。

目录结构

初始化是AS上的c++工程文件,这边先暂时实现简单的demo,CdemoActivity是NativeActivity的实现,我们暂时别管,因为实现是c++层控制的,有兴趣可以去百度下

主要涉及jnicallback等c文件和JNIUtil这java文件

JNIUtil

public class JNIUtil {

  static {

    System.loadLibrary("native-event");
  }

  // 注册函数
  public static native void setJniCallBack(JniCallBack callBack);

  // 解注册函数
  public static native void unJniCallBack();

  public interface JniCallBack{

    void onReceiveCInfo();

  }

}

jnicallback.h

//
// Created by 86130 on 2020/9/9.
//

#ifndef CDEMO_JNICALLBACK_H
#define CDEMO_JNICALLBACK_H

#ifdef __cplusplus
extern "C"{
#endif

//方法回调
void onReceiveMsg();

#ifdef __cplusplus
}
#endif
#endif //CDEMO_JNICALLBACK_H

jnicallback.cpp

#include <jni.h>
#include <malloc.h>
#include <cstring>

#include "jnicallback.h"

#ifdef __cplusplus
extern "C"{
#endif

  JavaVM *g_VM;
  jobject subscriber;

extern "C"
JNIEXPORT void JNICALL Java_com_demo_cdemo_JNIUtil_setJniCallBack(JNIEnv* env,jclass clazz, jobject call_back) {

  env->GetJavaVM(&g_VM);
  subscriber = env->NewGlobalRef(call_back);

}

extern "C"
JNIEXPORT void JNICALL Java_com_demo_cdemo_JNIUtil_unJniCallBack(JNIEnv* env, jclass clazz) {

  if (subscriber != NULL)
  {
    env->DeleteGlobalRef(subscriber);
  }
}

JNIEnv *getEnv()
{
  JNIEnv *env;
  if (g_VM ==NULL)
  {
    return NULL;
  }
  int envStat = g_VM->GetEnv((void **) &env, JNI_VERSION_1_6);
  if (envStat == JNI_EDETACHED)
  {
    if (g_VM->AttachCurrentThread(&env, NULL) != 0)
    {
      return NULL;
    }
  }
  return env;
}

jmethodID getMethodIdByNameAndSig(JNIEnv *env, const char *name, const char *sig)
{
  if (env == NULL || subscriber == NULL)
  {
    return NULL;
  }
  jclass subscriberClass = env->GetObjectClass(subscriber);
  if (subscriber == 0)
  {
    return NULL;
  }
  jmethodID methodId = env->GetMethodID(subscriberClass, name, sig);
  if (methodId == 0)
  {
    return NULL;
  }
  return methodId;
}

// 头文件方法实现
void onReceiveMsg()
{
  JNIEnv *env = getEnv();
  jmethodID onReceiveMethodId = getMethodIdByNameAndSig(env, "onReceiveCInfo", "()V");
  if (onReceiveMethodId == NULL)
  {
    return;
  }
  env->CallVoidMethod(subscriber, onReceiveMethodId);

}

#ifdef __cplusplus
}
#endif

在其他的cpp文件中引入jnicallback的头文件就可以使用相应的方法。

CMake文件

project(Native-Activity)

cmake_minimum_required(VERSION 3.4.1)

#引入native_app_glue头文件
include_directories(F:/AndroidSdk/Sdk/ndk-bundle/sources/android/native_app_glue)

add_library(native-activity SHARED
    main.cpp
     jnicallback.cpp)

add_library(native-event SHARED jnicallback.cpp)

find_library(log-lib log)
find_library(OPENGLES3_LIBRARY GLESv3 "OpenGL ES v3.0 library")
find_library(EGL_LIBRARY EGL "EGL 1.4 library")
find_library(android-lib android)

#编译为静态库
add_library(app_glue STATIC
    android_native_app_glue.c)

target_link_libraries(native-event
    ${log-lib} #链接log库
    ${android-lib} #链接android库
    )

target_link_libraries(native-activity
    app_glue #链接静态库native_app_glue
    ${log-lib} #链接log库
    ${android-lib} #链接android库
    ${OPENGLES3_LIBRARY} #链接OpenGLES库
    ${EGL_LIBRARY} #链接EGL库
    )

JNIUtil的使用

package com.demo.cdemo;

import android.app.NativeActivity;
import android.content.Intent;
import android.os.Bundle;
import android.os.Looper;
import android.util.Log;
public class CdemoActivity extends NativeActivity {

  static {
    System.loadLibrary("native-activity");
  }

  boolean isFirst = false;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    JNIUtil.setJniCallBack(new JNIUtil.JniCallBack() {
      @Override
      public void onReceiveCInfo() {
        boolean isMain = isMainThread();
        Log.d("zzkk",
            "CdemoActivity onReceiveCInfo isMain = " + isMain);
        if (!isFirst)
        {
          isFirst = true;
          runOnUiThread(new Runnable() {
            @Override public void run() {
              Intent intent = new Intent(CdemoActivity.this
                  , MainActivity.class);
              startActivity(intent);
            }
          });
        }
      }
    });

  }

  private boolean isMainThread() {
    return Looper.getMainLooper() == Looper.myLooper();
  }

}

可以看见onReceiveCInfo这行日志的打印

综上

到此这篇关于jni中的java接口使用的文章就介绍到这了,更多相关java接口使用内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 解析Java的JNI编程中的对象引用与内存泄漏问题

    JNI,Java Native Interface,是 native code 的编程接口.JNI 使 Java 代码程序可以与 native code 交互--在 Java 程序中调用 native code:在 native code 中嵌入 Java 虚拟机调用 Java 的代码. JNI 编程在软件开发中运用广泛,其优势可以归结为以下几点: 利用 native code 的平台相关性,在平台相关的编程中彰显优势. 对 native code 的代码重用. native code 底层操作

  • Java的JNI快速入门教程(推荐)

    1. JNI简介 JNI是Java Native Interface的英文缩写,意为Java本地接口. 问题来源:由于Java编写底层的应用较难实现,在一些实时性要求非常高的部分Java较难胜任(实时性要求高的地方目前还未涉及,实时性这类话题有待考究). 解决办法:Java使用JNI可以调用现有的本地库(C/C++开发任何和系统相关的程序和类库),极大地灵活Java的开发. 2. JNI快速学习教程 2.1 问题: 使用JNI写一段代码,实现string_Java_Test_helloworld

  • 详解Java 本地接口 JNI 使用方法

    详解Java 本地接口 JNI 使用方法 对于Java程序员来说,Java语言的好处和优点,我想不用我说了,大家自然会说出很多一套套的.但虽然我们作为java程序员,但我们不得不承认java语言也有一些它本身的缺点.比如在性能.和底层打交道方面都有它的缺点.所以java就提供了一些本地接口,他主要的作用就是提供一个标准的方式让java程序通过虚拟机与原生代码进行交互,这也就是我们平常常说的java本地接口(JNI--java native Interface).它使得在 Java 虚拟机 (VM

  • 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工程中使用,完毕. 下面说说具体步骤(含实

  • java jni调用c函数实例分享(java调用c函数)

    从C/C++到Java,再从Java回到C/C++,今天终于有机会了解了连接Java.C/C++的桥梁--JNI.哈哈!分享一下! 一.简介JNI是Java native interface的简写,可以译作Java原生接口.Java可以通过JNI调用C/C++的库,这对于那些对性能要求比较高的Java程序或者Java无法处理的任务无疑是一个很好的方式. 二.目的:Java代码中调用C/C++代码三.实现:假设我们的Java程序为J2C.java, C程序为J2C.c, Java与C之间的通信函数

  • 安卓应用开发通过java调用c++ jni的图文使用方法

    首先建议一个工程 HelloJni如下图: 按照默认的配置下一步,直到完成 . 如下图操作,点击windows菜单->Prefrence菜单: 弹出如下图:选择Andriod ->NDK: 配置完成以后,点击工程属性菜单: 做这一步的目的是,增加对c++代码的支持,他会自动生成一些东西,你会看到多一个jni的文件夹. 这个名字可以默认,就用工程的名字,实际上就是产生的c++代码生成.so文件的名称(windows上的dll文件). 完成以后.可以看代码,生成一个HelloJni的.cpp文件.

  • Ubuntu中为Android HAL编写JNI方法提供JAVA访问硬件服务接口

    在上两篇文章中,我们介绍了如何为Android系统的硬件编写驱动程序,包括如何在Linux内核空间实现内核驱动程序和在用户空间实现硬件抽象层接口.实现这两者的目的是为了向更上一层提供硬件访问接口,即为Android的Application Frameworks层提供硬件服务.我们知道,Android系统的应用程序是用Java语言编写的,而硬件驱动程序是用C语言来实现的,那么,Java接口如何去访问C接口呢?众所周知,Java提供了JNI方法调用,同样,在Android系统中,Java应用程序通过

  • 深入浅析jni中的java接口使用

    JNI中的java接口使用 项目需求,需要在c++函数中监听相应的状态,并在java端进行一些列的处理. 这个需要在JNI中写一个subscriber,注册后在需要的地方进行引入使用. 目录结构 初始化是AS上的c++工程文件,这边先暂时实现简单的demo,CdemoActivity是NativeActivity的实现,我们暂时别管,因为实现是c++层控制的,有兴趣可以去百度下 主要涉及jnicallback等c文件和JNIUtil这java文件 JNIUtil public class JNI

  • 浅析Python 中的 WSGI 接口和 WSGI 服务的运行

    HTTP格式 HTTP GET请求的格式: GET /path HTTP/1.1 Header1: Value1 Header2: Value2 Header3: Value3 每个Header一行一个,换行符是\r\n. HTTP POST请求的格式: POST /path HTTP/1.1 Header1: Value1 Header2: Value2 Header3: Value3 body data goes here... 当遇到连续两个\r\n时,Header部分结束,后面的数据全部

  • 浅析php中抽象类和接口的概念以及区别

    复制代码 代码如下: //抽象类的定义:abstract class ku{  //定义一个抽象类  abstract function kx();  ......}function aa extends ku{  //实现抽象类的方法  function kx(){    echo 'sdsf';  }}//使用方法$aa=new aa;$aa->kx();//1.定义一些方法,子类必须完全实现这个抽象中所有的方法//2.不能从抽象类创建对象,它的意义在于被扩展//3.抽象类通常具有抽象方法,

  • Android在JNI中使用ByteBuffer的方法

    本文实例讲述了Android在JNI中使用ByteBuffer的方法.分享给大家供大家参考.具体如下: 一.ByteBuffer 定义 在NIO中,数据的读写操作始终是与缓冲区相关联的(读取时信道(SocketChannel)将数据读入缓冲区,写入时首先要将发送的数据按顺序填入缓冲区) 缓冲区是定长的,基本上它只是一个列表,它的所有元素都是基本数据类型.ByteBuffer是最常用的缓冲区,它提供了读写其他数据类型的方法,且信道的读写方法只接收ByteBuffer. ByteBuffer有以下几

  • Android系统进程间通信Binder机制在应用程序框架层的Java接口源代码分析

    在前面几篇文章中,我们详细介绍了Android系统进程间通信机制Binder的原理,并且深入分析了系统提供的Binder运行库和驱动程序的源代码.细心的读者会发现,这几篇文章分析的Binder接口都是基于C/C++语言来实现的,但是我们在编写应用程序都是基于Java语言的,那么,我们如何使用Java语言来使用系统的Binder机制来进行进程间通信呢?这就是本文要介绍的Android系统应用程序框架层的用Java语言来实现的Binder接口了. 熟悉Android系统的读者,应该能想到应用程序框架

  • java中的interface接口实例详解

     java中的interface接口实例详解 接口:Java接口是一些方法表征的集合,但是却不会在接口里实现具体的方法. java接口的特点如下: 1.java接口不能被实例化 2.java接口中声明的成员自动被设置为public,所以不存在private成员 3.java接口中不能出现方法的具体实现. 4.实现某个接口就必须要实现里面定义的所有方法. 接下来看一个实现接口的案例: package hello;   interface competer{ //定义接口 void set_comp

  • java中实现Comparable接口实现自定义排序的示例

    实例如下所示: class Student implements Comparable{ String name; int gpa; @Override public int compareTo(Object arg0) { // TODO Auto-generated method stub Student s = (Student)arg0; if(gpa == s.gpa) return name.compareTo(s.name); else if(gpa < s.gpa) return

  • Java中抽象类和接口的区别_动力节点Java学院整理

    接口 1 因为java不支持多重继承,所以有了接口,一个类只能继承一个父类,但可以实现多个接口,接口本身也可以继承多个接口. 2 接口里面的成员变量默认都是public static final类型的.必须被显示的初始化. 3 接口里面的方法默认都是public abstract类型的.隐式声明. 4 接口没有构造方法,不能被实例化. 5 接口不能实现另一个接口,但可以继承多个接口. 6 类如果实现了一个接口,那么必须实现接口里面的所有抽象方法,否则类要被定义为抽象类. 抽象类 1 如果将一个类

  • 在JavaScript中调用Java类和接口的方法

    前言 本文中所有的代码使用 JavaScript 编写,但你也可以用其他兼容 JSR 223 的脚本语言.这些例子可作为脚本文件也可以在交互式 Shell 中一次运行一个语句的方式来运行.在 JavaScript 中访问对象的属性和方法的语法与 Java 语言相同. 本文包含如下几部分: 1.访问 Java 类 为了在 JavaScript 中访问原生类型或者引用 Java 类型,可以调用 Java.type() 函数,该函数根据传入的完整类名返回对应对象的类型.下面代码显示如何获取不同的对象类

随机推荐