JDK native源码查看

本文查看的JDK为OpenJDK。
OpenJDK官网

OpenJDK中有很多项目,可点击OpenJDK Projects查看。

回到正题,我们知道,在JDK中,使用Java语言编写的代码,可以通过IDE直接阅读。比如Mac下使用IntelliJ IDEA,查看forName方法,command + 单击

Class.forName("String");

直接进入forName方法中

    @CallerSensitive
    public static Class<?> forName(String className)
                throws ClassNotFoundException {
        Class<?> caller = Reflection.getCallerClass();
        return forName0(className, true, ClassLoader.getClassLoader(caller), caller);
    }

比较纳闷的是,再点击forName0时,发现forName0是个native方法,无法继续往下查看源码。

    /** Called after security check for system loader access checks have been made. */
    private static native Class<?> forName0(String name, boolean initialize,
                                            ClassLoader loader,
                                            Class<?> caller)
        throws ClassNotFoundException;

如何查看native源码呢,继续往下走。

  • 安装Mercurial

Mercurial 是一种轻量级分布式版本控制系统,采用 Python 语言实现,类似Git,SVN。
OpenJDK Projects采用Mercurial管理,因此首先需要安装Mercurial:
https://www.mercurial-scm.org/downloads
下载对应系统的Mercurial,安装成功后,可以通过hg -h查看帮助信息:

$ hg -h
  • clone JDK Project

使用hg clone JDK Project,例如JDK8,这里选择的是jdk8u,读者根据自己的兴趣选择即可

$ hg clone http://hg.openjdk.java.net/jdk8u/jdk8u
$ cd jdk8u
$ sh ./get_source.sh

要吐槽一下,不翻墙,还真搞不定。
有兴趣的读者可以下载云盘的jdk8u源码
jdk8u源码
提取码:bqch

以下是forName0的C源码,对应路径:jdk8u/jdk/src/share/native/java/lang/Class.c


JNIEXPORT jclass JNICALL
Java_java_lang_Class_forName0(JNIEnv *env, jclass this, jstring classname,
                              jboolean initialize, jobject loader, jclass caller)
{
    char *clname;
    jclass cls = 0;
    char buf[128];
    jsize len;
    jsize unicode_len;

    if (classname == NULL) {
        JNU_ThrowNullPointerException(env, 0);
        return 0;
    }

    len = (*env)->GetStringUTFLength(env, classname);
    unicode_len = (*env)->GetStringLength(env, classname);
    if (len >= (jsize)sizeof(buf)) {
        clname = malloc(len + 1);
        if (clname == NULL) {
            JNU_ThrowOutOfMemoryError(env, NULL);
            return NULL;
        }
    } else {
        clname = buf;
    }
    (*env)->GetStringUTFRegion(env, classname, 0, unicode_len, clname);

    if (VerifyFixClassname(clname) == JNI_TRUE) {
        /* slashes present in clname, use name b4 translation for exception */
        (*env)->GetStringUTFRegion(env, classname, 0, unicode_len, clname);
        JNU_ThrowClassNotFoundException(env, clname);
        goto done;
    }

    if (!VerifyClassname(clname, JNI_TRUE)) {  /* expects slashed name */
        JNU_ThrowClassNotFoundException(env, clname);
        goto done;
    }

    cls = JVM_FindClassFromCaller(env, clname, initialize, loader, caller);

 done:
    if (clname != buf) {
        free(clname);
    }
    return cls;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容