一、描述一个对象生成的过程虚拟机内存的分配情况
图片描述:共享的区域有方法区和堆,虚拟机栈 和PC计数器为线程私有
堆:是用来存放对象和数组,以及对象的成员变量
方法区:将加载进来的class文件进行解析后,得到的方法代码,和一些常量存于此
虚拟机栈:用于存放局部变量、但如果局部变量为对象,只会存放内存地址,具体也是存放在堆中。基本数据类型则直接存在虚拟机栈
程序计数器:用于确认当前线程执行到哪一步
使用例子解释java代码运行时具体各区说执行的工作
// 创建一个对象
public class Person{
int age;
String name;
public void walk() {
System.out.println("我正在走路。。。。");
}
}
// 调用测试类的main方法去启动这个过程
public class Test {
public static void main(String[] args) {
Person person = new Person();
person.name = "java的架构师技术栈";
person.age = 18;
person.walk();
}
}
我们分析一下这个过程
1、jvm去方法去寻找这个Test类的代码信息,如果有的话,则直接调用,如果没有,使用类加载机制把类加载进来,同时把静态变量、静态方法、常量加载进来。这里加载的是( “java的架构师技术栈”) 18是基本数据类型不存放在常量池(只有被final修饰的基本数据类型和String类型变量在编译时才会被确定下来,因此存放在常量池中。 )
2、jvm程序指针到main方法,看到Person person=new Person()。首先分析Person这个类,同样的寻找Person类的代码信息,有就加载,没有的话类加载机制加载进来。同时也加载静态变量、静态方法、常量(“我正在走路。。。”)
3、jvm接下来看到了person,person在main方法内部,因而是局部变量,存放在栈空间中。
4、jvm接下来看到了new Person()。new出的对象(实例),存放在堆空间中。
5、jvm接下来看到了“=”,把new Person的地址告诉person变量,person通过四字节的地址(十六进制),引用该实例
6、jvm看到person.name = “java的架构师技术栈”;person通过引用new Person实例的name属性,该name属性通过地址指向常量池的"java的架构师技术栈"。
7、jvm看到person.age = 18; person的age属性是基本数据类型,直接赋值。
8、jvm看到person.walk(); 调用实例的方法时,并不会在实例对象中生成一个新的方法,而是通过地址指向方法区中类信息的方法。
引发线程安全的原因
定义:类中的成员变量,也叫实例变量,也叫全局变量,它是非线程安全,是所有线程共享的变量,定义在方法中的私有变量是线程安全的,是每个线程私有的