JAVA面向对象基础--封装 继承 多态
一、封装
- 该显示的显示,该隐藏的隐藏
- 程序设计追求“高内聚,低耦合”
- 高内聚:类的内部数据操作细节自己完成,不允许外部干涉
- 低耦合:仅暴露少量的方法给外部使用
- 封装即数据的隐藏,通常应禁止直接访问一个对象中数据的实际表示,而通过操作接口来访问,这称为信息隐藏
- 属性私有,通过get/set方法操作数据
1、封装的特点
- 提高程序的安全性,保护数据
- 隐藏代码的实现细节
- 统一接口
- 增加系统的可维护性
2、举例
- 创建一个Student类
//private : 私有
public class Student02 {
//属性私有
private String name;//姓名
private int age;//年龄
private char sex;//性别
//提供一些可以操作这些属性的方法
//get/set方法
//get方法:获取这个数据
public String getName(){
return this.name;
}
//set方法:给这个数据设置值
public void setName(String name){
this.name = name;
}
}
- 通过调用get,set方法赋值获取
public class Appliance {
public static void main(String[] args) {
Student02 s1 = new Student02();
s1.setName("张三");
System.out.println(s1.getName());//输出张三
}
}
注意:当属性前添加了private时,在main方法中直接使用s1.name,如图:
注意:快捷键Alt+Insert(Fn+Alt+Insert),可以自动生成get set方法,步骤如下:
- 按下Alt+Insert-->选择Getter and Setter
- 选择需要的属性
3、封装的好处--判断输入合法性
//private : 私有
public class Student02 {
//属性私有
private int age;//年龄
//get方法:获取这个数据
public int getAge() {
return age;
}
//set方法:设置这个数据的值
public void setAge(int age) {
//进行合法性判断
if(age<120 && age>0){
this.age = age;
}else {
this.age = 3;
}
}
}
public class Appliance {
public static void main(String[] args) {
Student02 s1 = new Student02();
//s1.setAge(130);//错误数据
//System.out.println(s1.getAge());//输出3
s1.setAge(18);//有效数据
System.out.println(s1.getAge());//输出18
}
}
二、继承
1、什么是继承?
-
继承的本质是对某一批类的抽象,从而实现对现实世界更好的建模
-
继承关系的两个类,一个为子类(派生类),一个为父类(基类)。
- 子类继承父类,使用关键字extends来表示
- extends的意思是“扩展”,子类是父类的扩展
- 子类和父类之间,从意义上来讲具有”is a“的关系
-
继承是类和类之间的一种关系。除此之外,类和类之间的关系还有依赖、组合、聚合等
-
Java中类只有单继承,没有多继承(即子类只能有一个父类,一个父类可以有多个子类)
实例如下:
- 先创建一个Person类作为父类
//父类
public class Person {
//私有属性
private String name;
//方法
public void say(){
System.out.println("在说话!");
}
//get set方法
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
(注意:get/set方法请查看封装介绍)
- 创建一个Student类和Teacher类作为子类
//Person的子类:子类继承父类的属性和方法
public class Student extends Person {
}
//Person的子类:子类继承父类的属性和方法
public class Teacher extends Person {
}
- 再创建一个带有main方法的Application类
Student类中什么也没有,但是继承了父类中的name属性和方法,可以使用
public class Application {
public static void main(String[] args) {
Student student = new Student();
student.setName("张三");//获取父类中的属性name
System.out.print(student.getName());//输出张三
student.say();//调用父类中的say()
}
}
2、Object类
-
快捷键Ctrl+H:查看类之间的父子类关系
-
注意:需把鼠标停留在某个类中,再按快捷键
-
图例如下:
注意:我们发现最上层有一个Object类,是Person类的父类
- 每一个类默认直接或间接的以Object类作为父类!!!!
例如:
public class Person /*extends Object*/ {
//什么也没写
}
public class Application {
public static void main(String[] args) {
Person person = new Person();
//person对象还是可以点出很多方法,这些方法都是Object类的方法
person.toString();
person.equals();
person.getClass();
}
}
以下均为Object类中的方法:
3、super
用于调用父类中的方法和属性
实例:
- Person类(父类)
public class Person /*extends Object*/ {
//protected:受保护的
protected String name = "张三";
//print方法
public void print(){
System.out.println("Person");
}
}
- Student类(子类)
//Person的子类
public class Student extends Person {
private String name = "李四";
//test方法
public void test(String name){
//this.name中的name表示Student类中的name属性
//等号右边的name表示test方法的参数name
this.name = name;
//super.name中的name表示父类Person类中的name属性
super.name = name;
System.out.println(this.name);
System.out.println(super.name);
}
//print方法
public void print(){
System.out.println("Student");
}
//方法调用也是一样
public void test_2(){
print();//调用本类的print方法,输出Student
this.print();//调用本类的print方法,输出Student
super.print();//调用父类的print方法,输出Person
}
}
- Application类,包含main方法
public class Application {
public static void main(String[] args) {
Student student = new Student();
student.test("王五");
System.out.println("================");
student.test_2();
}
}
运行结果如下图:
注意点:
- super调用父类的构造方法,必须在构造方法的第一个
- super必须且只能出现在子类的方法或构造方法中
public class Person {
public Person() {
System.out.println("Person类的无参执行了!");
}
}
public class Student extends Person {
//无参构造
public Student() {
System.out.println("Student类的无参执行了!");
}
}
public class Application {
public static void main(String[] args) {
Student student = new Student();//只new了一个对象,其他操作不做
}
}
运行结果如下:Person类(父类)的构造方法先执行了,之后才执行了Student类(子类)的构造方法!!!!
原因如下:Student类中的无参构造有一句隐藏代码super(),调用了父类的无参构造
注意:super();必须放在构造器中的第一行位置,其他位置会报错无法调用
public class Student extends Person {
//无参构造
public Student() {
//隐藏代码:调用了父类的无参构造(代码如下:)
super();//调用父类的构造器必须放在子类构造器的第一行
System.out.println("Student类的无参执行了!");
}
}
- super和this不能同时调用构造方法
Student类修改如下:
public class Student extends Person {
//无参构造
public Student() {
//隐藏代码:调用了父类的无参构造(代码如下:)
super();//调用父类的构造器必须放在子类构造器的第一行
this();//调用本类的有参构造 //报错!!!
System.out.println("Student类的无参执行了!");
}
//有参构造
public Student(String name) {
}
}
报错:报错内容即‘this()'必须放在构造器的第一行,即super和this无法同时调用
- 当父类只有有参构造器,没有无参构造器时,子类的无参构造中必须调用父类的有参构造,否则子类无法定义无参构造
注意:当父类的无参构造器显示定义时即有无参,super()不会报错!!
public class Person {
//有参构造
public Person(String name) {
System.out.println("Person类的无参执行了!");
}
}
public class Student extends Person {
//无参构造
public Student() {
//隐藏代码:调用了父类的无参构造(代码如下:)
super();//报错!!!!!
//原因:父类没有无参构造
System.out.println("Student类的无参执行了!");
}
}
-
报错原因:父类没有无参构造
-
**解决办法:利用super("张三");调用父类的有参构造器如下:
super("张三");
super 与 this
- 代表的对象不同:
- this:本身调用者这个对象
- super:代表父类对象的应用
- 使用前提:
- this:没有继承也可以使用
- super:只有在继承条件下才可以使用
- 构造方法:
- this():本类的构造
- super():父类的构造
三、重写
重写都是对方法的重写,与属性无关
静态方法和非静态方法的差别很大!!!
注意:父子类需是同一种方法,即均为静态或均为非静态
1、静态方法:
- 方法的调用只和左边定义的数据类型有关
//父类
public class B {
//静态方法
public static void test(){
System.out.println("B=>test()");
}
}
//子类
public class A extends B {
//静态方法
public static void test(){
System.out.println("A=>test()");
}
}
public class Application {
public static void main(String[] args) {
A a = new A();
a.test();//输出A=>test();
B b = new A();//父类的引用(b)指向了A类
b.test();//输出B=>test();!!!!
}
}
2、非静态方法:
- 重写
注意:重写的方法不能是私有的方法(即private的方法)
//父类
public class B {
//非静态方法
public void test(){
System.out.println("B=>test()");
}
}
//子类
public class A extends B {
//非静态方法
//子类重写了B(父类)的方法,那么就是执行子类的方法
public void test(){
System.out.println("A=>test()");
}
}
public class Application {
public static void main(String[] args) {
A a = new A();
a.test();//输出A=>test();
B b = new A();
b.test();//输出A=>test();!!!!!
}
}
快速重写父类方法:
- 在子类中按快捷键Alt+Insert(Fn+Alt+Insert),选择Override Methods...
- 点击后会自动生成以下代码
//@Override:重写
@Override //注解:有功能的注释
//子类重写了B(父类)的方法,那么就是执行子类的方法
//没有重写则调用父类的方法
public void test() {
//内容可自行修改
System.out.println("A=>test()");
//super.test();//自动生成,super用于调用父类的方法
}
3、小结:
重写需要有继承关系,子类重写父类的方法!
- 方法名必须相同
- 参数列表必须相同
- 修饰符:范围可以扩大但不能缩小
- public>Protected>Default>private
- 抛出的异常:范围可以被缩小但不能扩大
- 例如:ClassNotFoundException(小) --->Exception(大)
- 子类的方法和父类的方法必须一致,但方法体不同
- 方法重写快捷键:Alt+Insert
为什么要重写?
- 父类的功能,子类不一定需要或者不一定满足
四、多态
多态:即同一方法可以根据发送对象的不同而采用多种不同的行为方式
- 一个对象的实际类型是确定的,但可以指向对象的引用类型有很多(父类,有关系的类)
1、情况一:无重写
//父类
public class Person {
public void run(){
System.out.println("run");
}
}
//父类
public class Person {
public void run(){
System.out.println("run");
}
}
//子类
public class Student extends Person {
public void say(){
System.out.println("say");
}
}
//测试类
public class Applicaton {
public static void main(String[] args) {
//一个对象的实际类型是确定的
//new Student(); 或者 new Person();
//但一个对象可以指向的引用类型是不确定的:父类的引用可以指向子类
//Studnet子类:可以调用自己的方法或者继承父类的方法
Student s1 = new Student();
//Person父类:引用对象可以指向子类,但无法调用子类独有的方法
Person s2 = new Student();
Object s3 = new Student();
//对象可以执行哪些方法,主要是看对象左边的类型,和右边关系不大!!
s1.run();//子类调用父类的方法,输出run
s2.run();//输出run
//s2.say();//错误!!!父类引用对象无法调用子类独有的方法
//((Student)s2).say();//强制类型转换即可调用
}
}
2、情况二:重写
Student类重写了父类的say()方法:
//子类
public class Student extends Person {
//重写父类方法:会执行子类的方法
@Override
public void run() {
System.out.println("run_02");
}
public void say(){
System.out.println("say");
}
}
测试类和Person类不变:
//父类
public class Person {
public void run(){
System.out.println("run");
}
}
//测试类
public class Applicaton {
public static void main(String[] args) {
//Studnet子类:可以调用自己的方法或者继承父类的方法
Student s1 = new Student();
//Person父类:引用对象可以指向子类,但无法调用子类独有的方法
Person s2 = new Student();
Object s3 = new Student();
s1.run();//输出run_02
s2.run();//重写父类方法:会执行子类的方法 输出run_02!!!!
}
}
3、注意事项:
- 多态是方法的多态,属性没有多态
- 多态的使用需要类之间有父子类关系
- 类型转换异常会报错:ClasCastException
- 多态存在条件:
- 需要有父子类关系(继承关系)
- 方法需要重写
- 父类的引用指向子类对象
- Father f1 = new Son();
- 无法被重写的方法有:
- static方法:静态方法属于类和类一起加载,不属于实例,无法被重写
- final 常量
- private 方法
五、instanceof
判断两个类是否相似,是否可以比较转换(类型转换),相似返回true,不相似返回false,语法如下:
注意:有关系的两个类才能进行比较
System.out.println(X instanceof Y);
public class Application {
public static void main(String[] args) {
//instanceof 二者有关系才能进行判断
//Object > String
//Object > Person > Student
//Object > Person > Teacher
Object obj = new Student();
System.out.println(obj instanceof Person);//True
System.out.println(obj instanceof Student);//True
System.out.println(obj instanceof Object);//True
System.out.println(obj instanceof Teacher);//False; //Student类和Teacher类是同级
System.out.println(obj instanceof String);//False; //String是Object的子类
System.out.println("==============================");
Person person = new Student();
System.out.println(person instanceof Person);//True
System.out.println(person instanceof Student);//True
System.out.println(person instanceof Object);//True
System.out.println(person instanceof Teacher);//False;
//System.out.println(person instanceof String);//编译报错 //Person与String没有联系无法比较
System.out.println("==============================");
Student student = new Student();
System.out.println(student instanceof Person);//True
System.out.println(student instanceof Student);//True
System.out.println(student instanceof Object);//True
//System.out.println(student instanceof Teacher);//编译报错 //Student类和Teacher类没有联系无法比较
//System.out.println(person instanceof String);//编译报错
}
}
1、强制类型转换
//父类
public class Person {
public void to(){
System.out.println("to");
}
}
//子类
public class Student extends Person {
public void go(){
System.out.println("go");
}
}
//测试类
public class Application {
public static void main(String[] args) {
//类型转换:父类 子类
Student s1 = new Student();
s1.go();//子类方法
//子类转为父类,可能会丢失自己本来的一些方法
//子类可以直接转为父类
Person s2 = s1;
s2.to();//父类方法
//父类需强制转换为子类
Person p1 = new Person();
//p1.go();//报错,父类无法使用子类的方法,需要强制转换
//强制转换
Student p2 = (Student) p1;
p2.go();//子类方法
//以上两句合为一句如下:
((Student)p1).go();
}
}
2、小结:
- 父类引用指向子类的对象‘
- 向上转型:把子类转为父类
- 向下转型:把父类转为子类(强制转换)
- 强制转换的作用:
- 方便方法的调用,减少重复代码