欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

OpenJDK源碼解析之System.out.println詳解

 更新時(shí)間:2021年04月25日 16:02:58   作者:黃智霖-blog  
這篇文章主要介紹了OpenJDK源碼解析之System.out.println詳解,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java的小伙伴們有非常好的幫助,需要的朋友可以參考下

一、前戲

可能不少小伙伴習(xí)慣在代碼中使用sout打印一些信息,就像這樣:

System.out.println("hello world!")

做為一位資深干碼人,本著弘揚(yáng)黨求真務(wù)實(shí)的精神,必須得來看看這個(gè)sout有何玄機(jī)~~

首先看調(diào)用就知道,out是System類的一個(gè)公共靜態(tài)成員變量,進(jìn)入System.java中:

public final static PrintStream out = null;

嗯,不止是public,還是final的。不管,來找找out是在哪里賦值的。。。。。。日嘛找半天沒找到?那就試試直接在類中搜索: out = ,結(jié)果如下:

在這里插入圖片描述

完?duì)僮?,整個(gè)System類一共將近1300行的代碼,只找到一個(gè)和out賦值相關(guān)的,還是啥子局部變量fdOut,看起來和out屬性沒有什么關(guān)聯(lián)啊~ 往上滑滑看看這個(gè)是什么方法:

private static void initializeSystemClass() {
	......
	FileInputStream fdIn = new FileInputStream(FileDescriptor.in);
	FileOutputStream fdOut = new FileOutputStream(FileDescriptor.out);
	FileOutputStream fdErr = new FileOutputStream(FileDescriptor.err);
	setIn0(new BufferedInputStream(fdIn));
	setOut0(newPrintStream(fdOut, props.getProperty("sun.stdout.encoding")));
	setErr0(newPrintStream(fdErr, props.getProperty("sun.stderr.encoding")));
	......
}

原來如此,看到initializeSystemClass方法,似乎一切都明了了~~

二、JVM源碼分析

initializeSystemClass不是給我們調(diào)用的,這個(gè)方法會(huì)在vm線程初始化后被虛擬機(jī)調(diào)用。其定義在thread.cpp中:

static void call_initializeSystemClass(TRAPS) {
  Klass* k =  SystemDictionary::resolve_or_fail(vmSymbols::java_lang_System(), true, CHECK);
  instanceKlassHandle klass (THREAD, k);

  JavaValue result(T_VOID);
  JavaCalls::call_static(&result, klass, vmSymbols::initializeSystemClass_name(),
                                         vmSymbols::void_method_signature(), CHECK);
}

首先獲取Klass(類元信息在虛擬機(jī)中的結(jié)構(gòu)表示,就是一個(gè)c++中的類),在vmSymbols.hpp中找找java_lang_System對(duì)應(yīng)的值:

 template(java_lang_System,                          "java/lang/System") 

可以看到代表的就是java.lang.System類,同時(shí),initializeSystemClass_name也定義在vmSymbols.hpp中:

template(initializeSystemClass_name,                "initializeSystemClass")   

這里使用JavaCalls::call_static就是調(diào)用java.lang.System的靜態(tài)方法initializeSystemClass。還要再啰嗦一句,call_initializeSystemClass是在何處調(diào)用的?我們回到thread.cpp中,進(jìn)入Threads::create_vm方法,在其中找到了call_initializeSystemClass方法的調(diào)用:

jint Threads::create_vm(JavaVMInitArgs* args, bool* canTryAgain) {
	......
	 call_initializeSystemClass(CHECK_0);
	......
}

然后看看Threads::create_vm是在何處調(diào)用的,我們進(jìn)入jni.cpp,找到JNI_CreateJavaVM方法:

_JNI_IMPORT_OR_EXPORT_ jint JNICALL JNI_CreateJavaVM(JavaVM **vm, void **penv, void *args) {
	......
	result = Threads::create_vm((JavaVMInitArgs*) args, &can_try_again);
	if (result == JNI_OK) {
		......
	} else {
		......	
	}
	......
}

現(xiàn)在已經(jīng)知道了,JVM啟動(dòng)后會(huì)在初始化JVM的時(shí)候調(diào)用CreateJavaVM,進(jìn)而調(diào)用initializeSystemClass方法。但是out屬性是如何設(shè)置的呢?我們回到j(luò)ava.lang.System#initializeSystemClass方法:

private static void initializeSystemClass() {
	......
	FileOutputStream fdOut = new FileOutputStream(FileDescriptor.out);
	setOut0(newPrintStream(fdOut, props.getProperty("sun.stdout.encoding")));
	......
}

private static PrintStream newPrintStream(FileOutputStream fos, String enc) {
       if (enc != null) {
            try {
                return new PrintStream(new BufferedOutputStream(fos, 128), true, enc);
            } catch (UnsupportedEncodingException uee) {}
        }
        return new PrintStream(new BufferedOutputStream(fos, 128), true);
    }

這個(gè)方法中唯一和out相關(guān)的就是這個(gè)setOut0方法調(diào)用了,我們來看看這個(gè)方法:

private static native void setOut0(PrintStream out);

嗯,這個(gè)是一個(gè)native方法,沒辦法,又只有回到JVM源碼了。怎么找呢?首選組裝一下本地方法名,根據(jù)規(guī)則setOut0對(duì)應(yīng)的本地方法應(yīng)該叫:Java_java_lang_System_setOut0,我們到源碼中找找,然后在System.c中找到了該方法

JNIEXPORT void JNICALL
Java_java_lang_System_setOut0(JNIEnv *env, jclass cla, jobject stream)
{
    jfieldID fid =
        (*env)->GetStaticFieldID(env,cla,"out","Ljava/io/PrintStream;");
    if (fid == 0)
        return;
    (*env)->SetStaticObjectField(env,cla,fid,stream);
}

首先獲取了java.io.PrintStread類型的out靜態(tài)成員,嗯,這個(gè)就是我們java.lang.System類的靜態(tài)成員out:

public final static PrintStream out;

然后將stream參數(shù)賦值給它,這個(gè)stream就是initializeSystemClass方法中通過newPrintStream方法創(chuàng)建的PrintStream 對(duì)象。到現(xiàn)在我們已經(jīng)明白了,out原來是這樣賦值的,真麻煩~

趁熱打鐵,弄明白了out,接下來看看println。既然out是PrintStream對(duì)象,那么到PrintStream中看看println方法:

public void println(String x) {
        synchronized (this) {
            print(x);
            newLine();
        }
    }

Java代碼看起來是要親切多了,但是這個(gè)synchronized是個(gè)什么鬼???

三、坑?

前面我們發(fā)現(xiàn)println方法竟然有個(gè)synchronized關(guān)鍵字,經(jīng)常在項(xiàng)目中使用sout的小伙伴會(huì)不會(huì)感覺腦袋嗡嗡的?為什么要嗡嗡?不要忘記我們前面通過JVM源碼跟蹤的System.out的賦值過程,這個(gè)out可是單例!你個(gè)加了synchronized(this)的虛方法,還是單例的,如果在系統(tǒng)中進(jìn)行并發(fā)使用,后果不用我多說吧?

那可能有人就要說了,那我不用println(“hello world!”)了嘛,我換成print總行了吧?

System.out.print("hello world!");
System.out.print("\n");

因?yàn)閜rint方法好像沒有加鎖啊:

public void print(String s) {
        if (s == null) {
            s = "null";
        }
        write(s);
    }

那我們看看write方法,不好意思:

private void write(String s) {
        try {
            synchronized (this) {
                ensureOpen();
                textOut.write(s);
                textOut.flushBuffer();
                charOut.flushBuffer();
                if (autoFlush && (s.indexOf('\n') >= 0))
                    out.flush();
            }
        }
		......
    }

四、總結(jié)

不知道有沒有小伙伴經(jīng)常在項(xiàng)目中使用System.out.println來輸出"日志"?千萬不要亂用喲,不然說不定哪天就被out了~~~

到此這篇關(guān)于OpenJDK源碼解析之System.out.println詳解的文章就介紹到這了,更多相關(guān)OpenJDK源碼解析內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • SpringMVC數(shù)據(jù)響應(yīng)詳細(xì)介紹

    SpringMVC數(shù)據(jù)響應(yīng)詳細(xì)介紹

    Spring MVC 是 Spring 提供的一個(gè)基于 MVC 設(shè)計(jì)模式的輕量級(jí) Web 開發(fā)框架,本質(zhì)上相當(dāng)于 Servlet,Spring MVC 角色劃分清晰,分工明細(xì),本章來講解SpringMVC數(shù)據(jù)響應(yīng)
    2023-02-02
  • java自定義任務(wù)類定時(shí)執(zhí)行任務(wù)示例 callable和future接口使用方法

    java自定義任務(wù)類定時(shí)執(zhí)行任務(wù)示例 callable和future接口使用方法

    Callable是類似于Runnable的接口,實(shí)現(xiàn)Callable接口的類和實(shí)現(xiàn)Runnable的類都是可被其它線程執(zhí)行的任務(wù)
    2014-01-01
  • JAVA socket.io注解原理及用法圖解

    JAVA socket.io注解原理及用法圖解

    這篇文章主要介紹了JAVA socket.io注解原理及用法圖解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-06-06
  • Java中的Optional類詳細(xì)解讀

    Java中的Optional類詳細(xì)解讀

    這篇文章主要介紹了Java中的Optional類詳細(xì)解讀,Optional是Java中的一個(gè)類,它的作用是用于解決空指針異常的問題,它提供了一些有用的方法,可以幫助我們避免顯式進(jìn)行空值檢測(cè),需要的朋友可以參考下
    2023-08-08
  • java實(shí)現(xiàn)兩個(gè)線程交替打印的實(shí)例代碼

    java實(shí)現(xiàn)兩個(gè)線程交替打印的實(shí)例代碼

    在本篇文章里小編給大家整理的是一篇關(guān)于java實(shí)現(xiàn)兩個(gè)線程交替打印的相關(guān)知識(shí)點(diǎn)內(nèi)容,有需要的朋友們參考下。
    2019-12-12
  • 完整的logback配置示例ELK整合包含生成json日志

    完整的logback配置示例ELK整合包含生成json日志

    這篇文章主要為大家介紹了完整的logback配置示例ELK整合包含生成json日志,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-03-03
  • Spring @async方法如何添加注解實(shí)現(xiàn)異步調(diào)用

    Spring @async方法如何添加注解實(shí)現(xiàn)異步調(diào)用

    這篇文章主要介紹了Spring @async方法如何添加注解實(shí)現(xiàn)異步調(diào)用,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-01-01
  • SpringBoot整合分布式鎖redisson的示例代碼

    SpringBoot整合分布式鎖redisson的示例代碼

    這篇文章主要介紹了SpringBoot整合分布式鎖redisson,本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-02-02
  • Java實(shí)現(xiàn)簡(jiǎn)單的抽牌游戲

    Java實(shí)現(xiàn)簡(jiǎn)單的抽牌游戲

    這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)簡(jiǎn)單的抽牌游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-04-04
  • springboot整合mybatis流程詳解

    springboot整合mybatis流程詳解

    這篇文章主要為大家詳細(xì)介紹了springboot整合mybatisplus的方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-05-05

最新評(píng)論