def calc_cor_inc_ref_counts(grades: list[str], choices: str) -> tuple[int, int, int]:
    if len(grades) != len(choices):
        raise ValueError("corrects and choices must have the same length")
    n_cor, n_inc, n_ref = 0, 0, 0
    for grade, choice in zip(grades, choices):
        if (choice == "N") or (grade == "C"):
            n_ref += 1
        elif grade == "A":
            n_cor += 1
        else:
            n_inc += 1
    return n_cor, n_inc, n_ref

def calc_cor_inc_ref_programmatically(
    grades: list[str],
    confs: list[float],
    r_cor: float,
    r_inc: float,
    r_ref: float,
) -> tuple[int, int, int]:
    if len(grades) != len(confs):
        raise ValueError("grades and confs must have the same length")
    n_cor, n_inc, n_ref = 0, 0, 0
    for grade, conf in zip(grades, confs):
        if (conf * r_cor + (1 - conf) * r_inc < r_ref) or (grade == "C"):
            n_ref += 1
        elif grade == "A":
            n_cor += 1
        else:
            n_inc += 1
    return n_cor, n_inc, n_ref

def calc_normalized_reward(n_cor: int, n_inc: int, n_ref: int, r_cor: float, r_inc: float, r_ref: float) -> float:
    N = n_cor + n_inc + n_ref
    raw_reward = n_cor * r_cor + n_inc * r_inc + n_ref * r_ref
    max_reward = N * r_cor
    return raw_reward / max_reward
