java把内存分为栈内存、堆内存、方法区(常量池、静态池)、程序计数、本地方法栈、执行引擎。
今天,我就来聊一聊堆栈,在我的理解里,java算是一场堆栈的游戏,所有的东西,都是堆栈的各种形式。
- 堆内存是存放对象的具体信息,在程序之中是由new来创建
- 栈内存保存的是一块堆内存的地址,即通过地址找到堆内存对象信息
- New拥有开辟内存的最高级别
Person p1=new Person();
p1.name ="猪悟能";
Person p2 = new Person();
System.out.print(p2.name);
Person p3=p1;
System.out.print(p3.name);
p3.name="沙悟净"
System.out.print(p1.name);
通过上图来深刻理解一下堆栈
Person p1 = new Person();
p1.name=“猪悟能”;
声明p1,在栈中建立一个p1
new Person(),相应在堆中开辟一个空间,在声明与创建的过程中,并未赋值,所以,id=0,name=null
p1.name=“猪”,这一步,为其name属性赋值,所以此时name="猪”;
栈中存放的便是对应的堆中的地址。所以p1指向0x0a1。
Person p2 = new Person();
同理,在栈中声明p2,存放的是堆中p2的地址,堆中开辟一个空间p2,地址为0x0a2。
Person p3 = p1;
在栈中声明一个p3,而等号后并未出现new关键字,所以并未在堆中开辟空间,而是将p1的地址,赋给p3,如图所示,p3的箭头指向0x0a1
那么,此时p3.name也就是p1.name,也就是猪
p3.name=“沙”
此时,输出p1.name=“沙”
也就是说,p1和p2操作的是堆中的一空间,p1和p2的属性值永远相等,永远是同一个。
创建Student类型的数组stus,new了五个连续的空间存放student对象
stus[0]=new Student();实例化stus[0],也就是新开辟一个空间,用来存放studengt对象,Student对象已经赋了初值0和张三,所以此时输出stus[0].name=“张三”。
sout(stus[1]),在此之前,并未实例化stu[1],那么,输出结果也就是null.
sout(stus[1].number),输出空指针异常,我带大家分析一下
首先,并未实例化stus[1],如果此时输出stus[1],那么结果毫无疑问就是空,但是,此时你非要去操作,其中的number属性,也就是操作了本来就是null的一个数组的一个属性,那么必然会出现空指针异常。
也就是说,以后你可以去输出一个并未实例化的的对象数组某个元素,但是你不能去进一步操作对象数组里的属性。
stus[1]=new Student();
sout(stus[1].number);
此时,结果肯定是0,因为你已经实例化了。
要特别注意的是,对于引用数据类型而言,其类型的数组(例如这里的Stu数组),他在堆内存中也是开辟了一块空间,而每一个数组元素并不是放数据的,是放地址的,当你并未实例化Student对象时,每个数组元素都为空,实例化以后,相当于在堆内存中又额外开辟了空间,这个空间存放的是Student对象,而数组元素防治的是该对象的地址。
如果可以,你也可以这样理解,他就相当于一个一级索引表,内部不是放数据的,而是放改数据的地址的。
所以说,对于对象数组而言,如果你不去实例化,那么其不会自动赋初值,必然是null,如果在此基础上,你去操作对象数组中的某个元素的某个属性,必然会出现空指针异常
java中,会自动给基本数据类型赋初值,除基本数据类型外,其他数据类型全部为空,基本数据类型包括int、double、float、byte… , 其他数据类型中有 引用数据类型String、类、接口、数组。
接下来,分享一个实例
创建一个对象数组版的学生信息管理系统,一个学生实体类,然后创建一个类数组,去实现增删改查。
//学生管理系统,使用数组和类
/**
* 设置最大能存4个学生对象
* 当选择添加学生时,可以选择你想添加的学生数量,介于一到四之间,但是会出现一系列的问题
* 没有办法解决数组的扩容和缩容的问题,应该是现在所学的东西,没办法支持我完成下去
*
* 假设我要添加两个学生对象,我确实添加成功了,但时当我查询所有学生信息的时候,因为他只有两个对象,所以确实把前两个输出来了,但时当他
* 遍历到第三个时,会直接报空指针异常,这个问题,我本来打算用一个全局变量去改变遍历的次数,但时却不能实现,因为我要动态获取学生对象的值
* 而全局变量是固定值
* **/
public class StudentManagementSystem {
public static void main(String[] args) {
Student[] student = new Student[4];
Scanner scanner = new Scanner(System.in);
//
System.out.println("---------欢迎使用学生信息管理系统(类数组版)--------");
while(true){
System.out.println("1、添加学生");
System.out.println("2、查询某个学生信息");
System.out.println("3、添加全部学生信息");
System.out.println("4、修改学生信息");
System.out.println("5、退出系统");
switch (scanner.nextInt()){
case 1:
addStu(scanner,student);
break;
case 2:
selectStuBySno(scanner,student);
break;
case 3:
selectAllStudent(student);
break;
case 4:
updateStuBySno(scanner,student);
break;
case 5:
System.out.println("您已退出系统");
return;
default:
System.out.println("请输入正确编号");
}
}
}
//添加学生
public static void addStu(Scanner scanner,Student[] students){
System.out.println("请输入您要添加的学生数量,最多四个,我们将为您合理分配空间");
final int num = scanner.nextInt();
if (num<1||num>4){
System.out.println("输入非法字符");
}
if (num>=1&&num<=4){
for(int i=0;i<num;i++){
students[i]=new Student();//!!!!!!!!!!!!
System.out.println("请输入第"+(i+1)+"位学生的学号:");
int sno=scanner.nextInt();
students[i].setSno(sno);
System.out.println("请输入第"+(i+1)+"位学生的姓名:");
String name=scanner.next();
students[i].setSname(name);
System.out.println("请输入第"+(i+1)+"位学生的性别:");
String sex=scanner.next();
students[i].setSex(sex);
System.out.println("请输入第"+(i+1)+"位学生的年龄:");
int age=scanner.nextInt();
students[i].setAge(age);
System.out.println("请输入第"+(i+1)+"位学生的电话:");
int phone=scanner.nextInt();
students[i].setTelephone(phone);
// students[i]=new Student(sno,name,sex,age,phone);
}
}
}
//查询某个学生信息
public static int selectStuBySno(Scanner scanner,Student [] students){
System.out.println("请输入的学生学号:");
int sno = scanner.nextInt();
int subscript=-1;
for (int i = 0;i < students.length; i++){
students[i]=new Student();!!!!!!!!!!!!!
if(students[i].getSno()==sno){
System.out.println(students[i]);
subscript=i;
}
}
if(subscript==-1){
System.out.println("未找到该学号的学生");
}
return subscript;
}
//查询所有学生的信息
public static void selectAllStudent(Student[] students){
for (int i = 0 ;i < students.length ; i++){
students[i]=new Student();!!!!!!!!!!!!!!
System.out.println(students[i]);
}
}
//修改某个学生信息
public static void updateStuBySno(Scanner scanner,Student[] students){
int subscript = selectStuBySno(scanner,students);
if (subscript>=0){
System.out.println("请输入修改后的学生姓名:");
students[subscript].setSname(scanner.next());
System.out.println("请输入修改后的学生性别:");
students[subscript].setSex(scanner.next());
System.out.println("请输入修改后的学生年龄:");
students[subscript].setAge(scanner.nextInt());
System.out.println("请输入修改后的学生电话:");
students[subscript].setTelephone(scanner.nextInt());
}
}
}
问题在代码开头都描述出来了,所有出现///!!!都是值得推敲的地方,这个实例也就是堆栈的运用吧,但是遇到了瓶颈,不知道有没有大佬能帮忙解决这个问题,应该用怎样的方法解决动态添加学生对象。