任何一个学过JAVA的人应该都对这段代码非常熟悉。空闲时间翻了下代码,看看它的底层是怎么实现的
public class HelloWorld {
public static void main(String[] args) {
System.out.print("Hello, World!");
}
}
首先点开out
,发现它是System
类中的一个public static final
变量,类型为PrintStream
。为了找到它是怎么初始化的,一直往前翻到System
类的构造函数
data:image/s3,"s3://crabby-images/157c7/157c79db742710a3e6ba05d93bbf05fed2b06286" alt=""
从System
类的注释中发现,VM会调用initPhase1
这个方法来初始化这个类。先不管VM,先看下initPhase1
方法做了什么
data:image/s3,"s3://crabby-images/2d22d/2d22d81ee9cdc7e740235c37c43a81f78fa49605" alt=""
发现它用FileDescriptor.out
创建了FileOutputStream
对象,再用这个对象创建了PrintStream
对象,最后调用native的setOut0
data:image/s3,"s3://crabby-images/7c850/7c850bf4e325dd5cac57ebd686d9ecaadf6a6a59" alt=""
在创建PrintStream
对象时,先将FileOutputStream
封装成BufferedOutputStream
,然后把BufferedOutputStream
封装成OutputStreamWriter
。这一步中会根据传入的字符集创建OutputStreamWriter
中的编码器StreamEncoder
data:image/s3,"s3://crabby-images/892d0/892d0b88c7010c31921e0410b2fc8c71be58083e" alt=""
这样OutputStreamWriter
就创建好了,在print的时候会调用这个类的方法,最后根据调用栈发现调用了FileOutputStream
中的writeBytes
方法
data:image/s3,"s3://crabby-images/1fd13/1fd13619891bdc86b51a461def9904905aae4548" alt=""
data:image/s3,"s3://crabby-images/d59df/d59df22481dd3d3d7cdc2b3d03f50809c6f445f3" alt=""
发现这个方法是native的,也就是说不在JAVA中实现,打开openjdk,checkout到tag jdk18
data:image/s3,"s3://crabby-images/ed6a8/ed6a8bdad9a31dc06383a48603b97e9b34d1f66a" alt=""
找到在jdk中的实现,发现调用了IO_Append
和IO_Write
,而这两个是宏定义,指向了handleWrite
方法
data:image/s3,"s3://crabby-images/822a7/822a774e9ad9458f6b8653dd9ffe7deca31b36a2" alt=""
data:image/s3,"s3://crabby-images/1072d/1072df975f04e31ede593d456d03e8a38b220b65" alt=""
在不同的平台下,这个方法有不同的实现
在windows下调用了WriteFile
这个Win32 API
data:image/s3,"s3://crabby-images/aaa33/aaa33300e895b5ac77b53391dbe229791e256c69" alt=""
在linux下调用了unistd.h
中定义的write
方法
data:image/s3,"s3://crabby-images/99921/99921ad7204743f2c799172bbef5083a2cd97e53" alt=""
打开glibc,在write_nocancel.c
下看到提供的write方法实现,通过一堆的宏定义最终是一个系统调用,调用了linux的write
方法
data:image/s3,"s3://crabby-images/4f6a8/4f6a815147e9a666e494beb072767f868615bac9" alt=""
在linux内核源代码中,找到write
的SYSCALL,其中调用了ksys_write
方法
data:image/s3,"s3://crabby-images/d94ad/d94adf9a7a14ba741b3f0b13a0d2512c6e13f349" alt=""
这个方法中会获取fd,然后再通过vfs_write
写入,顺着调用链一路找到了下面这个write
方法
data:image/s3,"s3://crabby-images/dc79d/dc79d3663917d30b8466757564693a7dfa5dcb59" alt=""
后面的两个参数很好理解,第一个tty_struct
是什么?
In many computing contexts, “TTY” has become the name for any text terminal, such as an external console device, a user dialing into the system on a modem on a serial port device, a printing or graphical computer terminal on a computer’s serial port or the RS-232 port on a USB-to-RS-232 converter attached to a computer’s USB port, or even a terminal emulator application in the window system using a pseudoterminal device.
简单来说,就是一个文本终端。它也是一个虚拟文件系统,模拟了终端设备
回头看ksys_write
方法的第一行,打开了一个文件描述符,其中调用了__fget_light
方法
data:image/s3,"s3://crabby-images/356e1/356e1e4d29fc2f60e3115c42d77b7f789338c24d" alt=""
从第一行就能看到,从current
中找到了files_struct
。current
是一个宏定义,获取当前正在运行的任务current_task
,而current_task->files
是这个当前正在运行的任务所打开的文件信息
data:image/s3,"s3://crabby-images/f55fa/f55faaa03b1f3471bfb365f042ba5abf42d891c6" alt=""
data:image/s3,"s3://crabby-images/eff69/eff69ab85e2d00be88234afd6eb09609407128b7" alt=""
也就是说,进程打开了一个由虚拟文件系统管理的虚拟文件,它是一个伪终端PTY,然后向其中写入数据
在linux中输入tty,就可以看到对应的伪终端设备文件路径。往里写入数据,就可以实现向另一个终端中打印东西
data:image/s3,"s3://crabby-images/1c0f9/1c0f902b6eb6f43e0cbdb2a5e7b831d5ce1a11a4" alt=""
参考:
- https://github.com/openjdk/jdk
- https://sourceware.org/git/glibc.git
- https://learn.microsoft.com/en-us/windows/win32/api/fileapi/nf-fileapi-writefile
- https://www.linusakesson.net/programming/tty/index.php
- https://en.wikipedia.org/wiki/Devpts
- https://man7.org/linux/man-pages/man7/pty.7.html
- https://github.com/GNOME/vte