基于聚类与引力斥力优化的选址算法

news2025/4/3 3:47:54

在众多实际场景中,诸如消防设施选址、基站布局规划以及充电桩站点部署等,都面临着如何利用最少的资源,实现对所有目标对象全面覆盖的难题。为有效解决这类问题,本文提出一种全新的组合算法模型 —— 基于聚类与引力斥力优化的选址算法。

算法简介

1. K - means 聚类初始化
  • 目的:快速生成初始中心点,初步划分数据分布。

  • 方法:通过 K - means 算法将二维点聚类为 k 组,每组中心作为初始候选点。结合肘部法则或轮廓系数确定最优 k,并使用 K - means++ 优化初始中心选择。

2. 贪心算法补充覆盖
  • 目的:确保所有数据点被覆盖,补充初始聚类未覆盖的区域。

  • 方法:设置中心点覆盖半径 r,标记未被覆盖的点。每次从未覆盖点中选择能覆盖最多未覆盖点的点作为新中心点,直至所有点被覆盖。

3. 引力搜索算法(GSA)优化
  • 目的:动态调整中心点位置,剔除冗余点,实现最少中心点全覆盖。

  • 方法:

  • 引力与斥力:中心点受未覆盖点引力(向数据移动)和其他中心点斥力(避免密集)。

  • 合力驱动:根据引力和斥力的矢量和移动中心点,验证覆盖后更新位置。

  • 剪枝操作:定期尝试删除冗余中心点,确保覆盖性的同时最小化中心点数量。

4. 组合算法优势
  • 高效性:K - means 快速聚类,贪心算法快速覆盖,GSA 精细调优。

  • 鲁棒性:结合几何覆盖与物理启发式优化,适应复杂分布。

  • 可解释性:每一步逻辑清晰,便于根据场景调整参数或替换算法。

算法步骤与公式

第一步:K - means 聚类
1. 初始聚类中心

使用算法初始化聚类中心,选择距离已有中心最远的点作为新中心:

2. 聚类分配与中心更新

计算每个点到中心的欧氏距离并分配:

重新计算中心坐标:

其中Sj是第j个聚类的数据点集合。

第二步:贪心算法覆盖未被覆盖点
1. 覆盖条件

点x被中心cj覆盖当且仅当:

2. 选择最优新中心

每次选择覆盖最多未覆盖点的点:

其中I(·)是指示函数。

第三步:引力搜索算法(GSA)优化中心点
1. 质量定义

中心点ci的质量mi由覆盖点数决定:

2. 引力计算

数据点x对中心ci的引力:

其中是时变引力常数。

3. 斥力计算

其他中心cj对ci的斥力:

4. 合力与位置更新

总合力:

位置更新:

其中 λ是方向系数,step_size是步长。

5. 中心点剔除条件

若剔除中心ci后所有点仍被覆盖:

代码

python代码
import numpy as np
import matplotlib.pyplot as plt
from sklearn.cluster import KMeans

# 第一步:K - means 聚类
# 随机生成一些示例数据
points = np.random.rand(100, 2)
# 设置聚类中心数
k = 10
# 执行 K - means 聚类
kmeans = KMeans(n_clusters=k, random_state=42)
idx = kmeans.fit_predict(points)
centers = kmeans.cluster_centers_

# 第二步:贪心算法覆盖未被覆盖点
# 设置覆盖半径
r = 0.2
# 标记未被覆盖的点
covered = np.zeros(points.shape[0], dtype=bool)
for center in centers:
    distances = np.linalg.norm(points - center, axis=1)
    covered[distances <= r] = True
uncovered_points = points[~covered]

while len(uncovered_points) > 0:
    max_coverage = 0
    best_point = None
    for point in uncovered_points:
        distances = np.linalg.norm(points - point, axis=1)
        current_coverage = np.sum((distances <= r) & ~covered)
        if current_coverage > max_coverage:
            max_coverage = current_coverage
            best_point = point
    centers = np.vstack((centers, best_point))
    distances = np.linalg.norm(points - best_point, axis=1)
    covered[distances <= r] = True
    uncovered_points = points[~covered]

# 第三步:引力搜索算法(GSA)优化中心点
# 设置参数
T = 100  # 迭代次数
G0 = 1  # 引力常数
alpha = 2  # 质量衰减系数

# 初始化质量
masses = np.zeros(centers.shape[0])
for i in range(centers.shape[0]):
    distances = np.linalg.norm(points - centers[i], axis=1)
    masses[i] = np.sum(distances <= r)

for t in range(T):
    G = G0 * np.exp(-alpha * t / T)
    for i in range(centers.shape[0]):
        force = np.zeros(2)
        # 计算斥力
        for j in range(centers.shape[0]):
            if i != j:
                distance = np.linalg.norm(centers[i] - centers[j])
                if distance > 0:
                    force += G * masses[j] * (centers[i] - centers[j]) / distance
        # 计算引力
        for j in range(points.shape[0]):
            distance = np.linalg.norm(centers[i] - points[j])
            if 0 < distance <= 2 * r:
                force -= G * (centers[i] - points[j]) / distance
        # 移动中心点(先假设移动)
        new_center = centers[i] + force
        # 检查移动后的中心点是否能覆盖所有数据点
        temp_centers = centers.copy()
        temp_centers[i] = new_center
        temp_covered = np.zeros(points.shape[0], dtype=bool)
        for center in temp_centers:
            distances = np.linalg.norm(points - center, axis=1)
            temp_covered[distances <= r] = True
        if np.all(temp_covered):
            centers[i] = new_center
    # 尝试剔除冗余的中心点
    for i in range(centers.shape[0]):
        temp_centers = np.delete(centers, i, axis=0)
        temp_covered = np.zeros(points.shape[0], dtype=bool)
        for center in temp_centers:
            distances = np.linalg.norm(points - center, axis=1)
            temp_covered[distances <= r] = True
        if np.all(temp_covered):
            centers = temp_centers
            masses = np.delete(masses, i)
            break

# 输出最终的中心点
print('最终的中心点坐标:')
print(centers)

# 可视化结果
plt.figure()
plt.scatter(points[:, 0], points[:, 1], s=20, c='b', marker='s', edgecolors='b')
plt.scatter(centers[:, 0], centers[:, 1], s=20, c='r', marker='^', edgecolors='r')
# 为每个中心点绘制覆盖半径的透明圆
theta = np.linspace(0, 2 * np.pi, 100)
for center in centers:
    x = center[0] + r * np.cos(theta)
    y = center[1] + r * np.sin(theta)
    plt.fill(x, y, 'r', alpha=0.1, edgecolor='r')
plt.xlabel('X')
plt.ylabel('Y')
plt.show()
结果图

图片

matlab代码
clear
clc
close all

% 第一步:K - means 聚类
% 假设我们有一组二维点数据 points
% 随机生成一些示例数据
points = rand(100, 2); 
% 设置聚类中心数
k = 10; 
% 执行 K - means 聚类
[idx, centers] = kmeans(points, k);

% 第二步:贪心算法覆盖未被覆盖点
% 设置覆盖半径
r = 0.2; 
% 标记未被覆盖的点
covered = false(size(points, 1), 1);
for i = 1:size(centers, 1)
    distances = sqrt(sum((points - repmat(centers(i, :), size(points, 1), 1)).^2, 2));
    covered(distances <= r) = true;
end
uncovered_points = points(~covered, :);

while ~isempty(uncovered_points)
    max_coverage = 0;
    best_point = [];
    for i = 1:size(uncovered_points, 1)
        distances = sqrt(sum((points - repmat(uncovered_points(i, :), size(points, 1), 1)).^2, 2));
        current_coverage = sum(distances <= r & ~covered);
        if current_coverage > max_coverage
            max_coverage = current_coverage;
            best_point = uncovered_points(i, :);
        end
    end
    centers = [centers; best_point];
    distances = sqrt(sum((points - repmat(best_point, size(points, 1), 1)).^2, 2));
    covered(distances <= r) = true;
    uncovered_points = points(~covered, :);
end

% 第三步:引力搜索算法(GSA)优化中心点
% 设置参数
T = 100; % 迭代次数
G0 = 1; % 引力常数
alpha = 2; % 质量衰减系数

% 初始化质量
masses = zeros(size(centers, 1), 1);
for i = 1:size(centers, 1)
    distances = sqrt(sum((points - repmat(centers(i, :), size(points, 1), 1)).^2, 2));
    masses(i) = sum(distances <= r);
end

for t = 1:T
    G = G0 * exp(-alpha * t / T);
    for i = 1:size(centers, 1)
        force = zeros(1, 2);
        % 计算斥力
        for j = 1:size(centers, 1)
            if i ~= j
                distance = sqrt(sum((centers(i, :) - centers(j, :)).^2));
                if distance > 0
                    force = force + G * masses(j) * (centers(i, :) - centers(j, :)) / distance;
                end
            end
        end
        % 计算引力
        for j = 1:size(points, 1)
            distance = sqrt(sum((centers(i, :) - points(j, :)).^2));
            if distance <= 2*r && distance > 0
                force = force - G * (centers(i, :) - points(j, :)) / distance;
            end
        end
        % 移动中心点(先假设移动)
        new_center = centers(i, :) + force;
        % 检查移动后的中心点是否能覆盖所有数据点
        temp_centers = centers;
        temp_centers(i, :) = new_center;
        temp_covered = false(size(points, 1), 1);
        for j = 1:size(temp_centers, 1)
            distances = sqrt(sum((points - repmat(temp_centers(j, :), size(points, 1), 1)).^2, 2));
            temp_covered(distances <= r) = true;
        end
        if all(temp_covered)
            centers(i, :) = new_center;
        end
    end
    % 尝试剔除冗余的中心点
    for i = 1:size(centers, 1)
        temp_centers = centers;
        temp_centers(i, :) = [];
        temp_covered = false(size(points, 1), 1);
        for j = 1:size(temp_centers, 1)
            distances = sqrt(sum((points - repmat(temp_centers(j, :), size(points, 1), 1)).^2, 2));
            temp_covered(distances <= r) = true;
        end
        if all(temp_covered)
            centers(i, :) = [];
            masses(i) = [];
            break;
        end
    end
end

% 输出最终的中心点
disp('最终的中心点坐标:');
disp(centers);

% 可视化结果
figure;
scatter(points(:, 1), points(:, 2), 20, 'bs', 'filled');
hold on;
scatter(centers(:, 1), centers(:, 2), 20,'r^', 'filled');
% 为每个中心点绘制覆盖半径的透明圆
theta = linspace(0, 2*pi, 100);
for i = 1:size(centers, 1)
    x = centers(i, 1) + r * cos(theta);
    y = centers(i, 2) + r * sin(theta);
    h = fill(x, y, 'r');
    set(h, 'FaceAlpha', 0.1); % 设置透明度为 0.1
    set(h, 'EdgeColor', 'r');
end
xlabel('X')
ylabel('Y')
title('基于聚类算法的选址结果')
legend('数据点', '中心点', '覆盖区域', 'Location', 'northeast')
结果图

图片

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2326935.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Mac 电脑移动硬盘无法识别的解决方法

在使用 Mac 电脑的过程中&#xff0c;不少用户都遇到过移动硬盘没有正常推出&#xff0c;导致无法识别的问题。这不仅影响了数据的传输&#xff0c;还可能让人担心硬盘内数据的安全。今天&#xff0c;我们就来详细探讨一下针对这一问题的解决方法。 当发现移动硬盘无法识别时&…

LeetCode Hot100 刷题笔记(4)—— 二叉树、图论

目录 一、二叉树 1. 二叉树的深度遍历&#xff08;DFS&#xff1a;前序、中序、后序遍历&#xff09; 2. 二叉树的最大深度 3. 翻转二叉树 4. 对称二叉树 5. 二叉树的直径 6. 二叉树的层序遍历 7. 将有序数组转换为二叉搜索树 8. 验证二叉搜索树 9. 二叉搜索树中第 K 小的元素 …

【计算机视觉】YOLO语义分割

一、语义分割简介 1. 定义 语义分割&#xff08;Semantic Segmentation&#xff09;是计算机视觉中的一项任务&#xff0c;其目标是对图像中的每一个像素赋予一个类别标签。与目标检测只给出目标的边界框不同&#xff0c;语义分割能够在像素级别上区分不同类别&#xff0c;从…

【SpringBoot + MyBatis + MySQL + Thymeleaf 的使用】

目录&#xff1a; 一&#xff1a;创建项目二&#xff1a;修改目录三&#xff1a;添加配置四&#xff1a;创建数据表五&#xff1a;创建实体类六&#xff1a;创建数据接口七&#xff1a;编写xml文件八&#xff1a;单元测试九&#xff1a;编写服务层十&#xff1a;编写控制层十一…

在ensp进行OSPF+RIP+静态网络架构配置

一、实验目的 1.Ospf与RIP的双向引入路由消息 2.Ospf引入静态路由信息 二、实验要求 需求&#xff1a; 路由器可以互相ping通 实验设备&#xff1a; 路由器router7台 使用ensp搭建实验坏境&#xff0c;结构如图所示 三、实验内容 1.配置R1、R2、R3路由器使用Ospf动态路由…

Redis安全与配置问题——AOF文件损坏问题及解决方案

Java 中的 Redis AOF 文件损坏问题全面解析 一、AOF 文件损坏的本质与危害 1.1 AOF 持久化原理 Redis 的 AOF&#xff08;Append-Only File&#xff09; 通过记录所有写操作命令实现持久化。文件格式如下&#xff1a; *2\r\n$6\r\nSELECT\r\n$1\r\n0\r\n *3\r\n$3\r\nSET\r\…

3.第二阶段x64游戏实战-分析人物移动实现人物加速

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 本次游戏没法给 内容参考于&#xff1a;微尘网络安全 上一个内容&#xff1a;2.第二阶段x64游戏实战-x64dbg的使用 想找人物的速度&#xff0c;就需要使用Ch…

Scala(三)

本节课学习了函数式编程&#xff0c;了解到它与Java、C函数式编程的区别&#xff1b;学习了函数的基础&#xff0c;了解到它的基本语法、函数和方法的定义、函数高级。。。学习到函数至简原则&#xff0c;高阶函数&#xff0c;匿名函数等。 函数的定义 函数基本语法 例子&…

什么是 Java 泛型

一、什么是 Java 泛型&#xff1f; 泛型&#xff08;Generics&#xff09; 是 Java 中一种强大的编程机制&#xff0c;允许在定义类、接口和方法时使用类型参数。通过泛型&#xff0c;可以将数据类型作为参数传递&#xff0c;从而实现代码的通用性和类型安全。 简单来说&…

Unity中根据文字数量自适应长宽的对话气泡框UI 会自动换行

使用Ugui制作一个可以根据文本数量自动调整宽度,并可以自动换行的文字UI 或者不要独立的Bg,那么一定要把bg的img设置成切片

【小也的Java之旅系列】02 分布式集群详解

文章目录 前言为什么叫小也 本系列适合什么样的人阅读正文单体优点缺点 CAP为什么CAP不可能全部满足&#xff1f;CAP 三选二 分布式事务分布式方案——SeataXA模式&#xff08;强一致&#xff09;AT模式&#xff08;自动补偿&#xff0c;默认模式&#xff09;TCC模式&#xff0…

Ubuntu里安装Jenkins

【方式1】&#xff1a;下载war包&#xff0c;直接运行&#xff0c;需提前搭建Java环境&#xff0c;要求11或17&#xff0c;不推荐&#xff0c;war包下载地址&#xff0c;将war包上传到服务器&#xff0c;直接使用命令启动 java -jar /data/jenkins/jenkins.war【方式2】&#…

C++包管理工具vcpkg的安装使用教程

前言 使用vcpkg可以更方便地安装各种库&#xff0c;省去配置的时间和配置失败的风险&#xff0c;类似python中的anaconda&#xff0c;懒人必备 参考 安装参考&#xff1a;https://bqcode.blog.csdn.net/article/details/135831901?fromshareblogdetail&sharetypeblogde…

微服务面试题:配置中心

&#x1f9d1; 博主简介&#xff1a;CSDN博客专家&#xff0c;历代文学网&#xff08;PC端可以访问&#xff1a;https://literature.sinhy.com/#/?__c1000&#xff0c;移动端可微信小程序搜索“历代文学”&#xff09;总架构师&#xff0c;15年工作经验&#xff0c;精通Java编…

Qt msvc2017程序无法用enigma vitrual box打包,用winrar打包

我们通常打包Qt程序用Enigma virtual box。这样我们的程序就可以在别的电脑上也能运行&#xff0c;但是有时候&#xff0c;我们发现Enigma virtual box在打包的时候&#xff0c;对于msvc2017需要编译的程序中引用webengineview模块&#xff0c;打包时候发现不能运行。 我们如何…

微服务集成测试 -华为OD机试真题(A卷、JavaScript)

题目描述 现在有n个容器服务&#xff0c;服务的启动可能有一定的依赖性&#xff08;有些服务启动没有依赖&#xff09;&#xff0c;其次&#xff0c;服务自身启动加载会消耗一些时间。 给你一个n n 的二维矩阵useTime&#xff0c;其中useTime[i][i]10表示服务i自身启动加载需…

Mac: 运行python读取CSV出现 permissionError

在MAC机器里&#xff0c;之前一直运行程序在某个指定的目录下读取excel和csv文件&#xff0c;没有出现错误&#xff0c;有一天突然出现错误&#xff1a;permissionError:[Errno 1] Operation not permitted&#xff0c; 具体错误信息如下&#xff1a; 经过调查得知&#xff0c…

UE5 学习笔记 FPS游戏制作30 显示击杀信息 水平框 UI模板(预制体)

文章目录 一制作单条死亡信息框水平框的使用创建一个水平框添加子元素调整子元素顺序子元素的布局插槽尺寸填充对齐 制作UI 根据队伍&#xff0c;设置文本的名字和颜色声明变量 将变量设置为构造参数根据队伍&#xff0c;设置文本的名字和颜色在构造事件中&#xff0c;获取玩家…

西门子TCP通讯过程中硬件连接突然断开

通信原理探秘又结合在工作中遇到的问题,关注到了通讯中的KeepAlive定时器的设置,所以做了如下实验。 硬件: 1513PLC TCP客户端 PC TCP服务器 前提条件:禁用PLC侧KeepAlive 程序: 测试流程: 打开PC端网络调试助手,设置为TCP服务器,打开链接; PC端打开WireShack软…

2025宁德时代测评Verify考什么?网申测评如何通过SHL笔试|附真题线上笔试考点、高分攻略、CATL新能源科技SHL测评宁德社招题目、面试攻略、求职建议

——职小豚 带你拆解新能源巨头招聘密码 一、宁德时代&#xff1a;新能源赛道「超级独角兽」 作为全球动力电池龙头&#xff0c;宁德时代&#xff08;CATL&#xff09;的江湖地位无需多言&#xff1a; 技术硬实力&#xff1a;麒麟电池、钠离子电池、无钴电池等黑科技加持&…