Java SE之Java工作原理

 在Java中引入了虚拟机的概念,即在机器和编译程序之间加入了一层抽象的虚拟的机器。这台虚拟的机器在任何平台上都提供给编译程序一个的共同的接口。编译程序只需要面向虚拟机,生成虚拟机能够理解的代码,然后由解释器来将虚拟机代码转换为特定系统的机器码执行。在Java中,这种供虚拟机理解的代码叫做字节码(ByteCode)(class文件的内容),它不面向任何特定的处理器,只面向虚拟机。每一种平台的解释器是不同的,但是实现的虚拟机是相同的。Java源程序经过编译器编译后变成字节码,字节码由虚拟机解释执行,虚拟机将每一条要执行的字节码送给解释器,解释器将其翻译成特定机器上的机器码,然后在特定的机器上运行。【From CNBlog's lovebeauty】

编译器(高级语言.java To 字节码.class)
------->【虚拟机(解释执行.class字节码文件,送达解释器)】------->解释器(翻译为机器码,面向特定系统的特定机器而运行)   即 .java--.class--机器码
特性:
  1.跨平台(可移植,平台无关性)
  2.安全性(参考文献:
http://www.bubuko.com/infodetail-187060.html
  3. 面向对象特性
       3.1封装:在面向对象语言中,封装特性是由类来体现的,我们将现实生活中的一类实体定义成类,其中包括属性和行为(在Java中就是方法),就好像人类,可以具有name,sex,age等属性,同时也具有eat(),sleep()等行为,我们在行为中实现一定的功能,也可操作属性,这是面向对象的封装特性;
       3.2继承:继承就像是我们现实生活中的父子关系,儿子可以遗传父亲的一些特性,在面向对象语言中,就是一个类可以继承另一个类的一些特性,从而可以代码重用,其实继承体现的是is-a关系,父类同子类在本质上还是一类实体;
       3.3多态:多态就是通过传递给父类对象引用不同的子类对象从而表现出不同的行为
       3.4抽象:抽象就是将一类实体的共同特性抽象出来,封装在一个抽象类中,所以抽象在面向对象语言是由抽象类来体现的。比如鸟就是一个抽象实体,因为抽象实体并不是一个真正的对象,它的属性还不能完全描述一个对象,所以在语言中体现为抽象类不能实例化
  4.垃圾自动回收机制
  5.健壮性
         5.1 异常处理机制
         5.2 垃圾自动回收机制与内存管理
  6.简单性:Eg.取消了指针,取消了goto .etc 
  7.网络技能
  8.多线程
      
学习重点:
    Java代码编译和执行的整个过程
    JVM内存管理及垃圾回收机制
 
个人理解:
   .Java File ----JVM执行完后输出的.Class File  可以视为 编译 
 
   jvm.Class File【解释器】----机器码  可以视为 解释   
 

[Process]
  
A.编译器(Javac,Java Compiler):将.java文件编译成.class文件
    A.1 系统平台无关
    A.2 面向虚拟机编译
    A.3 Questions:
              Q:C/C++与Java编译器的区别?(将有助于理解Java的跨平台特性)
              A:参考文献:https://openhome.cc/Gossip/JavaEssence/WhyJVM.html
 

 
 
 
B.虚拟机(Java virtual Machine
) :通过在实际的计算机上仿真模拟各种计算机功能来实现的。
    B.1 系统平台相关

C.类加载器(Java ClassLoader):类加载器把一个类装入JVM虚拟机需要经过三步骤:
              1. 装载(Load):查找和导入.class文件             
              2.链接(Link):检查装入.class文件的正确性,然后JVM虚拟机为变量分配内存,设置默认值

                  2.1校验
                  2.2准备
                  2.3解析        
              3.初始化(Initialize):把符号引用变成直接引用
       
           4.Questions:
               4.1
那为什么我要有验证这一步骤呢?首先如果由编译器生成的class文件,它肯定是符合JVM字节码格式的,但是万一有高手自己写一个class文件,让JVM加载并运行,用于恶意用途,就不妙了,因此这个class文件要先过验证这一关,不符合的话不会让它继续执行的,也是为了安全考虑吧。
               4.2在一个类加载器中一个类能初始化几次?Only once.
               4.3类的加载?
                   JVM的类加载是通过ClassLoader及其子类来完成的。
                   4.3.1 类的加载的最终产品是位于堆区中的Class对象
                   4.3.2 Class对象封装了类在方法区内的数据结构,并且向Java程序员提供了访问方法区内的数据结构的接口 
                   4.3.4 加载类的方式有以下几种:
                           1)从本地系统直接加载
                           2)通过网络下载.class文件
                           3)从zip,jar等归档文件中加载.class文件
                           4)从专有数据库中提取.class文件
                           5)将Java源文件动态编译为.class文件(服务器)
[类的加载:自顶向下]
[参考文献:http://blog.csdn.net/bingduanlbd/article/details/8363734]  
 
D.类的初始化
    类什么时候才被初始化:
1)创建类的实例,也就是new一个对象(初始化后,才为变量附值(默认值或指定值))
2)访问某个类或接口的静态变量,或者对该静态变量赋值
3)调用类的静态方法
4)反射(Class.forName("com.lyj.load"))
5)初始化一个类的子类(会首先初始化子类的父类)
6)JVM启动时标明的启动类,即文件名和类名相同的那个类
         只有这6中情况才会导致类的类的初始化。
     类的初始化步骤:
         1)如果这个类还没有被加载和链接,那先进行加载和链接
         2)假如这个类存在直接父类,并且这个类还没有被初始化(注意:在一个类加载器中,类只能初始化一次),那就初始化直接的父类(不适用于接口)
         3)加入类中存在初始化语句(如static变量和static块),那就依次执行这些初始化语句。
7)Java 中所有定义的基本类型或对象都必须初始化才能输出值
     Eg:
     import java.io.*;

     import java.util.*;
     public class foo{
        public static void main (String[] args){
           String s;    
           System.out.println("s=" + s);
        }
     }

A. 代码得到编译,并输出 “s=”
B. 代码得到编译,并输出 “s=null”
C. 由于 String s 没有初始化,代码不能编译通过
D. 代码得到编译,但捕获到 NullPointException异常

答案:C

解析:开始以为会输出 null 什么的,运行后才发现 Java 中所有定义的基本类型或对象都必须初始化才能输出值。     
          
E..class文件
      0.含义:class为后缀名的Java虚拟机可装载的文件。这个特定格式就是指JVM能够识别、能够装载的格式。
                因:
JVM在装载class文件时,要进行class文件验证,以保证装载的class文件内容符合正确的内部结构。 
      1.本质:指符合特定格式的字节流组成的二进制文件
      2.此类文件均由八位字节流组成,所有的16位,32位,64位分别通过读入2个,4个,8个字节来构造(按照高位字节在前---低地址的Big-endian顺序)。
      3.每个class文件包含且仅包含一个Java类型(类or接口)
      4.参考文献:
            
《The JavaTM Virtual Machine Specification》
            《Inside the Java Virtual Machine》(Second Edtion)  
             http://wenku.baidu.com/view/481b5b024b73f242336c5f95.html
原文地址:https://www.cnblogs.com/johnnyzen/p/7155591.html