如何修复 Java 中数组赋值错误导致的 null 输出问题

本文详解因数组元素赋值方向错误(如将 `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 显示问题。