关于面向对象三大特性,我们可以很自信的回答:封装、继承、多态
之前学习的封装,可以很直观的理解为了保护数据,我们在idea中可以用alt+insert进行一个选择
constructer构造方法
getter和setter方法就是对封装的一个体现,我们一般将类中的数据设为private,这样new对象时可以防止用户对数据进行修改,而getter和setter方法的建立就是为了进行修改和获得数据
这样更能体现封装性
多个类中存在相同属性和行为时,将这些内容抽取到单独一个类中,那么多个类无需再定义这些属性和行为,只要继承那一个类即可。
其中,多个类可以称为子类,单独那一个类称为父类、超类(superclass)或者基类。
继承描述的是事物之间的所属关系,这种关系是: is-a 的关系。
上面的是一个基本的概述,很明显的可以看出只要我们有一个相似的属性,我们可以提取公共性质,形成一个父类,后面再有其他的类需要进行继承这些属性和方法的时候,我们就可以不需要重写那些已经写过的方法。
我们也可以对继承的方法进行覆盖重写。具体容后再说
继承的定义:
就是子类继承父类的属性和行为,使得子类对象具有与父类相同的属性、相同的行为。子类可以直接访问父类中的非私有的属性和行为。
继承的好处:
继承的格式:
class 父类 { ... } class 子类 extends 父类 { ... }
我们在子类中进行一个变量的定义,如果定义父类中未曾出现的变量,这是可以吗?
答案是可以的,子类中出现与父类不同的变量,是可以的。
子父类中出现了同名的成员变量时,在子类中需要访问父类中非私有成员变量时,需要使用super 关键字,修饰父类成员变量,类似于之前学过的 this 。
使用格式:
super.父类成员变量名
Fu 类中的成员变量是非私有的,子类中可以直接访问。若Fu 类中的成员变量私有了,子类是不能直接访问的。通常编码时,我们遵循封装的原则,使用private修饰成员变量,那么如何访问父类的私有成员
变量呢?对!可以在父类中提供公共的getXxx方法和setXxx方法。
我们需要进行父类方法的重写,我们应该如何去操作呢?
如果子类父类中出现重名的成员方法,这时的访问是一种特殊情况,叫做方法重写 (Override)。
方法重写 :子类中出现与父类一模一样的方法时(返回值类型,方法名和参数列表都相同),会出现覆盖效果,也称为重写或者复写。声明不变,重新实现。
建议:在子类重写的方法体上@override
注:
重载(overroad):可以不同的参数列表
关于构造方法:
super和this的含义
super :代表父类的存储空间标识(可以理解为父亲的引用)。
this :代表当前对象的引用(谁调用就代表谁)。
super和this的用法
this.成员变量 ‐‐ 本类的 super.成员变量 ‐‐ 父类的 this.成员方法名() ‐‐ 本类的 super.成员方法名() ‐‐ 父类的
this(...) ‐‐ 本类的构造方法 super(...) ‐‐ 父类的构造方法
子类的每个构造方法中均有默认的super(),调用父类的空参构造。手动调用父类构造会覆盖默认的super()。super() 和 this() 都必须是在构造方法的第一行,所以不能同时出现。
Java是单继承,不是多继承,也就是说一个类只允许有一个父亲
java支持多层继承
class A{} class B extends A{} class C extends B{}
最顶层的父类(祖宗)也就是object类
父类中的方法,被它的子类们重写,子类各自的实现都不尽相同。那么父类的方法声明和方法主体,只有声明还有意义,而方法主体则没有存在的意义了。我们把没有方法主体的方法称为抽象方法。Java语法规定,包含抽象方法的类就是抽象类。
定义:
抽象的单词是abstract,所以抽象的方法也用abstract进行一个修饰。
抽象方法只有方法名,没有方法体
定义格式:
修饰符 abstract 返回值类型 方法名 (参数列表);
代码举例:
public abstract void run();
抽象类
如果一个类包含抽象方法,那么这个类必须是抽象类
定义格式:
abstract class 类名字 { }
代码举例:
public abstract class Animal { public abstract void run(); }
继承抽象类的子类必须重写父类所有的抽象方法。否则,该子类也必须声明为抽象类。最终,必须有子类实现该父类的抽象方法,否则,从最初的父类到最终的子类都不能创建对象,失去意义。
注意事项:
关于抽象类的使用,以下为语法上要注意的细节,虽然条目较多,但若理解了抽象的本质,无需死记硬背。
这里给出具体代码,加深理解,我就不进行诠释了
Person类:
package cn.qioha.test; public class Person { private String name; private int money; public Person() { } public Person(String name, int money) { this.name = name; this.money = money; } public void show(){ System.out.println("my name is"+name+",money is"+money); } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getMoney() { return money; } public void setMoney(int money) { this.money = money; } }
Manager类:
package cn.qioha.test; import java.util.ArrayList; public class Manager extends Person{ public Manager(String name, int money) { super(name, money); } public ArrayList<Integer> send(int totalMoney,int count){ ArrayList<Integer> list = new ArrayList<>(); int money = super.getMoney(); super.setMoney(super.getMoney()-totalMoney); if(totalMoney > money){ System.out.println("your money not enough"); return list; } int m = totalMoney / count; for (int i = 0; i < count - 1; i++) { list.add(m); } int n = totalMoney % count; list.add(m+n); return list; } }
Member类:
package cn.qioha.test; import java.util.ArrayList; import java.util.Random; public class Member extends Person{ public Member(String name, int money) { super(name, money); } public void resive(ArrayList<Integer> list){ int num = new Random().nextInt(list.size()); int money = list.remove(num); super.setMoney(super.getMoney()+money); } }
demo(实例类):
package cn.qioha.test; import java.util.ArrayList; public class demo { public static void main(String[] args) { Manager ma = new Manager("peter",100); Member q = new Member("anna",0); Member w = new Member("nana",0); Member e = new Member("qing",0); ma.show(); q.show(); w.show(); e.show(); ArrayList<Integer> list = new ArrayList<>(); list = ma.send(40,3); q.resive(list); w.resive(list); e.resive(list); System.out.println("================"); ma.show(); q.show(); w.show(); e.show(); } }
doc文档注释我就没有进行一个撰写了,因为比较的简单,希望各位同行多多指教!