草庐IT

2022年电工杯数模竞赛B题第一问解法分享(附Python代码)

RedJACK~ 2024-05-08 原文

2022年电工杯数模竞赛B题第一问解法分享(附Python代码)

题目

我们这里只分析第一问

1.图 1 给出 14 个地点, 其中实线代表各地点之间的路线情况。 若目前所有应急物资集中在第 9 个地点,配送车辆的最大载重量为 1000 千克,采取配送车辆(无人机不参与)的配送模式。请结合附件 1, 建立完成一次整体配送的数学模型, 并给出最优方案。

完成一次整体配送所需要的时间是按照配送车辆从出发开始至全部返回到出发地点的时间来计算。

附件1:

邻接矩阵

每个地点的需求量

题目分析

  • 所有地点的物资需求为762kg,小于车辆的最大载重量1000kg。也就是不需要车辆中途回去再取物资。
  • 整个物资配送路径最终得是个回路。
  • 由于给了个图,所以需要我们事先用图论相关算法求出任意两结点之间的最短路径。可以考虑迪杰斯特拉算法、弗洛伊德算法等。
  • 得到任意两点的最短路径矩阵(二维数组)后,我们就可以规划路线了。
  • 其实我们只要确定所有地点的访问顺序即可获得的具体的路线
  • 所有地点的访问顺序其实是一个全排列问题,即所有可能是: 1 × 13 × 12 × 11 × 10 × 9 × 8 × 7 × 6 × 5 × 4 × 3 × 2 × 1 × 1 = 6 , 227 , 020 , 800 1 \times 13 \times 12 \times 11 \times 10 \times 9 \times 8 \times 7 \times 6 \times 5 \times 4 \times 3 \times 2 \times 1 \times 1 = 6,227,020,800 1×13×12×11×10×9×8×7×6×5×4×3×2×1×1=6,227,020,800 种访问顺序。其中第一个地点和最后一个地点的编号确定的。
  • 但是暴力枚举是10亿的数量级,需要很久可能几十个小时。
  • 所以我们要对枚举范围进行优化,即排除一些没必要的枚举。其实对于任意一个结点我们只需枚举最近的几个结点就可以了。比如:枚举最近的4个结点,原来有13个结点可以选择,现在我们只要枚举4结点就可以了。

首先,我们使用多次迪杰斯特拉算法得到了任意两节点之间的最短路径,然在在此基础上做访问顺序规划。我们以下的建模求解思路就是在暴力枚举的基础上做了剪枝优化,只枚举最近的4个结点的路线。当然,我们是有可能漏掉最佳答案的,但是一般情况下,这种可能性很小很小,所以我们是可以得到全局最优解的。

结果展示

最短距离矩阵

地点配送顺序

具体路线
9 → 8 → 12 → 11 → 1 → 7 → 5 → 2 → 3 → 5 → 6 → 4 → 6 → 10 → 14 → 13 → 9 9 \to 8 \to 12 \to 11 \to 1 \to 7 \to 5 \to 2 \to 3 \to 5 \to 6 \to 4 \to 6 \to 10 \to 14 \to 13 \to 9 9812111752356461014139
总距离:582km

总耗时:11.64h

Python代码

from cmath import inf
from operator import truediv

print("--------------------------读取邻接矩阵---------------------------------")
'''
# 读取邻接矩阵
graph = []  # 保存邻接矩阵
with open("data.csv") as f:
    con = list(f.readlines())
    for line in con:
        line = line.split(',')
        line[-1] = line[-1][:-1]
        graph.append([int(i) if i != '' else inf for i in line])

n = len(graph)  # 地点个数

for i in range(n):  # 输出邻接矩阵
    print(graph[i])

'''
# 如果没有事先处理好本地文件直接用代码写入邻接矩阵也可
graph = [
    [inf , inf , inf , inf , 54  , inf , 55  , inf , inf , inf , 26  , inf , inf , inf ],
    [inf , inf , 56  , inf , 18  , inf , inf , inf , inf , inf , inf , inf , inf , inf ],
    [inf , 56  , inf , inf , 44  , inf , inf , inf , inf , inf , inf , inf , inf , inf ],
    [inf , inf , inf , inf , inf , 28  , inf , inf , inf , inf , inf , inf , inf , inf ],
    [54  , 18  , 44  , inf , inf , 51  , 34  , 56  , 48  , inf , inf , inf , inf , inf ],
    [inf , inf , inf , 28  , 51  , inf , inf , inf , 27  , 42  , inf , inf , inf , inf ],
    [55  , inf , inf , inf , 34  , inf , inf , 36  , inf , inf , inf , 38  , inf , inf ],
    [inf , inf , inf , inf , 56  , inf , 36  , inf , 29  , inf , inf , 33  , inf , inf ],
    [inf , inf , inf , inf , 48  , 27  , inf , 29  , inf , 61  , inf , 29  , 42  , 36  ],
    [inf , inf , inf , inf , inf , 42  , inf , inf , 61  , inf , inf , inf , inf , 25  ],
    [26  , inf , inf , inf , inf , inf , inf , inf , inf , inf , inf , 24  , inf , inf ],
    [inf , inf , inf , inf , inf , inf , 38  , 33  , 29  , inf , 24  , inf , inf , inf ],
    [inf , inf , inf , inf , inf , inf , inf , inf , 42  , inf , inf , inf , inf , 47  ],
    [inf , inf , inf , inf , inf , inf , inf , inf , 36  , 25  , inf , inf , 47  , inf ]
]

n = len(graph)  # 地点个数


print("-------------------------迪杰斯特拉算法----------------------------------")
# 迪杰斯特拉算法
def Dijkstra(s):
	vis = [False for i in range(n)]
	vis[s] = True
	for i in range(n):
		u = -1
		MIN = inf
		for j in range(n):
			if not vis[j] and MIN > graph[s][j]:
				u = j
				MIN = graph[s][j]
		if u == -1:
			return
		vis[u] = True
		for j in range(n):
			if not vis[j] and graph[u][j] != inf and graph[s][j] > graph[s][u]+graph[u][j]:
				graph[s][j] = graph[s][u] + graph[u][j]

for i in range(n):  # 多次迪杰斯特拉得到任意两点的最短路径
    Dijkstra(i)

for i in range(n):
    print(graph[i])

print("-------------------------最短路径----------------------------------")
# 最小路径回路模型
class MinLoop:
    '''
    该模型的使用前提是已经用算法整理了任意两点的最短路径的临界矩阵。
    我们这里使用的是用多次迪杰斯特拉算法得到的邻接矩阵
    '''
    def __init__(self,gp_min_dis,min_num,begin):
        '''
        gp_min_dis : List[List[float]] 或者 numpy二维数组 ---> 记录了图的任意两结点最短路径的邻接矩阵
        min_num : int ---> 枚举最近的结点路径数目
        begin : int ---> 路径的起始结点的下标
        '''
        self.gp_min_dis = gp_min_dis
        self.num = len(gp_min_dis)
        self.min_num = min_num
        self.begin = begin
        self.allPath = []       # 记录当前模型的所有可能路径
        self.allDistance = []   # 记录当前模型的所有可能路径的最短路径
        self.min_dis_dict = {}  # 将任意两点最短路径备份为字典以免修改原始数据
        for i in range(self.num):
            self.min_dis_dict[i] = gp_min_dis[i]
    
    # 获取当前模型的所有可能路径
    def genAllPath(self):
        path = [-1 for _ in range(self.num+1)]
        path[0] = self.begin
        path[-1] = self.begin
        self.DFS(0,self.begin,path)
        return self.allPath.copy()

    # 深度优先路径搜索
    def DFS(self,deep,pos,path):
        if deep == self.num-1:
            self.allPath.append(path.copy())
            return

        select = self.getSelect(path,pos)
        for nextPos in select:
            path[deep+1] = nextPos
            self.DFS(deep+1,nextPos,path)
            path[deep+1] = -1

    # 获取结点的未访问的最近几个结点
    def getSelect(self,path,pos):
        have = []
        for _ in range(self.min_num):
            min_dis_pos = -1
            min_dis = inf
            for j in range(self.num):
                if j not in path and j not in have:
                    if min_dis > self.min_dis_dict[pos][j]:
                        min_dis = self.min_dis_dict[pos][j]
                        min_dis_pos = j
            if min_dis_pos != -1:
                have.append(min_dis_pos)
        return have
    
    # 计算路径
    def getPathDistance(self,path):
        now = self.begin
        dis_sum = 0
        for to in path[1:]:
            dis_sum += self.min_dis_dict[now][to]
            now = to
        return dis_sum
    
    # 获取当前模型下的所有路径值
    def getAlldistance(self):
        distance_list = []
        for path in self.allPath:
            distance_list.append(self.getPathDistance(path))
        self.allDistance = distance_list
        return distance_list

    # 获取该模型的最短路径规划及路径值
    def getBestPath(self):
        bestDis = min(self.allDistance)
        for i in range(len(self.allDistance)):
            if bestDis == self.allDistance[i]:
                return (self.allPath[i].copy(),bestDis)
        return

model = MinLoop(graph,3,8)  # 选取最近的3个点进行枚举搜索,这个参数影响最终的运行时间
# 运行模型
model.genAllPath()
model.getAlldistance()
path,dis = model.getBestPath()

for i in range(len(path)):
    path[i] += 1    # 将下标改为标号

print("当前模型最短路径规划:",path) # [9, 8, 12, 11, 1, 7, 5, 2, 3, 6, 4, 10, 14, 13, 9]
print("当前模型最短路径值:",dis)   # 582.0
print("当前模型最短路径值耗时:",dis/50)   # 11.64

有关2022年电工杯数模竞赛B题第一问解法分享(附Python代码)的更多相关文章

  1. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  2. ruby - 如何在 buildr 项目中使用 Ruby 代码? - 2

    如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby​​

  3. ruby-on-rails - Rails 源代码 : initialize hash in a weird way? - 2

    在rails源中:https://github.com/rails/rails/blob/master/activesupport/lib/active_support/lazy_load_hooks.rb可以看到以下内容@load_hooks=Hash.new{|h,k|h[k]=[]}在IRB中,它只是初始化一个空哈希。和做有什么区别@load_hooks=Hash.new 最佳答案 查看rubydocumentationforHashnew→new_hashclicktotogglesourcenew(obj)→new_has

  4. ruby-on-rails - 浏览 Ruby 源代码 - 2

    我的主要目标是能够完全理解我正在使用的库/gem。我尝试在Github上从头到尾阅读源代码,但这真的很难。我认为更有趣、更温和的踏脚石就是在使用时阅读每个库/gem方法的源代码。例如,我想知道RubyonRails中的redirect_to方法是如何工作的:如何查找redirect_to方法的源代码?我知道在pry中我可以执行类似show-methodmethod的操作,但我如何才能对Rails框架中的方法执行此操作?您对我如何更好地理解Gem及其API有什么建议吗?仅仅阅读源代码似乎真的很难,尤其是对于框架。谢谢! 最佳答案 Ru

  5. ruby - 模块嵌套代码风格偏好 - 2

    我的假设是moduleAmoduleBendend和moduleA::Bend是一样的。我能够从thisblog找到解决方案,thisSOthread和andthisSOthread.为什么以及什么时候应该更喜欢紧凑语法A::B而不是另一个,因为它显然有一个缺点?我有一种直觉,它可能与性能有关,因为在更多命名空间中查找常量需要更多计算。但是我无法通过对普通类进行基准测试来验证这一点。 最佳答案 这两种写作方法经常被混淆。首先要说的是,据我所知,没有可衡量的性能差异。(在下面的书面示例中不断查找)最明显的区别,可能也是最著名的,是你的

  6. ruby - 寻找通过阅读代码确定编程语言的ruby gem? - 2

    几个月前,我读了一篇关于ruby​​gem的博客文章,它可以通过阅读代码本身来确定编程语言。对于我的生活,我不记得博客或gem的名称。谷歌搜索“ruby编程语言猜测”及其变体也无济于事。有人碰巧知道相关gem的名称吗? 最佳答案 是这个吗:http://github.com/chrislo/sourceclassifier/tree/master 关于ruby-寻找通过阅读代码确定编程语言的rubygem?,我们在StackOverflow上找到一个类似的问题:

  7. ruby - Net::HTTP 获取源代码和状态 - 2

    我目前正在使用以下方法获取页面的源代码:Net::HTTP.get(URI.parse(page.url))我还想获取HTTP状态,而无需发出第二个请求。有没有办法用另一种方法做到这一点?我一直在查看文档,但似乎找不到我要找的东西。 最佳答案 在我看来,除非您需要一些真正的低级访问或控制,否则最好使用Ruby的内置Open::URI模块:require'open-uri'io=open('http://www.example.org/')#=>#body=io.read[0,50]#=>"["200","OK"]io.base_ur

  8. Python 相当于 Perl/Ruby ||= - 2

    这个问题在这里已经有了答案:关闭10年前。PossibleDuplicate:Pythonconditionalassignmentoperator对于这样一个简单的问题表示歉意,但是谷歌搜索||=并不是很有帮助;)Python中是否有与Ruby和Perl中的||=语句等效的语句?例如:foo="hey"foo||="what"#assignfooifit'sundefined#fooisstill"hey"bar||="yeah"#baris"yeah"另外,类似这样的东西的通用术语是什么?条件分配是我的第一个猜测,但Wikipediapage跟我想的不太一样。

  9. java - 什么相当于 ruby​​ 的 rack 或 python 的 Java wsgi? - 2

    什么是ruby​​的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht

  10. 程序员如何提高代码能力? - 2

    前言作为一名程序员,自己的本质工作就是做程序开发,那么程序开发的时候最直接的体现就是代码,检验一个程序员技术水平的一个核心环节就是开发时候的代码能力。众所周知,程序开发的水平提升是一个循序渐进的过程,每一位程序员都是从“菜鸟”变成“大神”的,所以程序员在程序开发过程中的代码能力也是根据平时开发中的业务实践来积累和提升的。提高代码能力核心要素程序员要想提高自身代码能力,尤其是新晋程序员的代码能力有很大的提升空间的时候,需要针对性的去提高自己的代码能力。提高代码能力其实有几个比较关键的点,只要把握住这些方面,就能很好的、快速的提高自己的一部分代码能力。1、多去阅读开源项目,如有机会可以亲自参与开源

随机推荐