import numpy as np
import random
import torch


def setup_seed(seed):
    torch.manual_seed(seed)
    torch.cuda.manual_seed_all(seed)
    np.random.seed(seed)
    random.seed(seed)
    torch.backends.cudnn.deterministic = True


def generate_cvrp_instance(n_customers, seed=42):
    """智能生成CVRP实例"""
    random.seed(seed)
    np.random.seed(seed)

    n_nodes = n_customers + 1  # +1 for depot

    # 生成节点坐标 (depot在中心附近，客户分散分布)
    coordinates = np.zeros((n_nodes, 2))

    # Depot位置 (在区域中心附近)
    coordinates[0] = [50, 50]  # depot at center

    # 客户位置 (在100x100的区域内随机分布)
    for i in range(1, n_nodes):
        coordinates[i] = [
            np.random.uniform(10, 90),  # x coordinate
            np.random.uniform(10, 90)  # y coordinate
        ]

    # 计算欧几里得距离矩阵
    distance_matrix = np.zeros((n_nodes, n_nodes))
    for i in range(n_nodes):
        for j in range(n_nodes):
            if i != j:
                dist = np.sqrt((coordinates[i][0] - coordinates[j][0]) ** 2 +
                               (coordinates[i][1] - coordinates[j][1]) ** 2)
                distance_matrix[i][j] = round(dist, 1)

    # 生成客户需求 (合理的需求分布)
    customer_demands = {}
    for i in range(1, n_nodes):
        # 需求在5-25之间，符合正态分布
        demand = max(5, min(25, int(np.random.normal(12, 5))))
        customer_demands[i] = demand

    return distance_matrix, customer_demands, coordinates


def moving_average(data, window_size):
    """
    计算移动平均值
    
    Args:
        data: 输入数据数组
        window_size: 窗口大小
    
    Returns:
        移动平均后的数组
    """
    if len(data) < window_size:
        return data
    
    # 转换为numpy数组
    data = np.array(data)
    
    # 使用卷积计算移动平均
    kernel = np.ones(window_size) / window_size
    
    # 'valid'模式：只返回完全重叠的部分
    smoothed = np.convolve(data, kernel, mode='valid')
    
    return smoothed


# def moving_average(a, window_size):
#     cumulative_sum = np.cumsum(np.insert(a, 0, 0))
#     middle = (cumulative_sum[window_size:] - cumulative_sum[:-window_size]) / window_size
#     r = np.arange(1, window_size-1, 2)
#     begin = np.cumsum(a[:window_size-1])[::2] / r
#     end = (np.cumsum(a[:-window_size:-1])[::2] / r)[::-1]
#     return np.concatenate((begin, middle, end))