
在面向对象设计中,uml(统一建模语言)类图是描述类结构的重要工具。然而,uml中构造器的表示方式与特定编程语言(如java)的实现约定可能存在差异。
根据UML规范,一个构造器通常被表示为一个带有«Create»构造型(stereotype)的操作,并且其返回类型是类本身。例如,一个名为Student的类的构造器可能表示为:
+ «Create» Student(name:String):Student
但在Java等主流面向对象语言中,构造器遵循一套约定俗成的规则:它的名称必须与类名完全一致,并且没有显式的返回类型(尽管它隐式地返回一个新创建的对象实例)。因此,在许多情况下,UUML图中的Student(name:String)会被自然地理解为Java中的构造器。
根据上述理解,如果我们有一个UML类图指示Student类有一个接受name参数的构造器,那么在Java中,其基本实现骨架如下:
public class Student {
private String name;
private int[] homeworkScores; // 存储作业分数
private int[] examScores; // 存储考试分数
// ... 其他字段和方法
public Student(String name) {
this.name = name;
// 构造器中初始化数组
this.homeworkScores = new int[6]; // 6个作业分数
this.examScores = new int[3]; // 3个考试分数
}
// ... 其他方法
}在上述构造器中,我们不仅初始化了name字段,还根据需求创建了固定大小的空数组来存储作业分数和考试分数。这是在对象创建时建立其初始有效状态的关键一步。
立即学习“Java免费学习笔记(深入)”;
除了构造器中的初始化,一个完整的Student类还需要包含处理这些分数的方法,例如计算平均分和最终成绩。
在Java中,初始化固定大小的数组非常直接,使用new关键字后跟数组类型和大小即可。例如,new int[6]会创建一个包含6个整数元素的数组,所有元素默认初始化为0。
根据需求,我们需要计算作业平均分和最终总分。
import java.util.Arrays; // 用于数组操作,如复制
public class Student {
private String name;
private int[] homeworkScores; // 6个作业分数
private int[] examScores; // 3个考试分数
public Student(String name) {
this.name = name;
this.homeworkScores = new int[6];
this.examScores = new int[3];
}
// Getter for name
public String getName() {
return name;
}
// Setter for name
public void setName(String name) {
this.name = name;
}
/**
* 计算学生作业的平均分数。
* @return 作业平均分数,如果无作业则返回0。
*/
public double getHomeworkAverage() {
if (homeworkScores.length == 0) {
return 0.0;
}
int sum = 0;
for (int score : homeworkScores) {
sum += score;
}
return (double) sum / homeworkScores.length;
}
/**
* 计算学生的最终总分。
* 评分标准:Exam1占15%,Exam2占25%,Exam3占30%,作业平均分占30%。
* @return 最终总分。
*/
public int getFinalScore() {
// 确保考试分数数组有足够的元素
if (examScores.length < 3) {
throw new IllegalStateException("考试分数不足,无法计算最终成绩。");
}
double exam1Weight = 0.15;
double exam2Weight = 0.25;
double exam3Weight = 0.30;
double homeworkWeight = 0.30;
double finalScore = (examScores[0] * exam1Weight) +
(examScores[1] * exam2Weight) +
(examScores[2] * exam3Weight) +
(getHomeworkAverage() * homeworkWeight);
return (int) Math.round(finalScore); // 四舍五入到整数
}
// ... 其他getter和setter (见下一节的改进版本)
}在Java中处理包含数组字段的类时,必须特别注意封装性,以防止外部代码通过直接引用修改对象内部状态。
如果getter方法直接返回内部数组的引用,例如:
public int[] getHomeworkScores() {
return homeworkScores; // 返回了内部数组的引用
}那么外部代码就可以获取到这个引用,并直接修改数组的内容,绕过了类的控制逻辑,可能导致对象状态的不一致或破坏。
Student student = new Student("Alice");
int[] scores = student.getHomeworkScores();
scores[0] = 100; // 直接修改了Student对象内部的homeworkScores数组为了避免上述问题,getter方法应该返回内部数组的一个副本,而不是原始引用。
public int[] getHomeworkScores() {
return Arrays.copyOf(homeworkScores, homeworkScores.length); // 返回数组的副本
// 或者 return homeworkScores.clone();
}
public int[] getExamScores() {
return Arrays.copyOf(examScores, examScores.length);
}同样,setter方法在接收外部数组作为参数时,也应该复制其内容,而不是直接将内部字段指向外部数组的引用。这可以防止外部数组在之后被修改时,意外地影响到Student对象的状态。
public void setHomeworkScores(int[] newHomeworkScores) {
if (newHomeworkScores == null || newHomeworkScores.length != this.homeworkScores.length) {
throw new IllegalArgumentException("作业分数数组必须包含 " + this.homeworkScores.length + " 个元素。");
}
System.arraycopy(newHomeworkScores, 0, this.homeworkScores, 0, newHomeworkScores.length);
// 或者 this.homeworkScores = Arrays.copyOf(newHomeworkScores, newHomeworkScores.length);
}
public void setExamScores(int[] newExamScores) {
if (newExamScores == null || newExamScores.length != this.examScores.length) {
throw new IllegalArgumentException("考试分数数组必须包含 " + this.examScores.length + " 个元素。");
}
System.arraycopy(newExamScores, 0, this.examScores, 0, newExamScores.length);
}综合以上讨论,一个健壮且符合封装原则的Student类实现如下:
import java.util.Arrays;
public class Student {
private String name;
private int[] homeworkScores; // 6个作业分数
private int[] examScores; // 3个考试分数
/**
* 构造一个新的学生对象。
* @param name 学生的姓名。
*/
public Student(String name) {
if (name == null || name.trim().isEmpty()) {
throw new IllegalArgumentException("学生姓名不能为空。");
}
this.name = name;
this.homeworkScores = new int[6]; // 初始化为6个0分的作业
this.examScores = new int[3]; // 初始化为3个0分的考试
}
// --- Getter 方法 ---
public String getName() {
return name;
}
/**
* 获取学生作业分数的副本,防止外部修改内部状态。
* @return 包含作业分数的整数数组副本。
*/
public int[] getHomeworkScores() {
return Arrays.copyOf(homeworkScores, homeworkScores.length);
}
/**
* 获取学生考试分数的副本,防止外部修改内部状态。
* @return 包含考试分数的整数数组副本。
*/
public int[] getExamScores() {
return Arrays.copyOf(examScores, examScores.length);
}
// --- Setter 方法 ---
public void setName(String name) {
if (name == null || name.trim().isEmpty()) {
throw new IllegalArgumentException("学生姓名不能为空。");
}
this.name = name;
}
/**
* 设置学生的所有作业分数。
* @param newHomeworkScores 包含6个作业分数的数组。
* @throws IllegalArgumentException 如果数组为空或长度不匹配。
*/
public void setHomeworkScores(int[] newHomeworkScores) {
if (newHomeworkScores == null || newHomeworkScores.length != this.homeworkScores.length) {
throw new IllegalArgumentException("作业分数数组必须包含 " + this.homeworkScores.length + " 个元素。");
}
// 复制内容而不是直接赋值引用
System.arraycopy(newHomeworkScores, 0, this.homeworkScores, 0, newHomeworkScores.length);
}
/**
* 设置学生的所有考试分数。
* @param newExamScores 包含3个考试分数的数组。
* @throws IllegalArgumentException 如果数组为空或长度不匹配。
*/
public void setExamScores(int[] newExamScores) {
if (newExamScores == null || newExamScores.length != this.examScores.length) {
throw new IllegalArgumentException("考试分数数组必须包含 " + this.examScores.length + " 个元素。");
}
// 复制内容而不是直接赋值引用
System.arraycopy(newExamScores, 0, this.examScores, 0, newExamScores.length);
}
/**
* 设置单个作业分数。
* @param index 作业索引 (0-5)。
* @param score 分数。
* @throws IndexOutOfBoundsException 如果索引超出范围。
*/
public void setHomeworkScore(int index, int score) {
if (index < 0 || index >= homeworkScores.length) {
throw new IndexOutOfBoundsException("作业索引超出范围: " + index);
}
this.homeworkScores[index] = score;
}
/**
* 设置单个考试分数。
* @param index 考试索引 (0-2)。
* @param score 分数。
* @throws IndexOutOfBoundsException 如果索引超出范围。
*/
public void setExamScore(int index, int score) {
if (index < 0 || index >= examScores.length) {
throw new IndexOutOfBoundsException("考试索引超出范围: " + index);
}
this.examScores[index] = score;
}
// --- 业务逻辑方法 ---
/**
* 计算学生作业的平均分数。
* @return 作业平均分数,如果无作业则返回0。
*/
public double getHomeworkAverage() {
if (homeworkScores.length == 0) {
return 0.0;
}
int sum = 0;
for (int score : homeworkScores) {
sum += score;
}
return (double) sum / homeworkScores.length;
}
/**
* 计算学生的最终总分。
* 评分标准:Exam1占15%,Exam2占25%,Exam3占30%,作业平均分占30%。
* @return 最终总分(四舍五入到整数)。
* @throws IllegalStateException 如果考试分数数组长度不符合预期。
*/
public int getFinalScore() {
if (examScores.length < 3) {
throw new IllegalStateException("考试分数不足,无法计算最终成绩。");
}
double exam1Weight = 0.15;
double exam2Weight = 0.25;
double exam3Weight = 0.30;
double homeworkWeight = 0.30;
double finalScore = (examScores[0] * exam1Weight) +
(examScores[1] * exam2Weight) +
(examScores[2] * exam3Weight) +
(getHomeworkAverage() * homeworkWeight);
return (int) Math.round(finalScore);
}
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", homeworkScores=" + Arrays.toString(homeworkScores) +
", examScores=" + Arrays.toString(examScores) +
", homeworkAverage=" + String.format("%.2f", getHomeworkAverage()) +
", finalScore=" + getFinalScore() +
'}';
}
}将UML类图转换为Java对象不仅仅是字段和方法的简单映射。它要求开发者深入理解UML构造器在Java中的惯用表达,并注重Java语言的特性,尤其是数组这类引用类型字段的封装。通过在构造器中合理初始化数组,并在getter和setter方法中采取防御性复制(如Arrays.copyOf()或System.arraycopy()),可以有效保护对象的内部状态,避免外部代码的意外修改,从而确保程序的健壮性和数据完整性。同时,为特定业务逻辑(如计算平均分和总分)实现清晰的方法,也使得类功能更加完善和易于使用。
以上就是从UML类图到Java对象:构造器设计与数组初始化最佳实践的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号