Java中成员变量与局部变量的区别、形式参数、匿名对象与封装

Java中成员变量与局部变量的区别、形式参数、匿名对象与封装

学习要循循渐进



一、成员变量与局部变量的区别

1、在类中定义的位置不同

成员变量:类中,但是在方法外
局部变量:定义在方法内部

2、在内存中的位置也不同

成员变量:在堆内存中
局部变量:在栈内存中

3、初始化值也不同

成员变量:系统会给予默认值
局部变量:没有系统给的默认值,必须在定义的时候赋值,亦或者在方法中使用之前赋值,然后才能使用。

4、生命周期不同

成员变量的生命周期:随着对象的创建而创建,随着对象的消失而消失
局部变量的生命周期:随着方法的调用而创建,随着方法的调用完毕而消失

注意:
(1)方法与方法之间里面的局部变量不能互相调用。
(2)局部变量可以和成员变量一样,在方法中使用的时候,采用就近原则

二、形式参数

1 、形式参数的问题:

(1)当形式参数是基本数据类型的时候,将来调用方法的时候传入的是常量值,或者是该类型的变量
(2)当形式参数是引用数据类型的时候,将来调用方法的时候传入的是该类对象的地址值
(3)当形式参数是基本数据类型的时候,在方法中对变量做修改,不会影响到外部实际的栈内存中的值。
(4)当形式参数是引用数据类型的时候,在方法中对变量做修改,会影响到外部实际的堆内存中的值。

2、举例:

package com.shujia.wxl.day08;
class Demo2 {
    public int getSum(int a,int b) {
        a=100;
        b=200;
        return a + b;
    }
}

class Student2{
    String name;
    public void speak(){
        System.out.println("我热爱学习");
    }
}

class StudentTest2{
    /**
     * 如果将来你看到一个方法的形式参数是一个类的类型,说明他是一个引用数据类型
     * 这里其实需要的是该类的对象
     * 调用的时候,把main方法中创建好的对象名传过来,实际上传的就是对象的地址值
     * @param s
     */
    public void function(Student2 s){ //Student2 s = new Student2();
        s.name = "小王";
        s.speak();
    }
}


public class XingShiCanShuDemo {
    public static void main(String[] args) {
        Demo2 d = new Demo2();
        int a = 10;
        int b = 20;
        System.out.println(d.getSum(a,b));
        System.out.println("a:"+a+",b:"+b);

        //要想调用function(),就必须创建StudentTest2对象
        StudentTest2 st2 = new StudentTest2();

        Student2 student2 = new Student2();
        System.out.println(student2.name);
        st2.function(student2);
        System.out.println(student2.name);
    }
}

三、匿名对象

1、匿名对象:

   就是没有名字对象

2、匿名对象的使用场景:

(1)调用方法的时候,仅仅调用一次的时候,可以使用匿名对象
 注意,当调用多次的时候,不适合,每次new的时候都是一个新的匿名对象,会频繁的在堆内存中开辟空间
 当匿名对象使用完毕之后,无法再使用第二次,就变成了一个垃圾(当栈中没有变量引用它的时候,判定是一个垃圾),等待被垃圾回收器回收。
(2)当方法的参数类型是一个类的时候,可以使用匿名对象调用方法。
package com.shujia.wxl.day08;
class Phone4{
    public void show(){
        System.out.println("手机可以打电话");
    }
}

class PhoneDemo{
    public void function(Phone4 p){
        p.show();
    }
}

public class AnonymousDemo {
    public static void main(String[] args) {
        //创建一个PhoneDemo对象
        PhoneDemo pd = new PhoneDemo();
        Phone4 phone = new Phone4();
        pd.function(phone);
        System.out.println("=============================");
        //当new完对象之后直接使用,没有用变量名接收的对象,称之为匿名对象
        new PhoneDemo().function(new Phone4());
        new PhoneDemo().function(new Phone4());


    }
}

四、封装

首先我们先定义一个学生类:

class Student3 {
    //定义成员变量
    String name;
    private int age;

    public void setAge(int a){
        if(a<0 || a>200){
            System.out.println("您要传入的年龄数据有误,请重新赋值");
        }else{
            age = a;
        }
    }

    //定义一个输出所有的成员变量
    public void show() {
        System.out.println("姓名:" + name);
        System.out.println("年龄:" + age);
    }

}


public class StudentDemo2 {
    public static void main(String[] args) {
        //创建一个学生对象
        Student3 s = new Student3();
        s.show();

        //给对象的成员变量赋值
        s.name = "王宇";
//        s.age = 18;
//        s.age = 10000;
//        s.setAge(10000);
        //age 在 com.shujia.wyh.day08.Student3 中是 private 访问控制
//        s.age = 10000; //由于我们在Student3类中name前面加了一个private关键字,不可以直接使用获取到成员变量,要想赋值,
                        //就必须使用我们提供的数据校验赋值方法
        s.setAge(10000);
        s.show();

    }
}

我们在定义一个学生类并使用的时候,发现了一个问题:

通过对象给成员变量赋值的时候,可以赋值一些非法的数据。
这样是不合理不符合现实生活的。我们应该在赋值之前,做一次判断,数据校验,
也就是对即将要赋值的数据做判断。

问题又来了,我们应该在哪里定义这个判断逻辑呢?

StudentDemo2是一个测试类
一般情况下,测试类负责创建对象并使用对象的地方,在这里定义判断逻辑是不合适的。
所以我们这里应该定义在Student3类中。
又因为,我们在定义成员变量的时候无法加入逻辑判断。
所以,我们在成员方法中定义。
因为做数据判断的时候,加的是一些逻辑判断语句,肯定不是只有一条。
我们应该在Student3类中提供一个方法来数据校验。
按照我们的想法,在类中提供了一个数值校验的方法,
如果符合就成功赋值,反之则不赋值。
但是,谁规定了我定义了方法就一定会被使用呢,如果我还是用之前的获取成员变量的方式进行赋值,问题依旧存在那么我们给出的方法意义就不是很大了。
实际上,我们应该定义了一个方法后,让调用必须使用我们的方法进行赋值,并且不能让调用者直接获取到我们成员变量。

怎么去强制要求不直接获取成员变量呢?

针对这样的情况,java就提供了一个关键字:private
private: 私有的。
其实说到现在,就是为了引出一个思想:封装
封装:其实就是隐藏对象的属性和相关的实现细节,仅仅对外提供公共的访问方式
接下来看一个案例,具体的实现是如何的
package com.shujia.wxl.day08;

/*
        规范定义一个类的标准版本1.0
            1、将成员变量使用private修饰
            2、提供公共getXxx()和setXxx()
            3、提供一个方法打印所有的成员变量值(先自己定义,后面我们会学习新的方法替代)
 */
public class Teacher {
    //定义成员变量
    private String name;
    private int age;

    //提供公共的getXxx()和setXxx()方法
    public void setName(String s) {
        name = s;
    }

    public String getName() {
        return name;
    }

    public void setAge(int a) {
        age = a;
    }

    public int getAge(){
        return age;
    }

    //提供一个方法打印所有的成员变量值
    public void show() {
        System.out.println("姓名:" + name + ",年龄:" + age);
    }
}

class TeacherTest{
    public static void main(String[] args){
        //创建一个老师对象
        Teacher t1 = new Teacher();
        //使用成员变量
        //不能直接获取,因为它们都被private修饰了
//        System.out.println(t1.name+"---"+t1.age);
        //通过公共的方法来获取成员变量
        System.out.println(t1.getName()+"---"+t1.getAge());

        //给成员变量赋值
        t1.setName("刘志城");
        t1.setAge(18);
        System.out.println(t1.getName()+"---"+t1.getAge());
        t1.show();

    }
}

五、this关键字

我们还是先定义一个类用来讲解
package com.shujia.wxl.day08;

/*
      定义一个医生类:

      回想一下我们之前定义变量的时候,提过一个规范:见名之意。
      我们根据见名之意的规范来重新定义形参的名字,运行后发现结果不对,值没有赋上。
 */

class Doctor {
    private String name;
    private int age;
    private String gender;

    public void setName(String name) {
//我们还说过,局部变量的使用原则:就近原则
//我们这里的name采用就近原则后发现,使用的都是形参的变量名name,
没有使用成员变量中name.
//我们理想上应该是将传进来的name赋值给调用该方法对象中的成员变量name
//Doctor.name = name;直接通过类名.成员变量的方式目前是不行,
我们还没有说过类似的用法,所以这个是有问题的
//如果这个Doctor对象存在的话,它是不是就表示了一个医生对象
//如果有一个东西可以代表是当前调用该方法的对象就好了
//java提供了一个关键字:this
//this在方法中使用的时候,就代表着调用该方法的对象
//既然都是一个对象了,我们学过对象.成员变量
// Doctor.name = name;
        this.name = name;
    }

    public String getName() {
        return name;
    }

    public void setAge(int i) {
        age = i;
    }

    public int getAge() {
        return age;
    }

    public void setGender(String s) {
        gender = s;
    }

    public String getGender() {
        return gender;
    }

    //提供一个公共的方法
    public void show() {
        System.out.println("姓名:" + name + ",年龄:" + age + ",性别:" + gender);
    }

}

public class DoctorDemo {
    public static void main(String[] args) {
        //创建一个医生对象
        Doctor d = new Doctor();

        d.setName("wxl");
        d.setAge(18);
        d.setGender("男");
        d.show();
    }
}

为了方便大家理解我们再创建一个医生类
package com.shujia.wxl.day08;

/*
        定义一个标准类的2.0版本
            成员变量:被private修饰
            成员方法:getXxx()和setXxx(),其中setXxx(..)形参的名字与成员变量名字一致,使用this关键字赋值
                    show()
 */
class Doctor2 {
    private String name;
    private int age;
    private String gender;

    public void setName(String name) {
        this.name = name;
    }

    public String getName() {
        return name; //其实这里隐藏了一个this,但是我们也可以不写,因为就近原则取的就是对象的成员变量
    }

    public void setAge(int age) {
        this.age = age;
    }

    public int getAge() {
        return age;
    }

    public void setGender(String gender) {
        this.gender = gender;
    }

    public String getGender() {
        return gender;
    }

    public void show() {
        System.out.println("姓名:" + name + ",年龄:" + age + ",性别:" + gender);
    }

}

public class DoctorDemo2 {
    public static void main(String[] args) {
        //创建一个医生对象
        Doctor2 d = new Doctor2();

        //给成员变量进行赋值
        d.setName("明旺");
        d.setAge(18);
        d.setGender("男");

        d.show();

    }
}

并且附上内存图,this究竟在内存的什么地方?

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值