【自学笔记】01Java基础-07面向对象基础-03常量、枚举类、抽象类、多态详解

记录java基础学习中有关常量、枚举类、抽象类和多态的内容。

1 常量

  • 什么是常量
    • 常量是使用了public static final修饰的成员变量,必须有初始化值,而且执行的过程中其值不能被改变
  • 常量名的命名规范:英文单词全部大写,多个单词下划线连接起来。
public class Constant { public static final String SCHOOL_NAME  = “清华校园";public static final String LOGIN_NAME  = “admin";public static final String PASS_WORD  =123456";
} 
  • 常量的作用:通常用来记录系统的配置数据。

2 枚举

枚举(enum)是一种特殊的类(class),它用于盛放一组有限且确定的常量集合。

2.1 枚举的声明

修饰符 enum  枚举名称{
   第一行都是罗列枚举类实例的名称。每个实例实际是构造方法调用而来。
}

在这里插入图片描述

在这个例子中,每个枚举常量都是枚举类型的唯一实例,并且在编译时就被默认调用无参构造创建了。

2.2 枚举的特点:

  1. 类型安全:枚举类型确保只能使用预定义的值,从而避免了拼写错误或意外地使用非法值的情况。

  2. 单例性:枚举中的每个元素都是一个单例对象,这意味着无论何时何地创建枚举实例,同一枚举常量的引用始终指向相同的内存地址。

  3. 继承自Enum类:所有的枚举都隐式地继承自 java.lang.Enum 类,因此可以访问到 Enum 类提供的方法,如 name()(返回枚举常量的名称)、ordinal()(返回枚举常量在其枚举类型中的位置索引)等。

  4. 可定制化:枚举不仅可以包含预定义的常量,还可以定义自己的方法、属性以及实现接口:

    public enum Color {
    //相当于调用了构造方法
    //public static final Color RED = new Color("红色");RED("红色"),GREEN("绿色"),BLUE("蓝色");private String chineseName;// 有参构造器Color(String chineseName) {this.chineseName = chineseName;}// 自定义方法public String getChineseName() {return this.chineseName;}
    }
    
  5. switch支持:Java的switch语句可以直接支持枚举类型作为判断条件。

  6. 序列化与反序列化支持:Java枚举类型自动实现了Serializable接口,可以被序列化和反序列化。

  7. 线程安全:枚举类的实例创建过程天然具有线程安全性。

2.3 使用场景

枚举在实际开发中广泛应用,例如:

  • 表示状态机的状态,如订单状态(PENDING、COMPLETED、CANCELLED)。
  • 星期几、月份、颜色、方向等固定数量且预先知道所有可能值的场合。
  • 在设计模式中,例如策略模式或者工厂模式,用以替代传统的基于字符串标识符的选择逻辑。

3 抽象类

abstract关键字修饰修饰的类或方法就是抽象类或抽象方法。详解见abstract关键字
当编程中遇到暂不明确实现的类或方法时,可以声明为抽象类/方法,在之后以继承的方式实现。

  • 抽象的使用场景
    • 抽象类可以理解成不完整的设计图,一般作为父类,让子类来继承。
    • 当父类知道子类一定要完成某些行为,但是每个子类该行为的实现又不同,于是该父类就把该行为定义成抽象方法的形式,具体实现交给子类去完成。此时这个类就可以声明成抽象类。

在这里插入图片描述

3.1 抽象类案例

  • 系统需求
    • 某加油站推出了2种支付卡,一种是预存10000的金卡,后续加油享受8折优惠,另一种是预存5000的银卡 ,后续加油享受8.5折优惠。
    • 请分别实现2种卡片进入收银系统后的逻辑,卡片需要包含主人名称,余额,支付功能。
  • 分析实现
    • 创建一张卡片父类:定义属性包括主人名称、余额、支付功能(具体实现交给子类)
    • 创建一张白金卡类:重写支付功能,按照原价的8折计算输出。
    • 创建一张银卡类:重写支付功能,按照原价的8.5折计算输出。
// 抽象卡片类:定义基本属性和支付功能接口
public abstract class Card {private String ownerName;private double balance;public Card(String ownerName, double initialBalance) {this.ownerName = ownerName;this.balance = initialBalance;}// 获取主人名称public String getOwnerName() {return ownerName;}// 获取余额public double getBalance() {return balance;}// 抽象的支付方法,由子类具体实现public abstract void pay(double originalPrice);// 充值方法public void recharge(double amount) {if (amount > 0) {balance += amount;}}
}// 金卡子类:实现8折优惠支付功能
public class PlatinumCard extends Card {public PlatinumCard(String ownerName, double initialBalance) {super(ownerName, initialBalance);}@Overridepublic void pay(double originalPrice) {double discountedPrice = originalPrice * 0.8;if (discountedPrice <= balance) {System.out.println("金卡用户" + ownerName + "成功支付:" + discountedPrice);balance -= discountedPrice;} else {System.out.println("金卡用户" + ownerName + "余额不足,无法完成支付!");}}
}// 银卡子类:实现8.5折优惠支付功能
public class SilverCard extends Card {public SilverCard(String ownerName, double initialBalance) {super(ownerName, initialBalance);}@Overridepublic void pay(double originalPrice) {double discountedPrice = originalPrice * 0.85;if (discountedPrice <= balance) {System.out.println("银卡用户" + ownerName + "成功支付:" + discountedPrice);balance -= discountedPrice;} else {System.out.println("银卡用户" + ownerName + "余额不足,无法完成支付!");}}
}

现在可以在收银系统中实例化不同类型的卡片,并调用其支付方法进行支付操作。例如:

public class CashierSystem {public static void main(String[] args) {Card platinumCard = new PlatinumCard("张三", 10000);Card silverCard = new SilverCard("李四", 5000);platinumCard.pay(200);  // 金卡用户张三成功支付:160.0silverCard.pay(150);   // 银卡用户李四成功支付:127.5// 更多支付操作...}
}

3.2 final和abstract是什么关系?

  • 互斥关系。
  • abstract定义的抽象类作为模板让子类继承,final定义的类不能被继承。
  • 抽象方法定义通用功能让子类重写,final定义的方法子类不能重写。

3.3 设计模式之模板方法模式

当系统中出现同一个功能多处在开发,而该功能中大部分代码是一样的,只有其中部分可能不同的时候。
实现步骤:
1、定义一个抽象类。
2、定义2个方法,把相同代码放模板方法中,建议定义为final让子类不可修改,不同代码定义成抽象方法。
3、子类继承抽象类,重写抽象方法。

4 多态

4.1 什么是多态?

多态,指对象可以有多种形态。
同类型的对象,执行统一行为,可以表现出不同的特征。如动物类下的猫和狗对象,发出叫声,声音不同。

多态实现了在同一个父类中使用不同子类的对象,对同一消息做出不同的响应。

多态的常见形式
父类类型 对象名称 = new 子类构造器();创建子类对象,并将其引用赋值给父类类型的变量

class Animal {// ...
}class Dog extends Animal {public Dog() {// 这里是Dog类的构造方法}
}public class Main {public static void main(String[] args) {Animal animal = new Dog(); // 创建Dog对象,并将Dog对象的引用赋值给Animal类型的变量		}
}

多态中成员访问特点
 方法调用:编译看左边确认是哪个父类,运行看右边确认是哪个子类。
 变量调用:编译和运行都看左边确认是哪个父类。

Java中的多态(Polymorphism)是面向对象编程的三大特性之一,另外两个特性是封装和继承。多态允许不同类的对象对同一消息做出不同的响应,它增强了代码的灵活性、可扩展性和重用性。

多态的概念:

  1. 静态多态(编译时多态):通过方法重载(Overloading)实现,即在同一个类中可以有多个同名的方法,但参数列表不同(参数数量、类型或顺序不同)。编译器根据调用方法时提供的参数自动选择对应的方法执行。

  2. 动态多态(运行时多态):通过继承和接口实现,具体表现为:

    • 方法重写(Override):子类继承父类并覆盖其非私有的实例方法。
    • 父类引用指向子类对象:声明为父类类型的变量可以引用子类对象,在程序运行期间调用实际对象的方法时,会调用该对象的实际类型所重写的方法。

多态示例:

class Animal {void sound() {System.out.println("Animal makes a sound");}
}class Dog extends Animal {@Overridevoid sound() {System.out.println("Dog barks");}
}public class PolymorphismExample {public static void main(String[] args) {Animal animal = new Animal(); // 动物对象Animal dog = new Dog(); // 动物引用指向狗对象animal.sound();  // 输出 "Animal makes a sound"dog.sound();     // 输出 "Dog barks",由于dog实际上是Dog对象,因此调用的是Dog类重写的sound方法// 这里体现了运行时多态,尽管animal变量被声明为Animal类型,但它实际绑定的是Dog对象}
}

注意事项:

  • 调用方法时遵循“动态绑定”,即在运行时决定调用哪个方法。但对于静态方法、final方法以及private方法,不存在多态,它们是在编译时期就确定了要调用的方法。
  • 当父类引用指向子类对象时,只能访问父类中定义的属性和方法,不能直接访问子类新增加的属性和方法,除非进行显式向下转型操作。

4.2 多态的优势与劣势

多态的好处:

  • 提高代码通用性,可以使用父类类型作为方法的参数或者返回值类型,使得方法能够处理多种子类对象。
  • 增强可扩展性,当添加新的子类时,无需修改父类的代码就能与新子类协同工作。
  • 遵循里氏替换原则,定义方法的时候,使用父类型作为参数,任何基类出现的地方都可以用子类替代,而不会影响程序的正确性。

多态的劣势:

  • 多态下不能使用子类的独有功能。

4.3 父类如何实现调用子类独有功能

将对象类型由父类转换为子类即可。
在这里插入图片描述

// 定义一个父类 Animal
class Animal {public void makeSound() {System.out.println("动物发出声音");}
}// 定义一个子类 Dog 继承自 Animal,并添加自己特有的方法
class Dog extends Animal {public void bark() {System.out.println("小狗汪汪叫");}@Overridepublic void makeSound() {System.out.println("狗发出声音");}
}public class Main {public static void main(String[] args) {// 创建一个 Dog 对象,但使用 Animal 类型的引用指向它Animal animal = new Dog(); // 只能调用 Animal 接口中的方法animal.makeSound(); // 输出 "狗发出声音",多态的表现// 虽然animal实际上是Dog对象,但由于类型是Animal,所以无法直接调用bark()// 若要调用Dog类特有的bark()方法,必须进行强制类型转换if (animal instanceof Dog) { // 检查是否可以安全转换Dog dog = (Dog) animal; // 强制类型转换dog.bark(); // 现在可以调用Dog类特有的方法,输出 "小狗汪汪叫"}}
}

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/608655.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【JAVA】throw 和 throws 的区别?

&#x1f34e;个人博客&#xff1a;个人主页 &#x1f3c6;个人专栏&#xff1a; JAVA ⛳️ 功不唐捐&#xff0c;玉汝于成 目录 前言 正文 throw&#xff1a; throws&#xff1a; 区别&#xff1a; 作用&#xff1a; 使用位置&#xff1a; 个数&#xff1a; 应…

软件测试|Python对JSON的解析和创建详解

简介 JSON&#xff08;JavaScript Object Notation&#xff09;是一种轻量级的数据交换格式&#xff0c;已经成为当今互联网应用中广泛使用的数据格式之一。Python提供了内置的模块来解析和创建JSON数据&#xff0c;使得在Python中处理JSON变得非常简单。本文将详细介绍Python…

【题解】—— LeetCode一周小结

1.经营摩天轮的最大利润 题目链接&#xff1a; 1599. 经营摩天轮的最大利润 你正在经营一座摩天轮&#xff0c;该摩天轮共有 4 个座舱 &#xff0c;每个座舱 最多可以容纳 4 位游客 。你可以 逆时针 轮转座舱&#xff0c;但每次轮转都需要支付一定的运行成本 runningCost 。摩…

Docker简介、基本概念和安装

Docker简介、基本概念和安装 1.docker简介 1.1 什么是docker Docker 最初是 dotCloud 公司创始人 Solomon Hykes (opens new window)在法国期间发起的一个公司内部项目&#xff0c;它是基于 dotCloud 公司多年云服务技术的一次革新&#xff0c;并于 2013 年 3 月以 Apache 2…

Objective-C中使用STL标准库Queue队列

1.修改.m文件为mm 2.导入queue头 #include<queue> 3.使用&#xff1a; #import <Foundation/Foundation.h> #include <cmath> #include <queue> using namespace std;int main(int argc, const char * argv[]) {autoreleasepool {NSLog("C标准…

云计算任务调度仿真01

云计算任务调度的研究大多数以来仿真研究&#xff0c;现梳理一些做过的代码研究 结果无数次的排错&#xff0c;终于finish with code 0 了 这个代码以来的是比较老的TensorFlow版本&#xff0c;我们都知道TensorFlow1.x和TensorFlow2.x之间有很大差别&#xff0c;但其实&#…

Realm Management Extension领域管理扩展之安全状态

RME基于Arm TrustZone技术。TrustZone技术在Armv6中引入,提供以下两个安全状态: 安全状态(Secure state)非安全状态(Non-secure state)以下图表显示了在AArch64中的这两个安全状态以及通常在每个安全状态中找到的软件组件: 该架构将在安全状态运行的软件与在非安全状态运…

openGauss学习笔记-190 openGauss 数据库运维-常见故障定位案例-服务启动失败

文章目录 openGauss学习笔记-190 openGauss 数据库运维-常见故障定位案例-服务启动失败190.1 服务启动失败190.1.1 问题现象190.1.2 原因分析190.1.3 处理办法 openGauss学习笔记-190 openGauss 数据库运维-常见故障定位案例-服务启动失败 190.1 服务启动失败 190.1.1 问题现…

Copilot 插件的使用介绍:如何快速上手

GitHub Copilot 本文主要介绍如何通过脚本工具激活 GitHub Copilot 插件&#xff0c;提供安装及激活图文教程&#xff0c;大家按下面操作即可激活GitHub Copilot插件&#xff0c;免费使用Ai编码工具 一、GitHub Copilot 介绍 GitHub Copilot 是由 GitHub 和 OpenAI 共同开发的…

告别冗余空白,批量删除空白行

你是否遇到过这样的尴尬情况&#xff1a;花费了大量时间整理的文档&#xff0c;却在最后发现其中充斥着无用的空白行&#xff0c;这些多余的空行不仅影响美观&#xff0c;还让整个文档显得杂乱无章。今天&#xff0c;我要给大家介绍一款强大且实用的工具——首助编辑高手&#…

移动端对大批量图片加载的优化方法(三)

移动端对大批量图片加载的优化方法&#xff08;三&#xff09;Flutter 本篇主要从Flutter开发中可以使用到的对大批量图片加载的优化方法进行整理。 1.合适的图片格式 详情请参考移动端对大批量图片加载的优化方法&#xff08;一&#xff09;。 2.缓存机制 在Flutter中&am…

浅谈对Promise的理解。

一、Promise定义 JS中用于处理异步操作的编程模式。一个Promise是一个代理&#xff0c;它代表一个创建Promise时不一定已知的值。它允许我们将处理的程序与异步操作的最终成功值或失败值原因想关联起来。这使得异步方法可以像同步方法一样返回值&#xff1a;异步方法不会立即返…

uniapp最简单的底部兼容安全区域显示

效果图&#xff1a; 1.html写上动态padding-bottom <view class"button-wrap" :style"padding-bottom:bottomPaddingrpx"><view class"com-btn cencel-btn">取消</view><view class"com-btn confirm-btn " cl…

【PostgreSQL创建索引的锁分析和使用注意】

1.1 创建普通B-tree索引的整体流程 如下是梳理的创建普通B-tree索引的大概流程&#xff0c;可供参考。 1.校验新索引的Catalog元数据|语法解析 ---将创建索引的sql解析成IndexStmt结构&#xff5c;校验B-Tree的handler -----校验内核是否支持该类型的索引,在pg_am中查找&q…

java 体育明星管理系统Myeclipse开发mysql数据库web结构java编程计算机网页项目

一、源码特点 java Web 体育明星管理系统是一套完善的java web信息管理系统&#xff0c;对理解JSP java编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用B/S模式开发。开发环境为 TOMCAT7.0,Myeclipse8.5开发&#xff0c;数据库为Mysq…

Clojure 实战(4):编写 Hadoop MapReduce 脚本

Hadoop简介 众所周知&#xff0c;我们已经进入了大数据时代&#xff0c;每天都有PB级的数据需要处理、分析&#xff0c;从中提取出有用的信息。Hadoop就是这一时代背景下的产物。它是Apache基金会下的开源项目&#xff0c;受Google两篇论文的启发&#xff0c;采用分布式的文件…

Spark Core--加强

RDD的持久化 RDD缓存 当RDD被重复使用&#xff0c;或者计算该RDD比较容易出错&#xff0c;而且需要消耗比较多的资源和时间的时候&#xff0c;我们就可以将该RDD缓存起来。 主要作用: 提升Spark程序的计算效率 注意事项: RDD的缓存可以存储在内存或者是磁盘上&#xff0c;甚至…

【数据结构】二叉树链式结构详解

目录 1.前言2.快速创建一颗二叉树3.二叉树的遍历3.1前序遍历3.2中序遍历3.3后序遍历3.4层序遍历 4.二叉树节点个数与高度4.1二叉树节点个数4.2二叉树叶子节点个数4.3二叉树高度4.4二叉树第k层节点个数4.5二叉树查找值为x的节点 5.二叉树的基础oj题练习6.二叉树的创建和销毁6.1通…

JVM,JRE,JDK的区别和联系简洁版

先看图 利用JDK&#xff08;调用JAVA API&#xff09;开发JAVA程序后&#xff0c;通过JDK中的编译程序&#xff08;javac&#xff09;将我们的文本java文件编译成JAVA字节码&#xff0c;在JRE上运行这些JAVA字节码&#xff0c;JVM解析这些字节码&#xff0c;映射到CPU指令集或…