import numpy as np


# 竞标赛选择法，每次随机选择两个个体，优先选择排序等级高的个体，如果排序等级一样，则优选选择拥挤度大的个体
# pop:交配池大小 tour_size:参赛选手个数
def tournament_selection(chromosome, pool_size, tour_size):
    pop, variables = chromosome.shape  # 获取种群的个体数量和决策变量数量
    rank = variables - 2  # 个体向量中排序值所在位置
    distance = variables - 1  # 个体向量中拥挤度所在位置
    res = np.zeros((pool_size, variables))  # 存储选中的个体

    for i in range(pool_size):
        candidate = np.zeros(tour_size, dtype=int)
        for j in range(tour_size):
            candidate[j] = np.random.randint(pop)  # 随机选择参赛个体
            while np.any(candidate[:j] == candidate[j]):  # 防止两个参赛个体是同一个
                candidate[j] = np.random.randint(pop)

        c_obj_rank = chromosome[candidate, rank]  # 记录每个参赛者的排序等级
        c_obj_distance = chromosome[candidate, distance]  # 记录每个参赛者的拥挤度

        min_candidate = np.where(c_obj_rank == np.min(c_obj_rank))[
            0
        ]  # 选择排序等级较小的参赛者
        if (
            len(min_candidate) != 1
        ):  # 如果两个参赛者的排序等级相等，则继续比较拥挤度，优先选择拥挤度大的个体
            max_candidate = np.where(
                c_obj_distance[min_candidate] == np.max(c_obj_distance[min_candidate])
            )[0]
            max_candidate = max_candidate[0]
            res[i, :] = chromosome[candidate[min_candidate[max_candidate]], :]
        else:
            res[i, :] = chromosome[candidate[min_candidate[0]], :]

    return res


# 精英选择策略
import numpy as np


def replace_chromosome(intermediate_chromosome, M, V, pop):
    """
    精英保留策略 - 修复版本
    从合并种群中选择前pop个最优个体组成新种群

    参数:
    intermediate_chromosome: 合并后的种群（父代+子代）
    M: 目标函数数量
    V: 决策变量个数
    pop: 种群大小

    返回:
    selected_chromosome: 选择出的新种群
    """
    N, m = intermediate_chromosome.shape

    # 按非支配排序等级升序排序
    index = np.argsort(intermediate_chromosome[:, M + V])
    sorted_chromosome = intermediate_chromosome[index]

    max_rank = np.max(intermediate_chromosome[:, M + V])
    selected_chromosome = np.zeros((pop, M + V + 2))

    current_count = 0  # 当前已选择的个体数量
    previous_index = 0

    # 按等级从低到高（从优到劣）选择个体
    for i in range(1, int(max_rank) + 1):
        # 找到当前等级的所有个体
        current_rank_indices = np.where(sorted_chromosome[:, M + V] == i)[0]

        if len(current_rank_indices) == 0:
            continue

        current_index = np.max(current_rank_indices)  # 当前等级在排序后种群中的最大索引

        # 当前等级个体数量
        current_rank_count = (
            current_index - previous_index + 1
            if previous_index == 0
            else current_index - previous_index
        )

        # 情况1: 加入当前等级所有个体后，总数不超过pop
        if current_count + current_rank_count <= pop:
            selected_chromosome[
                current_count : current_count + current_rank_count, :
            ] = sorted_chromosome[previous_index : current_index + 1, :]
            current_count += current_rank_count

            # 如果刚好填满种群，直接返回
            if current_count == pop:
                return selected_chromosome

        # 情况2: 加入当前等级所有个体后，总数会超过pop
        else:
            remaining = pop - current_count
            if remaining > 0:
                # 从当前等级中按拥挤度选择剩余个体
                temp_pop = sorted_chromosome[previous_index : current_index + 1, :]
                # 按拥挤度降序排序
                temp_sort_index = np.argsort(temp_pop[:, M + V + 1])[::-1]
                temp_sort_pop = temp_pop[temp_sort_index]

                selected_chromosome[current_count : current_count + remaining, :] = (
                    temp_sort_pop[0:remaining, :]
                )

            return selected_chromosome

        previous_index = current_index + 1

    # 如果所有等级处理完仍未填满种群（理论上不应发生），返回已选择的部分
    if current_count < pop:
        print(f"Warning: Only selected {current_count} individuals, expected {pop}")
        return selected_chromosome[0:current_count, :]

    return selected_chromosome
