首先说明下,这个和编译原理没有什么关系。
楼上已经给了How System.out.println() really works具体实现的链接。
1.看下System.out.println的流程。先看看System.java中out的定义
public final static PrintStream out = null;你会发现,out是system的静态变量,out实际是一个PrintStream对象,而println()方法又有狠多重载。
现在再来看看这个out是怎么被初始化的。
在system类里的initializeSystemClass方法里有
FileOutputStream fdOut = new FileOutputStream(FileDescriptor.out);
setOut0(new PrintStream(new BufferedOutputStream(fdOut, 128), true));这个重点呢就在这个setOut0方法里。来,咱们跟一下。
private static native void setOut0(PrintStream out);果不其然,居然是一个native方法,在openJDK找到对应部分。
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);
}
这是个JNI函数,我们来对它进行简单的分析。
(01) 函数名
JNIEXPORT void JNICALL Java_java_lang_System_setOut0(JNIEnv *env, jclass cla, jobject stream)
这是JNI的静态注册方法,Java_java_lang_System_setOut0(JNIEnv *env, jclass cla, jobject stream)会和System.java中的setOut0(PrintStream out)关联;而且,参数stream 对应参数out。简单来说,我们调用setOut0(),实际上是调用的Java_java_lang_System_setOut0()。
(02) jfieldID fid = (*env)->GetStaticFieldID(env,cla,"out","Ljava/io/PrintStream;");
这句话的作用是获取System.java的静态成员out的jfieldID,"Ljava/io/PrintStream;"是说明out是java.io.PrintStream对象。
获取out的jfieldID的作用,是我们需要通过操作“out的jfielID”来改变out的值。
(03) (*env)->SetStaticObjectField(env,cla,fid,stream);
这句话的作用是,设置fid(fid就是out的jfieldID)对应的静态成员的值为stream。
stream是我们传给Java_java_lang_System_setOut0()的参数,也就是传给setOut0的参数。
总结上面的内容。我们知道,setOut0(PrintStream ps)的作用,就是将ps设置为System.java的out静态变量。
前面,已经说过FileDescriptor.out就是机器的“标准输出(屏幕)”的文件标识符。我们可以通俗的将文件标识符就理解为,FileDescriptor.out就是代表的“标准输出”。
因此,在initializeSystemClass()中,上面的几步就是将“FileDescriptor.out”封装了起来。封装后的System.out既有缓冲功能;又有便利的操作接口,如print(), println(), printf()。
如果楼主想要深入研究Java IO这块,推荐来 Java IO系统总结 。
留一个Java的群号:170936712
还有一个python的:96763994