84669 人学习
152542 人学习
20005 人学习
5487 人学习
7821 人学习
359900 人学习
3350 人学习
180660 人学习
48569 人学习
18603 人学习
40936 人学习
1549 人学习
1183 人学习
32909 人学习
有一个很常见的代码:System.out.println,这是再常见不过的输出语句,但是学了一段时间突然想起去看了下源码,下面是源码:
public final static PrintStream out = null;
out是在System类下定义的,还没实例化,out仅仅是一个null的变量,甚至都不能称之为静态对象,但是却能调用PrintStream类下的println()方法,这就不太理解了?请问为什么?
详细解释:中文版http://www.cnblogs.com/skywan...英文版基于jdk7 https://luckytoilet.wordpress...
亲,首先,当你启动你的测试类,也就是实例化你的测试类时,你所调用的所有静态类都会先实例化了,所以其实当你执行到System.out.println语句时,System这个类里面的方法和属性都是已经被初始化好了的!
static {
registerNatives();
}上面这个静态方法的注解说会调用initializeSystemClass方法进行初始化,这个方法里面的 setOut0(newPrintStream(fdOut, props.getProperty("sun.stdout.encoding")));应该对out进行赋值, private static native void setOut0(PrintStream out)是本地方法;
一看到final static,第一反应就应该是这丫的是常量,常量必须是要初始化的,然后再找找,果然他是调用本地方法去初始化的;
额, 在initializeSystemClass()这个函数里面做的初始化工作, 这个函数是在系统线程初始化后调用的, 也就是说, 所有的静态成员变量(err out in)都是在这函数里面初始化了的, 例如说这个out : FileOutputStream fdOut = new FileOutputStream(FileDescriptor.out); setOut0(newPrintStream(fdOut, props.getProperty("sun.stdout.encoding")));而setOut0()函数是在native层,在native层把初始化的对象和这个out建立起连接
详细解释:中文版http://www.cnblogs.com/skywan...
英文版基于jdk7 https://luckytoilet.wordpress...
亲,首先,当你启动你的测试类,也就是实例化你的测试类时,你所调用的所有静态类都会先实例化了,所以其实当你执行到System.out.println语句时,System这个类里面的方法和属性都是已经被初始化好了的!
static {
}
上面这个静态方法的注解说会调用initializeSystemClass方法进行初始化,这个方法里面的 setOut0(newPrintStream(fdOut, props.getProperty("sun.stdout.encoding")));应该对out进行赋值,
private static native void setOut0(PrintStream out)是本地方法;
一看到final static,第一反应就应该是这丫的是常量,常量必须是要初始化的,然后再找找,果然
他是调用本地方法去初始化的;
额, 在initializeSystemClass()这个函数里面做的初始化工作, 这个函数是在系统线程初始化后调用的, 也就是说, 所有的静态成员变量(err out in)都是在这函数里面初始化了的,
例如说这个out :
FileOutputStream fdOut = new FileOutputStream(FileDescriptor.out);
setOut0(newPrintStream(fdOut, props.getProperty("sun.stdout.encoding")));
而setOut0()函数是在native层,在native层把初始化的对象和这个out建立起连接