Java多态数组找最贵最便宜对象方法
本文详细介绍了在Java中处理多态数组,并从中查找特定子类型(如UAV无人机)中最昂贵和最便宜对象的方法。针对开发者在处理此类问题时容易混淆对象价格与数组索引的常见错误,文章提出了清晰的解决方案:**价格与索引分离管理**。通过独立维护价格变量和索引变量,确保比较逻辑的正确性,避免返回错误结果。文中提供了完整的代码示例,展示了如何使用`instanceof`关键字进行类型检查和向下转型,以及如何处理空数组和无目标对象的情况。同时,强调了初始值选择、性能优化等注意事项。掌握这些技巧,能有效提升开发者在Java多态数组中查找极值的能力,编写出高效、准确、鲁棒的代码。

引言:多态数组中的对象筛选与极值查找
在Java等面向对象编程语言中,我们经常会遇到包含不同但相关联对象类型的数组或集合。例如,一个FlyingObjects数组可能包含Airplane、Helicopter以及各种UAV(无人机)的实例。在这种多态场景下,我们可能需要找出特定子类型(如所有UAV中)价格最高或最低的对象。
然而,在实现这类功能时,一个常见的陷阱是混淆对象属性(如价格)与对象在数组中的位置(索引)。原始问题中出现的错误正是将代表价格的变量与代表索引的变量混为一谈,导致比较逻辑混乱,最终返回了不正确的结果。要解决这个问题,关键在于明确区分并独立管理价格和索引信息。
核心概念:价格与索引分离管理
为了准确找到最昂贵和最便宜的特定类型对象,我们需要引入独立的变量来存储以下信息:
- 最小/最大价格(double类型):用于记录当前遍历到的最小或最大价格值。
- 最小/最大价格对应的索引(int类型):用于记录拥有最小或最大价格的对象在数组中的位置。
这种分离管理确保了比较操作始终发生在价格值之间,而索引变量则仅用于记录对应对象的位置。
初始化策略:
- 价格变量:
- minPrice应初始化为一个足够大的值,确保任何实际价格都能比它小,例如Double.MAX_VALUE。
- maxPrice应初始化为一个足够小的值,确保任何实际价格都能比它大,例如0.0(如果价格总是非负)。
- 索引变量:
- minIndex和maxIndex可以初始化为-1,表示尚未找到符合条件的对象。
- 为了更健壮地处理数组中可能没有目标类型对象的情况,或确保第一次找到目标对象时能正确初始化,我们通常会在循环中找到第一个符合条件的对象时,用其价格和索引来初始化这些变量。
实现步骤与代码示例
以下是查找多态数组中特定子类型(UAV)最昂贵和最便宜对象的具体步骤和修正后的Java代码示例:
- 定义变量: 声明用于存储最小/最大价格和对应索引的变量,以及一个布尔标志来跟踪是否已找到至少一个目标类型的对象。
- 空数组检查: 在遍历之前,首先检查传入的数组是否为空或null,避免NullPointerException。
- 遍历数组: 使用for循环遍历整个FlyingObjects数组。
- 类型检查与向下转型: 在循环内部,使用instanceof操作符检查当前对象是否为UAV类型或其子类。如果是,则将其向下转型为UAV类型,以便访问其getPrice()方法。
- 初始化与比较:
- 如果这是找到的第一个UAV对象,将其价格和索引作为minPrice、maxPrice、minIndex和maxIndex的初始值。
- 对于后续找到的UAV对象,将其价格与当前的minPrice和maxPrice进行比较。如果当前UAV的价格更低,则更新minPrice和minIndex;如果更高,则更新maxPrice和maxIndex。
- 结果输出: 循环结束后,根据foundAnyUav标志判断是否找到了UAV对象。如果找到了,则使用记录的索引从原数组中获取并打印最昂贵和最便宜的UAV信息。
修正后的Java代码示例:
public class FlyingObjectsProcessor {
// 假设 FlyingObjects, Uav 类及其继承关系已定义,
// 且 Uav 类有 getPrice() 方法返回 double 类型价格。
// 例如:
/*
public class FlyingObjects {
protected double price;
public FlyingObjects(double price) { this.price = price; }
public double getPrice() { return price; }
public void setPrice(double price) { this.price = price; }
@Override
public String toString() { return "FlyingObject [Price=" + price + "]"; }
}
public class Uav extends FlyingObjects {
// Uav特有属性和构造器
public Uav(double weight, double price) {
super(price); // 假设UAV构造器第二个参数是价格
// ... 其他初始化
}
@Override
public String toString() { return "UAV [Price=" + price + "]"; }
}
public class AgriculturalDrone extends Uav {
public AgriculturalDrone(double payload, double price, String model, double range) {
super(payload, price);
// ...
}
@Override
public String toString() { return "AgriculturalDrone [Price=" + price + "]"; }
}
// ... 其他类类似
*/
/**
* 在飞行物数组中查找最昂贵和最便宜的UAV对象。
*
* @param flyingObjects 包含各种飞行物对象的数组。
*/
public static void findLeastAndMostExpensiveUAV(FlyingObjects[] flyingObjects) {
// 用于记录最昂贵UAV的价格和索引
double maxPrice = 0.0;
int maxPricedUavIndex = -1;
// 用于记录最便宜UAV的价格和索引
double minPrice = Double.MAX_VALUE;
int minPricedUavIndex = -1;
// 标记是否至少找到一个UAV
boolean foundAnyUav = false;
// 1. 处理空数组或null数组的情况
if (flyingObjects == null || flyingObjects.length == 0) {
System.out.println("传入的飞行物数组为空,无法查找UAV。");
return;
}
// 2. 遍历数组查找UAV对象
for (int i = 0; i < flyingObjects.length; i++) {
// 3. 检查当前对象是否为UAV或其子类
if (flyingObjects[i] instanceof Uav) {
Uav currentUav = (Uav) flyingObjects[i]; // 向下转型
double currentPrice = currentUav.getPrice();
// 4. 如果是找到的第一个UAV,则初始化极值
if (!foundAnyUav) {
minPrice = currentPrice;
maxPrice = currentPrice;
minPricedUavIndex = i;
maxPricedUavIndex = i;
foundAnyUav = true;
} else {
// 5. 比较并更新最昂贵UAV
if (currentPrice > maxPrice) {
maxPrice = currentPrice;
maxPricedUavIndex = i;
}
// 6. 比较并更新最便宜UAV
if (currentPrice < minPrice) {
minPrice = currentPrice;
minPricedUavIndex = i;
}
}
}
}
// 7. 输出结果
if (!foundAnyUav) {
System.out.println("在数组中没有找到任何UAV对象。");
} else {
System.out.println("\n关于最昂贵UAV的信息: \n" + flyingObjects[maxPricedUavIndex] + "\n");
System.out.println("关于最便宜UAV的信息: \n" + flyingObjects[minPricedUavIndex]);
}
}
// 主方法用于测试
public static void main(String[] args) {
// 示例数据,请确保 FlyingObjects, Uav, AgriculturalDrone, Mav, Multirotor, Helicopter, Airplane 类已正确定义
// 并包含 getPrice() 方法以及合适的构造器
FlyingObjects[] test = new FlyingObjects[7];
test[0] = new Uav(10, 43); // 价格 43
test[1] = new AgriculturalDrone(8000, 780000, "Chase", 2400); // 价格 780000
test[2] = new Uav(10, 5); // 价格 5
test[3] = new Mav(0.5, 140000, "trooper", 10); // 价格 140000
test[4] = new Multirotor("Hexa", 140000, 200, 185, 2021, 1, 4); // 价格 140000
test[5] = new Helicopter("Robinson", 199000, 250, 100, 2018, 7); // 价格 199000
test[6] = new Airplane("Boeing", 350000, 450); // 价格 350000
findLeastAndMostExpensiveUAV(test);
// 测试不包含UAV的数组
System.out.println("\n--- 测试不包含UAV的数组 ---");
FlyingObjects[] noUavTest = new FlyingObjects[2];
noUavTest[0] = new Airplane("Small", 10000, 100);
noUavTest[1] = new Helicopter("Large", 50000, 200);
findLeastAndMostExpensiveUAV(noUavTest);
// 测试空数组
System.out.println("\n--- 测试空数组 ---");
FlyingObjects[] emptyTest = new FlyingObjects[0];
findLeastAndMostExpensiveUAV(emptyTest);
// 测试null数组
System.out.println("\n--- 测试null数组 ---");
findLeastAndMostExpensiveUAV(null);
}
}注意事项
- 初始值选择: minPrice初始化为Double.MAX_VALUE和maxPrice初始化为0.0(或第一个有效UAV的价格)是处理非负价格的常用且安全的方式。如果价格可能为负,则maxPrice应初始化为Double.MIN_VALUE,或采用foundAnyUav标志的通用方法。
- instanceof关键字: instanceof用于运行时类型检查,判断对象是否是某个类或其子类的实例。它在处理多态集合时非常有用。
- 向下转型(Downcasting): 在确认对象是Uav类型后,需要进行强制类型转换(Uav currentUav = (Uav) flyingObjects[i];)。如果instanceof检查不通过就强转,会抛出ClassCastException。
- 数组为空或无目标对象: 务必处理传入数组为空(null或长度为0)以及数组中不包含任何目标类型对象(本例中为UAV)的情况,避免运行时错误并提供友好的提示信息。
- 性能考虑: 对于非常大的数组,单次遍历是最高效的,避免不必要的多次遍历。上述代码示例采用单次遍历,并通过foundAnyUav标志灵活处理了初始化逻辑。
- toString()方法: 为了方便打印对象信息,确保FlyingObjects及其子类重写了toString()方法,以便System.out.println()能够输出有意义的描述。
总结
在Java中处理多态数组并查找特定类型对象的极值(如最昂贵或最便宜)是一个常见的任务。解决此问题的关键在于:
- 明确区分并独立管理数据值(价格)和其在集合中的位置(索引)。
- 使用instanceof进行运行时类型检查,确保操作的安全性。
- 采用健壮的初始化策略和循环逻辑,处理各种边界情况,如空数组或不含目标对象的数组。
通过遵循这些原则,您可以编写出高效、准确且鲁棒的代码,有效应对复杂的对象集合操作需求。
文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《Java多态数组找最贵最便宜对象方法》文章吧,也可关注golang学习网公众号了解相关技术文章。
requestAnimationFrame使用技巧与实战解析
- 上一篇
- requestAnimationFrame使用技巧与实战解析
- 下一篇
- Golang加速第三方包下载方法
-
- 文章 · java教程 | 20分钟前 |
- Java遍历Map的四种方式
- 100浏览 收藏
-
- 文章 · java教程 | 34分钟前 |
- Java中this关键字的使用场景详解
- 391浏览 收藏
-
- 文章 · java教程 | 35分钟前 | caffeine 并发 synchronized concurrenthashmap 线程安全缓存
- Java线程安全缓存实现技巧
- 490浏览 收藏
-
- 文章 · java教程 | 59分钟前 |
- Javanotify与notifyAll区别详解
- 450浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- SpringBootOpenAPI枚举校验方法与错误处理
- 296浏览 收藏
-
- 文章 · java教程 | 1小时前 | sql注入 输入校验 参数化查询 PreparedStatement SQL拼接
- Java后端如何防范SQL注入?
- 466浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- JavaCalendar类实用技巧全解析
- 409浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java注解简化异常处理方法
- 495浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java文件上传自定义命名技巧
- 248浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java如何创建PrintStream对象
- 327浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3193次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3406次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3436次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4544次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3814次使用
-
- 提升Java功能开发效率的有力工具:微服务架构
- 2023-10-06 501浏览
-
- 掌握Java海康SDK二次开发的必备技巧
- 2023-10-01 501浏览
-
- 如何使用java实现桶排序算法
- 2023-10-03 501浏览
-
- Java开发实战经验:如何优化开发逻辑
- 2023-10-31 501浏览
-
- 如何使用Java中的Math.max()方法比较两个数的大小?
- 2023-11-18 501浏览

