Java学习笔记-封装、继承、多态

Java学习笔记-封装、继承、多态

(此学习笔记基于狂神的课堂加上自己的理解总结~)

1、封装

  • 该显示的显示,该隐藏的隐藏

    • 程序设计的时候要追求 “高内聚,低耦合” 。高内聚就是类的内部数据操作细节自己完成,不允许外部干涉;低耦合即为仅暴露少量的方法给外部使用
  • 数据的隐藏

    • 通常,应禁止直接访问一个对象中数据的实际表示,而是应该通过接口来访问,这成为信息隐藏
  • 封装的意义

    • 提高程序的安全性,保护数据
    • 隐藏代码的实现细节
    • 运用了同一接口
    • 系统可维护性增加
//定义一个员工类
public class Employee{
    
    private int eID;			//员工ID,设为私有属性,外部无法直接访问
    private String eName;		//员工姓名,私有,外部也无法直接访问
    
    public int getEID(){			//外部对员工ID获取的唯一接口:getEID()
        return this.eID;
    }
    
    public void setEID(int eID){	//外部对员工ID设置的唯一接口:setEID()
        this.eID = eID;
    }
    
    public String getEName(){		//外部对员工姓名获取的唯一接口:getEName()
        reuturn this.eName;
    }
    
    public void setEName(String eName){		//外部对员工姓名设置的唯一接口:setEName()
        this.eName = eName;
    }
}
//在这个类中去对员工进行一些实质性的操作
public class Application{
    public static void main(String[] args){
        Employee emp1 = new Employee();		//实例化一个员工对象
        
        //注意,这个时候emp1.eID是会报错的,因为在员工类中该成员变量的访问修饰符为private
        
        emp1.setEID(1);		//通过我们在员工类中自定义的setEID方法去设置当前对象的员工ID
        emp1.setEName("张三");	//通过我们在员工类中自定义的setEName方法去设置当前对象的员工姓名
        
        //那么此时emp1对象所对应的员工就是ID为1的名字为'张三'的员工
        
        System.out.println(emp1.getEID());		//通过我们在员工类中自定义的getEID方法去获取当前对象的员工ID
        System.out.println(emp1.getEName());	//通过我们在员工类中自定义的getEName方法去获取当前对象的员工姓名
    }
}

那么可能到这里,还是会有朋友不明白为什么要多此一举做这个操作,我们再举个例子:

//定义一个“人”类
public class Human{
    
    private int age;		//假设当前类中只有年龄属性
    
    public int getAge(){	//创建对age属性的获取方法
        return this.age;
    }
    
    public void setAge(int age){	//创建对age属性的设置方法
        this.age = age;
    }
}
public class Application{
    public static void main(String[] args){
        Human human1 = new Human();		//创建一个人类的实例
        human1.setAge(999);		//显然,在现实生活中,年龄为999是非法的,那么我们只需要在Human类中对其设置方法做一些限定就								  //能实现规范化用户对属性的输入
    }
}
//改进后的“人”类
public class Human{
     private int age;		//年龄
    
    public int getAge(){	//创建对age属性的获取方法
        return this.age;
    }
    
    public void setAge(int age){	//创建对age属性的设置方法
        if(age < 0 || age > 120)		//对传回来的age进行判断,如果为年龄输入不合法,则将年龄改成18
            this.age = 18;
        else							//如果传回来的age合法,则修改当前对象的age
            this.age = age;
    }
}

2、继承(extends)

  • 继承的本质是对某一批类的抽象,从而实现对现实世界更好的建模

  • extends的意思是“扩展”,子类是父类的扩展

  • Java中类只有单继承,没有多继承(一个儿子只有一个爸爸,一个爸爸能有多个儿子)

  • 继承是类与类之间的一种关系,除此之外,类与类之间的关系还有依赖、组合、聚合等

  • 继承关系的两个类,一个叫子类(派生类),一个叫父类(基类)。子类继承父类,使用关键字extends来表示

  • 子类与父类之间的关系,从意义上讲应该具有“is a”的关系

  • 父类中private的成员变量和方法子类都无法继承

  • 在Java中,所有的类都默认直接或间接继承Object类

示例:

public class Human{
    
    public double height;		//人类共有的属性身高
    public double weight;		//体重
    
    public void walk(){			//走路
        System.out.println("I walk");
    }
}
public class Student extends Human{		//学生类继承人类。学生也是人
    public static void main(String[] args){
        Student student = new Student();
        student.height = 185;		//设置此学生的身高
        student.weight = 130;		//设置此学生的体重
        
        System.out.println(student.height + " " + student.weight);	//输出学生的身高和体重
        student.walk();			//学生走路
    }
}

我们可以看到,在学生类中,我们并没有定义学生类自己的height和weight,但是仍然可以调用该对象中的height和weight并进行赋值,就是因为在学生类中继承了父类->人类,而人类中有public修饰的属性height和weight,当子类继承父类的时候,子类学生类也拥有了父类的这两个属性,包括下面的walk()方法。

Super
  • super代表父类存储空间的标识,super可以通过这个标识访问父类的成员,super可以理解为父类对象
  • 调用父类的构造器,必须要在子类构造器的第一行
  • super必须只能出现在子类的方法或者构造方法中
  • super和this不能同时调用构造方法
  • 注意区分this和super ->super的详情
public class Human {	//“人”类 

    public Human(){
        System.out.println("Human的无参构造器执行");	//Human类的无参构造方法
    }

    public String name = "张三";		//Human类中的name

}
public class Employee extends Human {		//子类员工类继承父类人类

    public Employee(){
        //隐藏代码super();
        System.out.println("Employee无参构造器执行");		//Employee类的无参构造方法
    }

    public String name = "李四";		//Employee类的name

    public void print(){
        System.out.println(super.name);		//此时打印的是Human类中的name,也即张三
    }

    public void print(String name){
        this.name = name;				//此方法中带有传进来的形式参数name,赋值给Employee类中的name
        System.out.println(this.name);		//此时打印出来的是Employee中的已被赋值的name
    }

}

public class Application {
    public static void main(String[] args) {
        Employee emp1 = new Employee();		//实例化Employee,将下面内容注释,仔细看此时会输出的内容
        System.out.println(emp1.name);
        emp1.print();
        emp1.print("王五");
    }
}
重写

需要有继承关系,子类重写父类的方法

  1. 方法名必须相同

  2. 参数列表必须相同

  3. 修饰符:范围可以扩大但不能缩小

  4. 抛出的异常范围可以被缩小但不能扩大

    子类的方法必须和父类一致,方法体不同

    为什么需要重写:

    ​ 父类的功能子类不一定需要,或不一定满足

public class Car {

    public void print(){
        System.out.println("This is a car");
    }

}

public class Toyota extends Car {

    @Override
    public void print(){		//对父类的print方法进行重写
        System.out.println("This is Toyota");
    }

}

public class Application {
    public static void main(String[] args) {

        Car car = new Toyota();
        car.print();	//此时输出的是"This is Toyota"
        Toyota toyota = new Toyota();	
        toyota.print();		//此时输出的"This is Toyota"
        Car car2 = new Car();
        car2.print();		//此时输出的是"This is a car"
        
    }
}

3、多态

  • 即同一方法可以根据发送对象的不同而采取不同的行为方式
  • 一个对象的实际类型是确定的,但可以指向对象的引用的类型有很多
  • 多态存在的条件
    • 有继承关系
    • 子类重写父类方法
    • 父类引用指向子类对象
  • 注意:多态是方法的多态,属性没有多态性
    • static修饰的方法不能重写,他属于类,不属于实例
    • final修饰的方法也不能被重写,此时的方法已放入常量池
    • private修饰的方法也不能被重写
  • 对象能执行哪些方法主要看对象左边的类型,和右边关系不大,如果左边的类中的方法被右边的重写了则执行重写过的方法。
public class Human{
    public void walk(){
        System.out.println("walk");
    }
}
public class Student extends Human{
    @Ovrerride
    public void walk(){
        System.out.println("I walk");
    }
    
    public void eat(){
        System.out.println("I eat");
    }
}
public class Application{
    public static void main(String[] args){
        Student s1 = new Student();
        Human s2 = new Student();
 		Human s3 = new Human();
        
        s1.walk();		//此时的输出结果为I walk
        s1.eat();		//此时的输出结果为I eat
        
        s1.run();		//此时的输出结果为I walk
        s2.eat();		//此时程序会报错,因为在Human类中没有eat()方法,即使引用指向的是学生对象也无法调用
        
        h3.walk();		//此时的输出结果为walk
        
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值