2024年Android之JNI开发入门,2024年最新字节跳动开发岗面试

最后

希望大家能有一个好心态,想进什么样的公司要想清楚,并不一定是大公司,我选的也不是特大厂。当然如果你不知道选或是没有规划,那就选大公司!希望我们能先选好想去的公司再投或内推,而不是有一个公司要我我就去!还有就是不要害怕,也不要有压力,平常心对待就行,但准备要充足。最后希望大家都能拿到一份满意的 offer !如果目前有一份工作也请好好珍惜好好努力,找工作其实挺累挺辛苦的。

这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。

由于篇幅有限,这里以图片的形式给大家展示一小部分。

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • javah 命令:生成java代码中本地方法名对应的C语言的函数名

使用方法:javah com.xfhy.jnihelloworld.MainActivity

  • jdk 1.7 项目 src目录下运行javah

  • jdk 1.6 项目 bin目录下 classes文件夹

  • javah native方法声明的java类的全类名

5.jni简便开发流程

===========

  • ① 写java代码 native 声明本地方法

  • ② 添加本地支持 右键单击项目->andorid tools->add native surport

  • 如果发现 finish不能点击需要给工作空间配置ndk目录的位置

  • window->preferences->左侧选择android->ndk 把ndk解压的目录指定进来

  • ③ 如果写的是.c的文件 先修改一下生成的.cpp文件的扩展名 不要忘了 相应修改Android.mk文件中LOCAL_SRC_FILES的值

  • ④ javah生成头文件 在生成的头文件中拷贝c的函数名到.c的文件

  • ⑤ 解决CDT插件报错的问题

  • 右键单击项目选择 properties 选测 c/c++ general->paths and symbols->include选项卡下->点击add…->file system 选择ndk目录下 platforms文件夹 对应平台下(项目支持的最小版本)

usr 目录下 arch-arm -> include 确定后 会解决代码提示和报错的问题

  • ⑥编写C函数 如果需要单独编译一下c代码就在c/c++视图中找到小锤子,点一下小锤子就编译了

  • 如果想直接运行到模拟器上 就不用锤子了,直接右键run as,然后就自动编译了

  • ⑦ java代码中不要忘了 System.loadlibrary();

static {

System.loadLibrary(“hello”); //加载动态链接库

}

6. Java向C传递一些基本的类型,处理

======================

将一个jstring转换成一个c语言的char* 类型工具方法

char* _JString2CStr(JNIEnv* env, jstring jstr) {

char* rtn = NULL;

jclass clsstring = (*env)->FindClass(env, “java/lang/String”);

jstring strencode = (*env)->NewStringUTF(env,“GB2312”);

jmethodID mid = (*env)->GetMethodID(env, clsstring, “getBytes”, “(Ljava/lang/String;)[B”);

jbyteArray barr = (jbyteArray)(*env)->CallObjectMethod(env, jstr, mid, strencode); // String .getByte(“GB2312”);

jsize alen = (*env)->GetArrayLength(env, barr);

jbyte* ba = (*env)->GetByteArrayElements(env, barr, JNI_FALSE);

if(alen > 0) {

rtn = (char*)malloc(alen+1); //“\0”

memcpy(rtn, ba, alen);

rtn[alen]=0;

}

(*env)->ReleaseByteArrayElements(env, barr, ba,0);

return rtn;

}

int类型

JNIEXPORT jint JNICALL Java_com_xfhy_javapassdata_JNI_add

(JNIEnv * env, jobject clazz, jint x, jint y){

return x+y; //直接返回x+y

}

String类型

JNIEXPORT jstring JNICALL Java_com_xfhy_javapassdata_JNI_sayHelloInC

(JNIEnv *env, jobject clazz, jstring str){

//将jstring转换成char* 类型

char* cstr = _JString2CStr(env,str);

//调用C语言的strlen测量cstr字符串的长度

int length = strlen(cstr);

int i=0;

for(i=0; i<length; i++){

*(cstr+i) += 1; //将字符串+1

}

return (env)->NewStringUTF(env,cstr); //将char 类型转换成String类型返回

}

int[]类型

JNIEXPORT jintArray JNICALL Java_com_xfhy_javapassdata_JNI_arrElementsIncrease

(JNIEnv *env, jobject clazz, jintArray jArray) {

//jsize (GetArrayLength)(JNIEnv, jarray); 返回数组长度

int length = (*env)->GetArrayLength(env,jArray);

//jint* (GetIntArrayElements)(JNIEnv, jintArray, jboolean*); 最后一个参数表示是否拷贝,可以不用传值

//返回int* 返回该数组的首地址 这样就可以直接通过该指针直接操作该数组了

int* cArray = (*env)->GetIntArrayElements(env,jArray,NULL);

int i;

for(i=0; i<length; i++) {

*(cArray+i) += 10;

}

return jArray; //直接将原数组返回(这时已经是修改过了的)

}

7.C代码中向logcat输出内容

=================

1.Android.mk文件增加以下内容

LOCAL_LDLIBS += -llog

2.C代码中增加以下内容

#include <android/log.h>

#define LOG_TAG “xfhy”

#define LOGD(…) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, VA_ARGS)

#define LOGI(…) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, VA_ARGS)

  • define C的宏定义 起别名 #define LOG_TAG “xfhy” 给”xfhy”起别名LOG_TAG

  • #define LOGI(…) android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS),ANDROID_LOG_DEBUG表示优先级 debug ANDROID_LOG_INFO表示info 这些在log.h中可以看到

  • 给 __android_log_print函数起别名 写死了前两个参数 第一个参数 优先级 第二个参数TAG

  • VA_ARGS:是可变参数的固定写法

  • LOGI(…)在调用的时候 用法跟printf()一样

8. C代码回调java方法

===============

首先需要了解:

Java反射

public class Demo {

public static void main(String[] args) {

//1.获取字节码对象

Class util = Utils.class;

try {

//2.获取Method对象 方法名,参数类型

Method method = util.getMethod(“test”, String.class);

//3.通过字节码对象创建一个Obejct

Object obj = util.newInstance();

//4.通过对象调用方法 对带有指定参数的指定对象调用由此 Method 对象表示的底层方法

method.invoke(obj, “hello”);

} catch (Exception e) {

e.printStackTrace();

}

}

public class Utils {

public void test(String string) {

System.out.println(string);

}

}

}

C代码回调java方法

  • ① 找到字节码对象

  • //jclass (FindClass)(JNIEnv, const char*);

  • //第二个参数 要回调的java方法所在的类的路径 “com/itheima/callbackjava/JNI”

  • ② 通过字节码对象找到方法对象

  • //jmethodID (GetMethodID)(JNIEnv, jclass, const char*, const char*);

  • 第二个参数 字节码对象 第三个参数 要反射调用的java方法名 第四个参数 要反射调用的java方法签名

  • javap -s 要获取方法签名的类的全类名 项目/bin/classes 运行javap

  • ③ 通过字节码创建 java对象(可选) 如果本地方法和要回调的java方法在同一个类里,可以直接用 jni传过来的java对象 调用创建的Method

  • jobject obj =(*env)->AllocObject(env,claz);

  • 当回调的方法跟本地方法不在一个类里 需要通过刚创建的字节码对象手动创建一个java对象

  • 再通过这个对象来回调java的方法

  • 需要注意的是 如果创建的是一个activity对象 回调的方法还包含上下文 这个方法行不通!!!回报空指针异常

  • ④ 反射调用java方法

  • //void (CallVoidMethod)(JNIEnv, jobject, jmethodID, …);

  • 第二个参数 调用java方法的对象 第三个参数 要调用的jmethodID对象 可选的参数 调用方法时接收的参数

/*

  • Class: com_xfhy_callbackjava_JNI java里的类的完整路径

  • Method: callbackvoidmethod java里面的方法名

  • Signature: ()V 方法签名

*/

JNIEXPORT void JNICALL Java_com_xfhy_callbackjava_JNI_callbackvoidmethod

(JNIEnv *env, jobject clazz) {

//1.找到字节码对象

//jclass (FindClass)(JNIEnv, const char*);

//参数: env,需要反射的对象的全路径

jclass claz = (*env)->FindClass(env,“com/xfhy/callbackjava/JNI”);

//2.通过字节码对象找到方法对象

//jmethodID (GetMethodID)(JNIEnv, jclass, const char*, const char*);

//第二个参数 字节码对象 第三个参数 要反射调用的java方法名 第四个参数 要反射调用的java方法签名

//javap -s 要获取方法签名的类的全类名 项目/bin/classes 运行javap

jmethodID methodID = (*env)->GetMethodID(env,claz,“helloFromJava”,“()V”);

//3.通过字节码创建java对象(可选),如果本地方法和要调用的java方法在同一个类里,可以直接用jni传过来的java对象,调用创建的Method

//jobject obj =(*env)->AllocObject(env,claz);

//* 当回调的方法跟本地方法不在一个类里 需要通过刚创建的字节码对象手动创建一个java对象

//* 再通过这个对象来回调java的方法

//* 需要注意的是, 如果创建的是一个activity对象, 回调的方法还包含上下文 ,这个方法行不通!!!回报空指针异常

分享读者

作者2013年java转到Android开发,在小厂待过,也去过华为,OPPO等大厂待过,18年四月份进了阿里一直到现在。

被人面试过,也面试过很多人。深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长,而且极易碰到天花板技术停滞不前!

我们整理了一份阿里P7级别的Android架构师全套学习资料,特别适合有3-5年以上经验的小伙伴深入学习提升。

主要包括阿里,以及字节跳动,腾讯,华为,小米,等一线互联网公司主流架构技术。如果你有需要,尽管拿走好了。

35岁中年危机大多是因为被短期的利益牵着走,过早压榨掉了价值,如果能一开始就树立一个正确的长远的职业规划。35岁后的你只会比周围的人更值钱。

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

滞不前!

我们整理了一份阿里P7级别的Android架构师全套学习资料,特别适合有3-5年以上经验的小伙伴深入学习提升。

主要包括阿里,以及字节跳动,腾讯,华为,小米,等一线互联网公司主流架构技术。如果你有需要,尽管拿走好了。

[外链图片转存中…(img-TyryUZea-1714817599000)]

35岁中年危机大多是因为被短期的利益牵着走,过早压榨掉了价值,如果能一开始就树立一个正确的长远的职业规划。35岁后的你只会比周围的人更值钱。

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化学习资料的朋友,可以戳这里获取

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 21
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值