
本文详解因数组元素赋值方向错误(如将 `nama[i] = lulus1[i]` 误写为 `lulus1[i] = nama[i]` 的反向操作)导致输出全为 `null` 的典型 bug,并提供完整修正方案、逻辑梳理与健壮性优化建议。
在您提供的 Java 程序中,核心问题出在数组赋值逻辑颠倒,直接导致 lulus1、tlulus1、lulus2、tlulus2 四个结果数组始终输出 null。根本原因在于:您本意是“将合格学生姓名存入 lulus1”,但代码却错误地执行了 nama[i] = lulus1[i] —— 此时 lulus1[i] 初始值为 null,于是把 nama[i] 覆盖成了 null,不仅丢失原始姓名,还使后续 Arrays.toString() 显示 [null, null, ...]。
✅ 正确赋值逻辑(关键修复)
应严格遵循“目标数组 ← 源数据”原则:
// ✅ 正确:满足 KKM 的学生姓名存入 lulus1
if (nilai1[i] >= kkm1) {
lulus1[k] = nama[i]; // ← 注意:使用独立索引 k 避免空位
k++; // 记录实际存入数量
} else {
tlulus1[j] = nama[i]; // 同理,用独立索引 j 处理不通过者
j++;
}但原代码还存在更深层问题:未区分有效索引与数组容量。直接使用 i 作为 lulus1[i] 下标,会导致:
- 当某学生不满足 KKM 时,lulus1[i] 保持 null;
- 最终 Arrays.toString(lulus1) 输出含大量 null 的稀疏数组(如 [null, "Andi", null, "Budi"]),而非紧凑结果。
? 完整修正版(含健壮性增强)
import java.util.Arrays;
import javax.swing.JOptionPane;
public class Testing {
public static void main(String[] args) {
int batas = Integer.parseInt(JOptionPane.showInputDialog("Batas Siswa"));
String[] nama = new String[batas];
int[] nilai1 = new int[batas];
int[] nilai2 = new int[batas];
String mapel1 = JOptionPane.showInputDialog("Masukan Mapel 1");
String mapel2 = JOptionPane.showInputDialog("Masukan Mapel 2");
int kkm1 = Integer.parseInt(JOptionPane.showInputDialog("Input KKM 1"));
int kkm2 = Integer.parseInt(JOptionPane.showInputDialog("Input KKM 2"));
// 动态收集结果(避免 null 占位)
java.util.List lulus1List = new java.util.ArrayList<>();
java.util.List tlulus1List = new java.util.ArrayList<>();
java.util.List lulus2List = new java.util.ArrayList<>();
java.util.List tlulus2List = new java.util.ArrayList<>();
int total1 = 0, total2 = 0;
for (int i = 0; i < batas; i++) {
nama[i] = JOptionPane.showInputDialog("Nama Siswa ke-" + (i+1));
nilai1[i] = Integer.parseInt(JOptionPane.showInputDialog("Input Nilai " + mapel1));
total1 += nilai1[i];
if (nilai1[i] >= kkm1) {
lulus1List.add(nama[i]);
} else {
tlulus1List.add(nama[i]);
}
nilai2[i] = Integer.parseInt(JOptionPane.showInputDialog("Input Nilai " + mapel2));
total2 += nilai2[i];
if (nilai2[i] >= kkm2) {
lulus2List.add(nama[i]);
} else {
tlulus2List.add(nama[i]);
}
}
// 转为数组(紧凑无 null)
String[] lulus1 = lulus1List.toArray(new String[0]);
String[] tlulus1 = tlulus1List.toArray(new String[0]);
String[] lulus2 = lulus2List.toArray(new String[0]);
String[] tlulus2 = tlulus2List.toArray(new String[0]);
// 统计计算(修复 min/max 循环范围)
int max1 = Arrays.stream(nilai1).max().orElse(0);
int min1 = Arrays.stream(nilai1).min().orElse(0);
int max2 = Arrays.stream(nilai2).max().orElse(0);
int min2 = Arrays.stream(nilai2).min().orElse(0);
int Ttotal = total1 + total2;
int rata = batas > 0 ? Ttotal / (2 * batas) : 0; // 修正平均值逻辑:每科人均分
// 输出(清晰、无 null)
System.out.println("=== HASIL ANALISIS ===");
System.out.println("Nilai Rata-rata Keseluruhan: " + rata);
System.out.println("Nilai Tertinggi " + mapel1 + ": " + max1);
System.out.println("Nilai Terendah " + mapel1 + ": " + min1);
System.out.println("Nilai Tertinggi " + mapel2 + ": " + max2);
System.out.println("Nilai Terendah " + mapel2 + ": " + min2);
System.out.println("Siswa Lulus " + mapel1 + ": " + Arrays.toString(lulus1));
System.out.println("Siswa Tidak Lulus " + mapel1 + ": " + Arrays.toString(tlulus1));
System.out.println("Siswa Lulus " + mapel2 + ": " + Arrays.toString(lulus2));
System.out.println("Siswa Tidak Lulus " + mapel2 + ": " + Arrays.toString(tlulus2));
}
} ⚠ 关键注意事项
- 赋值方向勿颠倒:永远是 目标数组[索引] = 源值,而非反向。
- 避免固定索引填充:用 ArrayList 动态收集再转数组,杜绝 null 占位和 ArrayIndexOutOfBoundsException。
- 循环边界校验:原代码 for (int i = 0; i stream().max()/min() 或完整遍历 nilai1.length。
- 平均值逻辑修正:rata 应为 (total1 + total2) / (2 * batas)(两科总分 ÷ 总人数 × 2),而非简单除以 2。
- 空输入防护:生产环境需增加 try-catch 处理 NumberFormatException 和 NullPointerException。
通过以上修正,程序将准确输出按 KKM 分类的学生名单,且结果数组紧凑、可读性强,彻底解决 null 显示问题。
立即学习“Java免费学习笔记(深入)”;










