ToB企服应用市场:ToB评测及商务社交产业平台
标题:
Java面向对象(五)
[打印本页]
作者:
去皮卡多
时间:
2022-8-26 17:33
标题:
Java面向对象(五)
Java面向对象(五)
目录
Java面向对象(五)
十六、面向对象特征之三: 多态性
16.1 多态性的定义:
16.2 多态性的使用:
16.3 多态典型例题
十七、instanceof 关键字
17.1 instanceof 的引入:
17.2 instanceof 的使用:
17.3 向下转型和向上转型拓展
十八、Object 类的使用
18.1 Object 类介绍
18.2 == 与 equals() 方法
18.2.1 == 运算符
18.2.2 equals() 方法的使用
18.2.3 重写 equals() 方法
18.3 toString() 方法
十六、面向对象特征之三: 多态性
16.1 多态性的定义:
对象的多态性:父类的引用指向子类的对象(或子类的对象赋给父类的引用)。
可以直接应用在抽象类和接口上
。
Java 引用变量有两个类型:编译时类型和运行时类型。
编译时类型由声明该变量时使用的类型决定,运行时类型由实际赋给该变量的对象决定
。
简称:编译时,看左边;运行时,看右边。
若编译时类型和运行时类型不一致,就出现了对象的多态性。
public class Person {
......
}
public class Man extends Person{
......
}
public class Weman extends Person{
......
}
public class PersonTest {
public static void main(String[] args) {
Person p1 = new Person();
Man man = new Man();
//对象的多态性:父类的引用指向子类的对象
//编译时声明了一个Person类型的变量p2,运行时实际引用的是Man类型的实例对象
//所以,当调用子父类同名同参数的方法时,实际执行的是子类重写父类的方法 ---虚拟方法调用
Person p2 = new Man();
Person p3 = new Woman();
}
}
复制代码
16.2 多态性的使用:
虚拟方法调用(Virtual Method Invocation):
正常方法调用:
Person e = new Person();
e.getInfo();
Student e = new Student();
e.getInfo();
复制代码
虚拟方法调用(多态情况下):
子类中定义了与父类同名同参数的方法,在多态情况下,将此时父类的方法称为虚拟方法,父类根据赋给它的不同子类对象,动态调用属于子类的该方法。这样的方法调用在编译期是无法确定的。
Person e = new Student();
e.getInfo(); //调用Student类的getInfo()方法
/*
编译时e为Person类型,而方法的调用是在运行时确定的,所以调用的是Student类的getInfo()方法。——动态绑定
*/
复制代码
多态性的使用前提:① 类的继承关系 ② 方法的重写
对象的多态性,只适用于方法,不适用于属性(属性值编译和运行都看左边)。
多态性使用举例:
import java.sql.Connection;
//多态性的使用举例一:
public class AnimalTest {
public static void main(String[] args) {
AnimalTest test = new AnimalTest();
test.func(new Dog()); //输出:狗进食 狗叫
test.func(new Cat()); //输出:猫进食 猫叫
}
public void func(Animal animal){ //Animal animal = new Dog();
animal.eat();
animal.shout();
}
}
class Animal{
public void eat(){
System.out.println("动物:进食");
}
public void shout(){
System.out.println("动物:叫");
}
}
class Dog extends Animal{
public void eat(){
System.out.println("狗进食");
}
public void shout(){
System.out.println("狗叫");
}
public void watchDoor(){
System.out.println("狗看门");
}
}
class Cat extends Animal{
public void eat(){
System.out.println("猫进食");
}
public void shout(){
System.out.println("猫叫");
}
}
//举例二:
class Order{
public void method(Object obj){
}
}
//举例三:数据库连接
class Driver{
public void doData(Connection conn){//conn = new MySQlConnection(); / conn = new OracleConnection();
//规范的步骤去操作数据
// conn.method1();
// conn.method2();
// conn.method3();
}
}
复制代码
16.3 多态典型例题
public class InterviewTest1 {
public static void main(String[] args) {
Base1 base = new Sub1();
base.add(1, 2, 3); //输出:sub_1
Sub1 s = (Sub1)base;
s.add(1,2,3); //输出:sub_2
}
}
class Base1 {
public void add(int a, int... arr) {
System.out.println("base1");
}
}
class Sub1 extends Base1 {
public void add(int a, int[] arr) {
System.out.println("sub_1");
}
public void add(int a, int b, int c) {
System.out.println("sub_2");
}
}
// public void add(int a, int... arr) 和 public void add(int a, int[] arr) 是方法重写。
复制代码
十七、instanceof 关键字
17.1 instanceof 的引入:
有了对象的多态性以后,该对象实例在内存中不仅加载了父类的属性和方法,实际上是加载了子类特有的属性和方法的,但是由于变量声明为父类类型,导致编译时,只能调用父类中声明的属性和方法。子类特有的属性和方法不能调用。
如何才能调用子类特有的属性和方法?
向下转型:使用强制类型转换符。
Person p2 = new Man();
//p2变量声明为 Person 类型,不能调用子类 Man 类特有的方法
//p2.earnMoney();
//p2.isSmoking = true;
//向下转型
Man m1 = (Man)p;
m1.earnMoney();
m1.isSmoking = true;
//使用强转时,可能出现ClassCastException的异常。
// Woman w1 = (Woman)p2; 编译通过,运行报错。
// p2变量是一个Person类型的变量,但定义时new的是Man型,实际上指向Man型的对象
// w1.goShopping();
复制代码
为了避免在向下转型时出现 ClassCastException 的异常,我们在向下转型之前,先进行 instanceof 的判断,一旦返回 true,就进行向下转型。如果返回 false,不进行向下转型。
//问题一:编译时通过,运行时不通过
//举例一:
// Person p3 = new Woman();
// Man m3 = (Man)p3;
//举例二:
// Person p4 = new Person();
// Man m4 = (Man)p4;
//问题二:编译通过,运行时也通过
// Object obj = new Woman();
// Person p = (Person)obj;
//问题三:编译不通过
// Man m5 = new Woman();
// String str = new Date();
// 总结:可以这么理解,子类在创建时会加载所有的父类,转型也只能在已加载的类之间相互转型。
// 否则,就算是编译不报错,运行也会报错!
复制代码
17.2 instanceof 的使用:
a instanceof A:判断对象a是否是类A的实例。如果是,返回 true;如果不是,返回 false。
当类B是类A的父类。
如果 a instanceof A 返回 true,则 a instanceof B 也返回 true。
public class Person extends Object {…}
public class Student extends Person {…}
public class Graduate extends Person {…}
public void method1(Person e) {
if (e instanceof Person)
// 处理Person类及其子类对象
if (e instanceof Student)
//处理Student类及其子类对象
if (e instanceof Graduate)
//处理Graduate类及其子类对象
}
复制代码
17.3 向下转型和向上转型拓展
从子类到父类的类型转换可以自动进行。
从父类到子类的类型转换必须通过造型(强制类型转换)实现。
无继承关系的引用类型间的转换是非法的。(类型转换异常)
十八、Object 类的使用
18.1 Object 类介绍
Object 类是所有 Java 类的根父类。
如果在类的声明中未使用 extends 关键字指明其父类,则默认父类为 java.lang.Object 类。
Object 类中的功能(属性、方法)就具有通用性。
属性:无
方法:equals(),toString() ,getClass(),hashCode(),clone(),finalize(),wait(),notify(),notifyAll()。
Object 类只声明了一个空参的构造器。
18.2 == 与 equals() 方法
18.2.1 == 运算符
可以使用在基本数据类型变量和引用数据类型变量中。
如果比较的是基本数据类型变量:比较两个变量保存的数据是否相等。(不一定类型要相同)
public class EqualsTest {
public static void main(String[] args) {
//基本数据类型
int i = 10;
int j = 10;
double d = 10.0;
System.out.println(i == j);//true
System.out.println(i == d);//true
char c = 10;
System.out.println(i == c);//true
char c1 = 'A';
char c2 = 65;
System.out.println(c1 == c2);//true
}
}
复制代码
如果比较的是引用数据类型变量:比较两个对象的地址值是否相同.即两个引用是否指向同一个对象实体。
补充:
== 符号使用时,必须保证符号左右两边的变量类型一致。
18.2.2 equals() 方法的使用
1. 是一个方法,而非运算符。
2. 只能适用于引用数据类型。
3. Object类中equals()的定义:
复制代码
public boolean equals(Object obj) {
return (this == obj);
}
复制代码
说明:Object类中定义的equals()和==的作用是相同的:比较两个对象的地址值是否相同.即两个引用是否指向同一个对象实体。
像 String、Date、File、包装类等都重写了 Object 类中的 equals() 方法。
重写以后
,比较的不是两个引用的地址是否相同,而是
比较两个对象的"实体内容"是否相同
。
public class EqualsTest {
public static void main(String[] args) {
//引用类型:
Customer cust1 = new Customer("Tom",21);
Customer cust2 = new Customer("Tom",21);
System.out.println(cust1 == cust2);//false
String str1 = new String("xiaozhao");
String str2 = new String("xiaozhao");
System.out.println(str1 == str2); //false
System.out.println(cust1.equals(cust2)); //false(调用Object类中的equals()方法)
System.out.println(str1.equals(str2)); //true
Date date1 = new Date(32432525324L);
Date date2 = new Date(32432525324L);
System.out.println(date1.equals(date2)); //true
}
}
public class Customer {
public String name;
public int age;
}
复制代码
18.2.3 重写 equals() 方法
通常情况下,我们自定义的类如果使用equals()的话,也通常是比较两个对象的"实体内容"是否相同。
那么,我们就需要对 Object 类中的 equals() 进行重写。
重写的原则:比较两个对象的实体内容是否相同。
一般情况下,使用 IDE 自动重写 equals() 方法。
18.3 toString() 方法
当我们输出一个对象的引用时,实际上就是调用当前对象的 toString() 方法。
Object类中toString()的定义:
public String toString() {
return getClass().getName() + "@" + Integer.toHexString(hashCode());
}
复制代码
像 String、Date、File、包装类等都重写了 Object 类中的 toString() 方法。使得在调用对象的 toString() 时,返回"实体内容"信息。
自定义类也可以重写 toString() 方法,当调用此方法时,返回对象的"实体内容"。
可以手动重写和自动重写,一般使用自动重写。
public class ToStringTest {
public static void main(String[] args) {
Customer cust1 = new Customer("Tom",21);
System.out.println(cust1.toString()); //javase.ex.Customer@1b6d3586
System.out.println(cust1); //javase.ex.Customer@1b6d3586
String str = new String("xiaozhao");
System.out.println(str); //xiaozhao
Date date = new Date(4534534534543L);
System.out.println(date.toString()); //Mon Sep 11 08:55:34 CST 2113
}
}
public class Customer {
public String name;
public int age;
public Customer(String name, int age) {
this.name = name;
this.age = age;
}
/*自动重写的 toString 方法
@Override
public String toString() {
return "Customer{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
*/
}
复制代码
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
欢迎光临 ToB企服应用市场:ToB评测及商务社交产业平台 (https://dis.qidao123.com/)
Powered by Discuz! X3.4