`
_神谕_
  • 浏览: 3340 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

Java的Hello World原理解析【转载】

阅读更多
面试:深入理解Hello World
我们初学java的第一个程序是"hello world"
1 public class HelloWorld {
2     public static void main(String[] args) {
3         System.out.println("hello world");
4     }
5 }
上面程序到底是怎么在屏幕上输出“hello world”的呢?这就是本来要讲解的内容,即System.out.println("hello world")的原理。

我们先看看System.out.println的流程。先看看System.java中out的定义,源码如下:

1 public final class System {
2     ...
3
4     public final static PrintStream out = null;
5
6     ...
7 }

从中,我们发现,
(01) out是System.java的静态变量。
(02) 而且out是PrintStream对象,PrintStream.java中有许多重载的println()方法。
OK,我们知道了out是PrintStream对象。接下来,看它是如何被初始化的,它是怎么和屏幕输出关联的?
我们还是一步步来分析,首先看看System.java的initializeSystemClass()方法。
1. initializeSystemClass()的源码如下: 把out部分标记为红色

1 private static void initializeSystemClass() {
2
3     props = new Properties();
4     initProperties(props);  // initialized by the VM
5
6     sun.misc.VM.saveAndRemoveProperties(props);
7
8     lineSeparator = props.getProperty("line.separator");
9     sun.misc.Version.init();
10
11     FileInputStream fdIn = new FileInputStream(FileDescriptor.in);
12     FileOutputStream fdOut = new FileOutputStream(FileDescriptor.out);
13     FileOutputStream fdErr = new FileOutputStream(FileDescriptor.err);
14     setIn0(new BufferedInputStream(fdIn));
15     setOut0(new PrintStream(new BufferedOutputStream(fdOut, 128), true));
16     setErr0(new PrintStream(new BufferedOutputStream(fdErr, 128), true));
17
18     loadLibrary("zip");
19
20     Terminator.setup();
21
22     sun.misc.VM.initializeOSEnvironment();
23
24     Thread current = Thread.currentThread();
25     current.getThreadGroup().add(current);
26
27     setJavaLangAccess();
28
29     sun.misc.VM.booted();
30 }

我们只需要关注上面的红色代码部分:即
FileOutputStream fdOut = new FileOutputStream(FileDescriptor.out);
setOut0(new PrintStream(new BufferedOutputStream(fdOut, 128), true));
将这两句话细分,可以划分为以下几步:
第1步 FileDescriptor fd = FileDescriptor.out;
第2步 FileOutputStream fdOut = new FileOutputStream(fd);
第3步 BufferedOutputStream bufOut = new BufferedOutputStream(fdOut, 128);
第4步 PrintStream ps = new PrintStream(bufout, true);
第5步 setOut0(ps);
说明:
(01) 第1步,获取FileDescriptor.java中的静态成员out,out是一个FileDescriptor对象,它实际上是“标准输出(屏幕)”的标识符。关于FileDescriptor的详细介绍,可以参考博文“java io系列09之 FileDescriptor总结”。
FileDescriptor.java中与FileDescriptor.out相关代码如下:

1 public final class FileDescriptor {
2
3     private int fd;
4
5     public static final FileDescriptor out = new FileDescriptor(1);
6
7     private FileDescriptor(int fd) {
8         this.fd = fd;
9         useCount = new AtomicInteger();
10     }
11
12     ...
13 }

(02) 创建“标准输出(屏幕)”对应的“文件输出流”。
(03) 创建“文件输出流”对应的“缓冲输出流”。目的是为“文件输出流”添加“缓冲”功能。
(04) 创建“缓冲输出流”对应的“打印输出流”。目的是为“缓冲输出流”提供方便的打印接口,如print(), println(), printf();使其能方便快捷的进行打印输出。
(05) 执行setOut0(ps);

接下来,解析第5步的setOut0(ps)。查看System.java中setOut0()的声明,如下:
private static native void setOut0(PrintStream out);
从中,我们发现setOut0()是一个native本地方法。通过openjdk,我们可以找到它对应的源码,如下:

1 JNIEXPORT void JNICALL
2 Java_java_lang_System_setOut0(JNIEnv *env, jclass cla, jobject stream)
3 {
4     jfieldID fid =
5         (*env)->GetStaticFieldID(env,cla,"out","Ljava/io/PrintStream;");
6     if (fid == 0)
7         return;
8     (*env)->SetStaticObjectField(env,cla,fid,stream);
9 }

说明:
这是个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()。
关于JNI的更多内容,可以参考博文“Android JNI和NDK学习(02)--静态方式实现JNI”。
(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()中,上面的5步就是将“FileDescriptor.out”封装了起来。封装后的System.in既有缓冲功能;又有便利的操作接口,如print(), println(), printf()。

分享到:
评论

相关推荐

    理解java的helloworld.doc

    本文将深入解析 HelloWorld 示例背后的原理和常见问题,帮助读者更好地理解和掌握 Java 开发的基础。 首先,我们需要确保已经正确安装了 JDK(Java Development Kit)。JDK 提供了编译和运行 Java 程序所需的所有...

    spring-mvc-helloworld

    这个项目 "spring-mvc-helloworld" 是一个基础的 Spring MVC 示例,用于帮助初学者理解其工作原理。 1. **MVC 架构模式**: MVC 是一种设计模式,用于分离应用程序的数据、业务逻辑和用户界面。Model 负责处理数据...

    JMX HelloWorld Download

    **标题解析:** "JMX HelloWorld Download" 指的是一个关于Java Management Extensions(JMX)的简单示例,可能是用于教学或演示如何在Java应用中使用JMX技术。"Download"表明这是一个可以下载的资源,可能包含了...

    springmvc注解版 helloworld

    在这个"springmvc注解版 helloworld"项目中,我们将深入探讨如何使用注解来配置和实现一个简单的Hello World应用。这个项目特别适合初学者入门,因为每个步骤都有详细的注释,确保你能理解每一行代码的作用。 首先...

    webservice复习-helloworld

    通过这个简单的“helloworld”示例,开发者可以深入理解Web服务的工作原理,包括SOAP消息的构造、WSDL的解读以及服务的调用。这对于进一步学习和开发复杂的分布式系统至关重要。此外,源码和工具的标签表明,这个...

    CORBA Hello World 实例

    module HelloWorld { interface Hello { string say_hello(); }; }; ``` 这里定义了一个名为`Hello`的接口,它有一个`say_hello`方法,返回一个字符串。 ### 3. 生成 stub 和 skeleton 使用ORB提供的编译器...

    学Struts2从HelloWorld示例开始

    从“HelloWorld”示例开始学习Struts2是一个经典的入门方法,因为它能帮助我们理解框架的基本工作原理。 首先,我们需要创建一个展示信息的JSP文件,即`HelloWorld.jsp`。在这个文件中,我们使用Struts2的标签库,...

    springmvc-helloworld

    本教程将通过 "springmvc-helloworld" 这个示例项目,帮助你了解如何搭建 SpringMVC 环境以及其工作原理。 ### 一、SpringMVC 架构 SpringMVC 的核心组件包括 DispatcherServlet、Model、ViewResolver 和 ...

    Spring MVC HelloWorld入门例子

    本篇文章将带你通过一个简单的 "HelloWorld" 示例,深入了解 Spring MVC 的基本工作原理和配置过程。 首先,我们来了解一下 Spring MVC 的核心概念: 1. **DispatcherServlet**:这是 Spring MVC 的前端控制器,...

    spring MVC的HelloWorld完整实例

    总的来说,"spring MVC的HelloWorld完整实例"提供了一个简单的起点,帮助开发者了解Spring MVC的工作原理和基本配置。通过这个实例,你可以学习到如何创建控制器、配置Spring MVC、处理请求和返回响应,以及如何在...

    Struts2 实例 HelloWorld

    ### Struts2 实例 HelloWorld 知识点解析 #### 一、概述 在学习Struts2框架时,通常会从最简单的“Hello World”实例开始。这种实例不仅可以帮助初学者快速上手,还能让大家对Struts2的工作原理有一个基本的理解。...

    JAXB的HelloWorld源码

    总结来说,JAXB是Java中处理XML的强大工具,通过学习JAXB的HelloWorld源码,新手可以快速上手,理解XML和Java对象之间的绑定,以及如何进行序列化和反序列化操作。随着对JAXB的深入理解和实践,开发人员能够更高效地...

    springmvc项目helloworld

    在这个"springmvc项目helloworld"中,我们将深入理解Spring MVC的基本概念、工作原理以及如何创建一个简单的Hello World应用程序。 1. **Spring MVC 概述** - Spring MVC 是 Spring 框架为Web开发提供的模块,它...

    一个struts2的helloworld例子加源码

    通过这个简单的例子,我们可以理解Struts2的工作原理:当用户发起一个请求时,Struts2的DispatcherServlet会解析请求,根据`struts.xml`中的配置找到对应的Action,并调用其`execute()`方法。Action执行完成后,根据...

    SpringMVC------从HelloWorld开始

    让我们从一个简单的"HelloWorld"示例开始,深入了解SpringMVC的工作原理。 首先,创建一个`HelloWorld`控制器是开始任何SpringMVC项目的第一步。在Java中,我们通常会创建一个继承自`org.springframework....

    Hello World the WebWork way

    4. **HelloWorld.java代码解析** - **代码结构**: ```java package ch2.example1; import com.opensymphony.xwork.Action; public class HelloWorld implements Action { private String message; ...

    springmvc入门helloworld

    在"springmvc入门helloworld"这个主题中,我们将探讨如何从零开始搭建一个简单的Spring MVC项目,展示其基本工作流程和关键组件。以下是详细的知识点讲解: 1. **Spring MVC 概述** - Spring MVC 是Spring框架的一...

    spring MVC Helloworld

    **Spring MVC HelloWorld 实例详解** 在Java Web开发中,Spring MVC框架被广泛使用,它为构建基于模型-视图-控制器(MVC)模式的Web应用程序提供了强大的支持。本篇文章将详细讲解如何在MyEclipse2013环境中创建一...

    HelloWorld的javap -verbose HelloWorld 字节码初探

    标题中的“HelloWorld的javap -verbose HelloWorld”指的是在Java编程环境中,通过`javap`这个命令行工具来反汇编一个简单的“HelloWorld”程序,以深入理解字节码的工作原理。`javap`是Java Platform Debugger ...

    spring mvc 在 intellij 的 helloworld 基本配置

    为了搭建基于Spring MVC框架和IntelliJ IDEA开发环境的Hello World项目,首先要掌握以下知识点: 1. **Spring MVC框架的理解**: - Spring MVC是Spring框架的一部分,用于构建Web应用程序。 - 它提供了一种MVC...

Global site tag (gtag.js) - Google Analytics