android nativate 动态注册 静态注册
说明:在java函数的入口比较容易分析,
把activity的生命周期或者关键函数通过放在so层,分析起来就困难多了
1、在MainActivity中
package com.demo.nativate; import androidx.appcompat.app.AppCompatActivity; import android.os.Bundle;
import android.widget.TextView; import com.demo.nativate.databinding.ActivityMainBinding; public class MainActivity extends AppCompatActivity { // Used to load the 'nativate' library on application startup.
static {
System.loadLibrary("nativate");
} private ActivityMainBinding binding; // @Override
protected native void onCreate(Bundle savedInstanceState);
// {
// super.onCreate(savedInstanceState);
// setContentView(R.layout.activity_main);
// // Example of a call to a native method
// TextView tv = findViewById(R.id.sample_text);
// tv.setText(stringFromJNI());
// } /**
* A native method that is implemented by the 'nativate' native library,
* which is packaged with this application.
*/
public native String stringFromJNI();
}
2、静态注册
#include <jni.h>
#include <string>
#include <android/log.h> #define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, "Tag->", __VA_ARGS__) extern "C" JNIEXPORT jstring JNICALL
Java_com_demo_nativate_MainActivity_stringFromJNI(
JNIEnv *env,
jobject /* this */) {
std::string hello = "Hello from C++";
return env->NewStringUTF(hello.c_str());
}
extern "C"
JNIEXPORT void JNICALL
Java_com_demo_nativate_MainActivity_onCreate(JNIEnv *env, jobject thiz,
jobject saved_instance_state) {
// TODO: implement onCreate()
LOGD("静态注册 MainActivity OnCreate be called!");
//super.onCreate(savedInstanceState);
jclass MainActivityClass = env->GetObjectClass(thiz);
jclass AppCompatActivityClass = env->GetSuperclass(MainActivityClass);
jmethodID onCreate = env->GetMethodID(AppCompatActivityClass, "onCreate",
"(Landroid/os/Bundle;)V");
env->CallNonvirtualVoidMethod(thiz, AppCompatActivityClass, onCreate,
saved_instance_state); //调用父类方法
LOGD("静态注册 super.onCreate(savedInstanceState) be called!");
// setContentView(R.layout.activity_main);
jmethodID setContentView_id = env->GetMethodID(MainActivityClass, "setContentView", "(I)V");
jclass layoutClass = env->FindClass("com/demo/nativate/R$layout");
jfieldID activity_main_id = env->GetStaticFieldID(layoutClass, "activity_main", "I");
jint id_txt_main = env->GetStaticIntField(layoutClass, activity_main_id);
env->CallVoidMethod(thiz, setContentView_id, id_txt_main);
LOGD("静态注册 setContentView(R.layout.activity_main); be called!");
// TextView tv = findViewById(R.id.sample_textFId);
jmethodID findViewById_mid = env->GetMethodID(MainActivityClass, "findViewById",
"(I)Landroid/view/View;");
jclass RIdClass = env->FindClass("com/demo/nativate/R$id");
jfieldID sample_textFId = env->GetStaticFieldID(RIdClass, "sample_text", "I");
jint sample_textF = env->GetStaticIntField(RIdClass, sample_textFId);
LOGD("静态注册 TextView tv = findViewById(R.id.sample_textFId); be called!");
jobject tvobject = env->CallObjectMethod(thiz, findViewById_mid, sample_textF);
// tv.setText(stringFromJNI());
jclass tvClass = env->GetObjectClass(tvobject);
jmethodID setText_mid = env->GetMethodID(tvClass, "setText", "(Ljava/lang/CharSequence;)V");
jstring text = env->NewStringUTF("onCreate-->Native success by c++ !!!");
env->CallVoidMethod(tvClass, setText_mid, text);
env->DeleteLocalRef(text);
LOGD("静态注册 tv.setText(stringFromJNI());; be called!");
}
3、动态注册
#include <jni.h>
#include <string>
#include <android/log.h> #define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG, "Tag->", __VA_ARGS__) extern "C" JNIEXPORT jstring JNICALL
Java_com_demo_nativate_MainActivity_stringFromJNI(
JNIEnv *env,
jobject /* this */) {
std::string hello = "Hello from C++";
return env->NewStringUTF(hello.c_str());
}
extern "C"
JNIEXPORT void JNICALL
Java_com_demo_nativate_MainActivity_onCreate(JNIEnv *env, jobject thiz,
jobject saved_instance_state) {
// TODO: implement onCreate()
LOGD("静态注册 MainActivity OnCreate be called!");
//super.onCreate(savedInstanceState);
jclass MainActivityClass = env->GetObjectClass(thiz);
jclass AppCompatActivityClass = env->GetSuperclass(MainActivityClass);
jmethodID onCreate = env->GetMethodID(AppCompatActivityClass, "onCreate",
"(Landroid/os/Bundle;)V");
env->CallNonvirtualVoidMethod(thiz, AppCompatActivityClass, onCreate,
saved_instance_state); //调用父类方法
LOGD("静态注册 super.onCreate(savedInstanceState) be called!");
// setContentView(R.layout.activity_main);
jmethodID setContentView_id = env->GetMethodID(MainActivityClass, "setContentView", "(I)V");
jclass layoutClass = env->FindClass("com/demo/nativate/R$layout");
jfieldID activity_main_id = env->GetStaticFieldID(layoutClass, "activity_main", "I");
jint id_txt_main = env->GetStaticIntField(layoutClass, activity_main_id);
env->CallVoidMethod(thiz, setContentView_id, id_txt_main);
LOGD("静态注册 setContentView(R.layout.activity_main); be called!");
// TextView tv = findViewById(R.id.sample_textFId);
jmethodID findViewById_mid = env->GetMethodID(MainActivityClass, "findViewById",
"(I)Landroid/view/View;");
jclass RIdClass = env->FindClass("com/demo/nativate/R$id");
jfieldID sample_textFId = env->GetStaticFieldID(RIdClass, "sample_text", "I");
jint sample_textF = env->GetStaticIntField(RIdClass, sample_textFId);
LOGD("静态注册 TextView tv = findViewById(R.id.sample_textFId); be called!");
jobject tvobject = env->CallObjectMethod(thiz, findViewById_mid, sample_textF);
// tv.setText(stringFromJNI());
jclass tvClass = env->GetObjectClass(tvobject);
jmethodID setText_mid = env->GetMethodID(tvClass, "setText", "(Ljava/lang/CharSequence;)V");
jstring text = env->NewStringUTF("onCreate-->Native success by c++ !!!");
env->CallVoidMethod(tvClass, setText_mid, text);
env->DeleteLocalRef(text);
LOGD("静态注册 tv.setText(stringFromJNI());; be called!");
} void dynamicRegister(JNIEnv *env, jobject thiz, jobject saved_instance_state) {
LOGD("动态注册 MainActivity OnCreate be called!");
//super.onCreate(savedInstanceState);
jclass MainActivityClass = env->GetObjectClass(thiz);
jclass AppCompatActivityClass = env->GetSuperclass(MainActivityClass);
jmethodID onCreate = env->GetMethodID(AppCompatActivityClass, "onCreate",
"(Landroid/os/Bundle;)V");
env->CallNonvirtualVoidMethod(thiz, AppCompatActivityClass, onCreate,
saved_instance_state); //调用父类方法
LOGD("动态注册 super.onCreate(savedInstanceState) be called!");
// setContentView(R.layout.activity_main);
jmethodID setContentView_id = env->GetMethodID(MainActivityClass, "setContentView", "(I)V");
jclass layoutClass = env->FindClass("com/demo/nativate/R$layout");
jfieldID activity_main_id = env->GetStaticFieldID(layoutClass, "activity_main", "I");
jint id_txt_main = env->GetStaticIntField(layoutClass, activity_main_id);
env->CallVoidMethod(thiz, setContentView_id, id_txt_main);
LOGD("动态注册 setContentView(R.layout.activity_main); be called!");
// TextView tv = findViewById(R.id.sample_textFId);
jmethodID findViewById_mid = env->GetMethodID(MainActivityClass, "findViewById",
"(I)Landroid/view/View;");
jclass RIdClass = env->FindClass("com/demo/nativate/R$id");
jfieldID sample_textFId = env->GetStaticFieldID(RIdClass, "sample_text", "I");
jint sample_textF = env->GetStaticIntField(RIdClass, sample_textFId);
LOGD("动态注册 TextView tv = findViewById(R.id.sample_textFId); be called!");
jobject tvobject = env->CallObjectMethod(thiz, findViewById_mid, sample_textF);
// tv.setText(stringFromJNI());
jclass tvClass = env->GetObjectClass(tvobject);
jmethodID setText_mid = env->GetMethodID(tvClass, "setText", "(Ljava/lang/CharSequence;)V");
jstring text = env->NewStringUTF("onCreate-->Native success by c++ !!!");
env->CallVoidMethod(tvClass, setText_mid, text);
env->DeleteLocalRef(text);
LOGD("动态注册 tv.setText(stringFromJNI());; be called!"); } static int registerNativates(JNIEnv *env) {
jclass mainActivityClass = env->FindClass("com/demo/nativate/MainActivity");
if (mainActivityClass == nullptr) {
return JNI_FALSE;
}
JNINativeMethod methods[] = {
{"onCreate", "(Landroid/os/Bundle;)V", (void *) dynamicRegister}
};
int methodsNum = sizeof(methods) / sizeof(methods[0]);
if (env->RegisterNatives(mainActivityClass, methods, methodsNum) < 0) {
return JNI_FALSE;
}
return JNI_TRUE;
} jint JNI_OnLoad(JavaVM *vm, void *reserved) {
LOGD("调用的 JNI_Onload函数");
//获得JNI环境
JNIEnv *env = nullptr;
if ((vm->GetEnv(reinterpret_cast<void **>(&env), JNI_VERSION_1_6)) != JNI_OK) {
return -1;
} else {
if (!registerNativates(env)) {
return -1;
}
}
return JNI_VERSION_1_6;
}
4、小结
当动静态注册同时存在时,会使用动态注册。
在android studio中,代码会出现红色的显示,不用担心是正常现象。
android nativate 动态注册 静态注册的更多相关文章
- Android中的BroadCast静态注册与动态注册
1.静态注册 新建MyBroadcast类继承BroadcastReceiver,实现onReceive方法 /** * Author:JsonLu * DateTime:2015/9/21 16:4 ...
- BroadcastReceiver(广播)的静态注册和动态注册 --Android开发
BroadcastReceiver是安卓四大组件之一,本例通过代码的方式演示静态注册和动态注册. 1.静态注册 静态注册只需要AndroidManifest.xml中进行配置: AndroidMani ...
- Oracle监听静态注册和动态注册
静态注册和动态注册总结 一.什么是注册? 注册就是将数据库作为一个服务注册到监听程序.客户端不需要知道数据库名和实例名,只需要知道该数据库对外提供的服务名就可以申请连接到数据库.这个服务名可能与实例名 ...
- Oracle监听器—静态注册
注册就是将数据库作为一个服务注册到监听程序.客户端不需要知道数据库名和实例名,只需要知道该数据库对外提供的服务名就可以申请连接到数据库.这个服务名可能与实例名一样,也有可能不一样. 注册分: 1. 静 ...
- BroadcastReceiver的两种注册方式之------静态注册
activity_main.xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android&qu ...
- Android开发4: Notification编程基础、Broadcast的使用及其静态注册、动态注册方式
前言 啦啦啦~(博主每次开篇都要卖个萌,大家是不是都厌倦了呢~) 本篇博文希望帮助大家掌握 Broadcast 编程基础,实现动态注册 Broadcast 和静态注册 Broadcast 的方式以及学 ...
- Android(java)学习笔记173:BroadcastReceiver之 静态注册 和 动态注册
1. 广播接受者>什么是广播.收音机.电台:对外发送信号.收音机:接收电台的信号. >在android系统里面,系统有很多重要的事件: 电池电量低,插入充电器,sd卡被移除,有电话打出去, ...
- Android(java)学习笔记116:BroadcastReceiver之 静态注册 和 动态注册
1. 广播接受者>什么是广播.收音机.电台:对外发送信号.收音机:接收电台的信号. >在android系统里面,系统有很多重要的事件: 电池电量低,插入充电器,sd卡被移除,有电话打出去, ...
- Android只能动态注册的广播Action
只能动态注册的广播(部分): android.intent.action.SCREEN_ON android.intent.action.SCREEN_OFF android.intent.actio ...
- Android实现AppWidget、Broadcast静态注册
Android实现AppWidget.Broadcast静态注册 本篇博客是基于我上一篇博客继续修改的,详情请看Android实现AppWidget.Broadcast动态注册 开发工具:Andori ...
随机推荐
- Linux常用基础指令
Linux常用指令 一.基础命令 whoami查看当前用户 pwd查看当前所在位置 ls 查看当前文件夹的内容 ls -l或ll显示详细内容 cd 绝对路径:从根目录开始的路径 cd / 文件夹 返回 ...
- 在图片不被裁剪时opencv绕图片中任意点旋转任意角度
opencv绕图片中任意角度旋转任意角度 最近在做项目需要把把图片绕图片中任意点旋转任意角度,考虑到自己旋转需要编写插值算法,所以想到了用opencv,但是网上都是围绕图片中点旋转任意角度的,都是 ...
- Spring Cloud Consul 入门指引
1 概述 Spring Cloud Consul 项目为 Spring Boot 应用程序提供了与 Consul 的轻松集成. Consul 是一个工具,它提供组件来解决微服务架构中一些最常见的挑战: ...
- 标题,ico动态化
//获取ico元素 var link = document.querySelector("link[rel*='icon']"); link.href = "image/ ...
- Activate MFA报错:MFADevice entity at the same path and name already exists
MFA即:Multi-factor authentication (MFA) 今天在为自己账号Activate MFA时报错,如下图所示: Entity already exists This ent ...
- docker搭建yapi接口文档系统、Idea中上传接口、在线调用
一.前言 在我们后端开发中,必不可少的是接口的交接,有很多种方式,常见的就是swagger,不过这个侵入性太强了.还有就是接口文档的框架,比如今天小编带大家一起搭建的yapi,在公司还是挺常见的! 今 ...
- windows下利用_popen,_wpoen创建管道进行系统命令输出数据
转载: https://blog.csdn.net/greless/article/details/72383762 参考: http://www.linuxidc.com/Linux/2011-04 ...
- ASP.NET Core 中的模型绑定
微软官方文档:ASP.NET Core 中的模型绑定 Route 是通过MVC Route URL取值. 如:http://localhost:5000/Home/Index/2,id取出的值就会是2 ...
- 我说HashMap初始容量是16,面试官让我回去等通知
众所周知HashMap是工作和面试中最常遇到的数据类型,但很多人对HashMap的知识止步于会用的程度,对它的底层实现原理一知半解,了解过很多HashMap的知识点,却都是散乱不成体系,今天一灯带你一 ...
- HTTP缺点有哪些,如何解决
前言 大家好,我是蜗牛,在上一篇中,我们介绍了不同版本的HTTP区别和发展背景,这篇文章我们来聊聊HTTP的缺点,HTTP缺点大致总结有以下三点: 通信使用明文(不加密),内容可能会被窃听. 不验证通 ...