您的位置:首页 > 游戏 > 游戏 > 软件定制开发企业_安徽建设工程信息网查询平台蔡庆树_搜索引擎优化工具有哪些_郑州网站优化公司

软件定制开发企业_安徽建设工程信息网查询平台蔡庆树_搜索引擎优化工具有哪些_郑州网站优化公司

2024/12/7 13:42:17 来源:https://blog.csdn.net/aaajj/article/details/143666772  浏览:    关键词:软件定制开发企业_安徽建设工程信息网查询平台蔡庆树_搜索引擎优化工具有哪些_郑州网站优化公司
软件定制开发企业_安徽建设工程信息网查询平台蔡庆树_搜索引擎优化工具有哪些_郑州网站优化公司

在android代码中,经常可以看到native方法,需要查看其对应的C++方法,这些方法是一一对应的,对应关系是在jni注册里关联起来的。

比较直观的是这样的例子,

Parcel.java

//Java层的方法里调用了native方法nativeWriteInt(mNativePtr, val)

    public final void writeInt(int val) {

        nativeWriteInt(mNativePtr, val);

    }

//Java层中声明了native方法

    private static native void nativeWriteInt(long nativePtr, int val);

    private static native void nativeWriteLong(long nativePtr, long val);

以 nativeWriteInt方法为例,进而会调用到frameworks/base/core/jni/android_os_Parcel.cpp中的android_os_Parcel_writeInt方法。

搜nativeWriteInt,可以看到{"nativeWriteInt",            "!(JI)V", (void*)android_os_Parcel_writeInt}

以{"nativeWriteInt",            "!(JI)V", (void*)android_os_Parcel_writeInt}为例,

nativeWriteInt是Java层Parcel.java中声明的函数名称,而android_os_Parcel_writeInt是JNI层android_os_Parcel.cpp中对应的函数。

"!(JI)V"是函数签名,对函数的参数和返回值进行标记,采用了函数签名,即便是重载了的同名函数,都可以通过函数签名来进行区分。

通过这样的映射处理,Java层就和Native层关联起来了。

通过RegisterMethodsOrDie(env, kParcelPathName, gParcelMethods, NELEM(gParcelMethods))调用最终来实现了JNI方法的关联。android.os.Parcel类中的nativeWriteInt方法对应着android_os_Parcel.cpp中的static void android_os_Parcel_writeInt(JNIEnv* env, jclass clazz, jlong nativePtr, jint val)方法。

static void android_os_Parcel_writeInt(JNIEnv* env, jclass clazz, jlong nativePtr, jint val) {

    Parcel* parcel = reinterpret_cast<Parcel*>(nativePtr);

    if (parcel != NULL) {

        const status_t err = parcel->writeInt32(val);

        if (err != NO_ERROR) {

            signalExceptionForError(env, clazz, err);

        }

    }

}

android_os_Parcel_writeInt方法中又调用了C++层的Parcel类,进而通过JNI实现了Java层对C++层方法的调用。

gParcelMethods是JNINativeMethod类型的数组,存储了Java层函数和native函数的映射关系。

还有一种不太直观的描述,

比如libcore/ojluni/src/main/java/java/net/NetworkInterface.java

里的

393    private static NetworkInterface[] getAll() throws SocketException {
394        // Group Ifaddrs by interface name.
395        Map<String, List<StructIfaddrs>> inetMap = new HashMap<>();
396
397        StructIfaddrs[] ifaddrs;
398        try {
399            ifaddrs = Libcore.os.getifaddrs();
400        } catch (ErrnoException e) {
401            throw e.rethrowAsSocketException();
402        }

这里查找getifaddrs方法的实现,找到Libcore.java

继续找到libcore/luni/src/main/java/libcore/io/Linux.java

    public native StructIfaddrs[] getifaddrs() throws ErrnoException;

是个native方法,具体的实现是哪里呢,

可以搜到

libcore/luni/src/main/native/libcore_io_Linux.cpp

   NATIVE_METHOD(Linux, getifaddrs, "()[Landroid/system/StructIfaddrs;"),

这个跟上面的不一样,看不出来对应的方法,

原来,NATIVE_METHOD是个宏,

在libnativehelper/include/nativehelper/JniConstants.h

里有 


82#define NATIVE_METHOD(className, functionName, signature) \
83    { #functionName, signature, reinterpret_cast<void*>(className ## _ ## functionName) }

这个宏里进行了字符串拼接,展开这个宏,

{ "getifaddrs", "()[Landroid/system/StructIfaddrs;", reinterpret_cast<void*>(Linux_getifaddrs) }

 所以,是Linux_getifaddrs方法

1517static jobjectArray Linux_getifaddrs(JNIEnv* env, jobject) {
1518    static jmethodID ctor = env->GetMethodID(JniConstants::structIfaddrs, "<init>",
1519            "(Ljava/lang/String;ILjava/net/InetAddress;Ljava/net/InetAddress;Ljava/net/InetAddress;[B)V");
1520    if (ctor == NULL) {
1521        return NULL;