java内存管理机制(堆栈篇)

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());
        }
    }

}

问题在代码开头都描述出来了,所有出现///!!!都是值得推敲的地方,这个实例也就是堆栈的运用吧,但是遇到了瓶颈,不知道有没有大佬能帮忙解决这个问题,应该用怎样的方法解决动态添加学生对象。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值