类由静态到动态,会经历运行时数据区这一步: 

java 静态数据能修改吗 java静态数据存放在方法区_java 静态数据能修改吗

 

静态编译:把Java代码编译成字节码文件Class文件,它以静态方式存在

类加载器:把Java字节码文件加载到内存中

 

【方法区】与【堆】是运行时数据区在所有线程间共享的,它们是存数据的地方

【虚拟机栈】,【本地方法栈】,【程序计数器】是运行时数据区线程私有的,它们是执行逻辑的地方 

以下代码为例:

public class Person {
    String name;
    public void say(String name){
        System.out.println(name);
    }

    public static void main(String[] args) {
        Person person = new Person(); 
        person.say("Hello World!");
    }
}

方法区:

Person是存放在方法区的,方法区用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码(JIT)

JIT:Java编译成ByteCode(静态编译)然后交由CPU执行本地指令,在CPU执行本地指令之前,会将一些热点代码存储下来保存在方法区,以提高效率,例如:

for (int i=0;i<10000;i++){
            function();
        }

堆:

当我们new Person的时候,这个对象实例会存在堆中,几乎所有的对象实例都在这里分配内存

当堆里面没有可分配内存的时候,会报OOM错误(内存超出)

 程序计数器:类似汇编语言的寄存器(指令寄存器),用于存放下一条指令所在单元的地址的地方

虚拟机栈:存放局部变量表,操作数栈,动态连接/方法返回地址,分配基本类型和自定义对象的引用(局部变量表)

本地方法栈:为了Native的执行和调出(Java源码中的很多Native方法) 

在某些虚拟机中,本地方法栈+虚拟机栈+程序计数器=栈区

通常栈底是主方法Main,先入后出,第一个放入的方法是Main,最后执行的方法是Main

当死循环或递归调用栈中方法过多时候,会报StackOverFlow错误 

堆栈之间如何配合:

观察以下代码:

String s = "1" + "2" + "3";
        String s1 = "hello";
        String s2 = new String("hello");
        System.out.println(s1 == s2);
        System.out.println(s1.equals(s2));

打印的是:false\ntrue,说明s1和s2的值相同,地址不同

第一行创建了一个对象,编译的时候进行了字符串折叠(老版本JDK没有优化,会创建四个对象)

第二行只创建一个对象,先在常量池创建字符串“hello”,然后将地址给s1

第三行创建了一个String对象,然后创建常量池字符串,所以栈中变量s2先指向堆中String对象地址,String对象地址再指向常量池中的字符串

因此s1和s2的地址不同 

继续看上面的这段代码:

public class Person {
    String name;
    public void say(String name){
        System.out.println(name);
    }

    public static void main(String[] args) {
        Person person = new Person(); 
        person.say("Hello World!");
    }
}

(1)还没new之前,person存入栈中的局部变量表(局部变量表可以存八种基本类型和引用类型)

(2)new Person之后会产生一个实例,存在堆中,局部变量表中的person引用指向堆中的实例

(3)在方法区中保存了Person类信息,比如属性name和方法say

(4)调用say方法,局部变量表的person引用指向堆的Person实例,Person实例指向方法区say方法的地址,完成了一系列调用