遗传算法实践(七) 最小生成树问题

最小生成树问题描述

最小生成树问题时指在由m个节点和n条边组成的网络模型中寻找连接所有节点的生成树,使得其所有边的权值之和最小。最小生成树问题广泛应用于系统设计、选址规划等组合优化问题中。

在由m个节点和n条边组成的图G=(V,E)中,当各边的权值w_{ij}给定时,最小生成树问题可以作为0-1整数规划问题描述如下:
min\ \sum_{i=1}^{m}\sum_{j=1}^mw_{ij}x_{ij} \\s.t.\ \sum_{i=1}^{m}\sum_{j=1}^mx_{ij}=m-1 \\ \sum_{i=1}^{m}\sum_{j=1}^mx_{ij} \le |S|-1 \\ x_{ij}\in \{0,1\}
其中x_{ij}是边(i,j)的决策变量,选择该边时取1,反之取0。

求解最小生成树的代表算法有Kruskal法和Prim法等。

最小生成树问题示例

连接各节点的可能边如下图所示:

生成树模型的可用路径

各边的权值如下表所示:

k edge(i,j) weight w_{ij} k edge(i,j) weight w_{ij}
1 (1,2) 35 21 (5,7) 26
2 (1,3) 23 22 (5,8) 35
3 (1,4) 26 23 (5,9) 63
4 (1,5) 29 24 (5,10) 23
5 (1,6) 52 25 (6,7) 27
6 (2,3) 34 26 (6,8) 29
7 (2,4) 23 27 (6,9) 65
8 (2,5) 68 28 (6,10) 24
9 (2,6) 42 29 (7,8) 38
10 (3,4) 23 30 (7,11) 52
11 (3,7) 51 31 (7,12) 41
12 (3,8) 23 32 (8,9) 62
13 (3,9) 64 33 (8,11) 26
14 (3,10) 28 34 (8,12) 30
15 (4,5) 54 35 (9,10) 47
16 (4,7) 24 36 (9,11) 68
17 (4,8) 47 37 (9,12) 33
18 (4,9) 53 38 (10,11) 42
19 (4,10) 24 39 (10,12) 26
20 (5,6) 56 40 (11,12) 51

遗传算法求解最小生成树问题

个体编码

用PrimPred法进行个体的编码,其流程如下:

设有n个需要连接的节点,邻接点集S_i包含所有与节点i相邻的节点。

将出发点初始化为i\leftarrow 1,已连接点集合初始化为C\leftarrow [1],建立邻接点集S_i,i\in V,可选边集E\leftarrow \{(i,j)|j\in S_i\}

  • Step1: 从可选边集E中随机选择一条边(i,j),将染色体v的第j个元素设为v(j)\leftarrow i
  • Step2: 更新已连接点集合C\leftarrow C\bigcup\{j\}
  • Step3: 更新出发点i\leftarrow j
  • Step 4: 更新可选边集合E\leftarrow \{(i,j)|j\in S_i\},从中移除与已经连接的节点相关的边 E\leftarrow E\ \backslash \{(i,j)|i\in C \& j\in C \};
  • Step5: 重复Step1-4,直到获得n-1条边(n为定点数)。

输出获得的染色体编码v

解码

在解码时,路径由节点编号和对应的染色体给出。下图展示了一个根据PrimPred法获得的个体编码,其对应的生成树的边为(2,5),(3,1),(4,3),(5,8),(6,9),(7,8),(8,4),(9,5),(10,12),(11,8),(12,11)

基于PrimPred方法编码的解码

交叉操作

在交叉操作中,将两个父代染色体个体解码出的树进行叠加后,重新生成可用边集合和邻接节点集合,再在新集合的基础上重新使用PrimPred编码,得到一个子代染色体。用这种方法可以保留父代中优秀的子结构,从而加速收敛。

突变操作

用最小代价法(Low Cost Method)生成新的个体,先在父代染色体中随机删除一条边,使得原图分为两个互不连接的子图,然后从两个子图的节点集中,添加一条权数最小的边使其恢复连通,从而生成一个新个体。

代码示例

完整代码如下:

## 环境设定
import numpy as np
import matplotlib.pyplot as plt
from deap import base, tools, creator, algorithms
import random

params = {
    'font.family': 'serif',
    'figure.dpi': 300,
    'savefig.dpi': 300,
    'font.size': 12,
    'legend.fontsize': 'small'
}
plt.rcParams.update(params)

from copy import deepcopy
# 问题定义
creator.create('FitnessMin', base.Fitness, weights=(-1.0,)) #最小化问题
creator.create('Individual', list, fitness=creator.FitnessMin)

# 个体编码
edges = [
    '1,2', '1,3', '1,4', '1,5', '1,6',
    '2,3', '2,4', '2,5', '2,6',
    '3,4', '3,7', '3,8', '3,9', '3,10',
    '4,5', '4,7', '4,8', '4,9', '4,10',
    '5,6', '5,7', '5,8', '5,9', '5,10',
    '6,7', '6,8', '6,9', '6,10',
    '7,8', '7,11', '7,12',
    '8,9', '8,11', '8,12',
    '9,10', '9,11', '9,12',
    '10,11', '10,12',
    '11,12'
]
def generateSFromEdges(edges):
    '''用关联表存储图,从提供的边集中生成与各个节点i相邻的节点集合Si
    输入:edges -- list, 其中每个元素为每个节点上的边,每个元素均为一个str 'i,j'
    输出:nodeDict -- dict, 形如{'i':[j,k,l]},记录从每个节点能到达的其他节点
    '''
    nodeDict = {}
    for edge in edges:
        i,j = edge.split(',')
        if not i in nodeDict:
            nodeDict[i] = [int(j)]
        else:
            nodeDict[i].append(int(j))
        # 无向图中(i,j)与(j,i)是相同的
        if not j in nodeDict:
            nodeDict[j] = [int(i)]
        else:
            nodeDict[j].append(int(i))
    return nodeDict

def eligibleEdgeSet(nodeDict, i):
    '''辅助函数,生成从节点i出发的所有边(i,j)
    输入:nodeDict -- dict,记录每个节点能到达的其他节点
    i -- 起始节点,int
    输出:edgeSet -- list,记录从节点i出发可能的所有边的集合,其中每个元素为一条边,形如'i,j'的str
    '''
    endNodeSet = nodeDict[str(i)] # i节点的所有后续节点
    edgeSet = []
    for eachNode in endNodeSet:
        edgeSet.append(str(i) + ',' + str(eachNode))
    return edgeSet
    
def genEdgeFromNodeSet(nodeSet):
    '''辅助函数,给定一个noseSet,返回其中所有可能的边
    输入: nodeSet -- list,每个元素均为int,代表一个节点
    输出:edgesGen -- list, 每个元素代表一条边,形如'i,j'的str
    '''
    from itertools import combinations
    combs = combinations(nodeSet, 2)
    edgesGen = []
    for eachItem in combs:
        edgesGen.append(str(eachItem[0])+','+str(eachItem[1]))
        edgesGen.append(str(eachItem[1])+','+str(eachItem[0]))
    return edgesGen
    
def PrimPredCoding(edges=edges):
    '''从给定的节点集合中以PrimPred方法生成染色体
    输入:
    输出:ind -- 个体实数编码,长度为节点数-1
    '''
    nodeDict = generateSFromEdges(edges)
    nodeCount = len(nodeDict) # 这个长度等于节点数
    i = 1
    nodeSet = [i] # 用于保存迭代中间变量
    edgeSet = eligibleEdgeSet(nodeDict, i) # 从i出发所有可能的边
    iterIdx = 1
    ind = [0]*(nodeCount-1) # [1]作为默认起始点,需要的编码长度为节点数-1
    while iterIdx < nodeCount:
        edgeSelected = edgeSet[random.randint(0,len(edgeSet)-1)] # 随机选取一条可行边
        i = int(edgeSelected.split(',')[0]) # 所选边的起点
        j = int(edgeSelected.split(',')[1]) # 所选边的终点,范围为2到len(nodeDict)+1
#         print(len(ind), j-1)
        ind[j-2] = i # 注意j是从1到len(#node)的,作为index应该减去1
        nodeSet.append(j)
        i = j
        if not i==len(nodeDict) +1: # 当i时最终节点时,没有可用的边了
            edgeSet = edgeSet + eligibleEdgeSet(nodeDict, i)
        edgesToExclude = genEdgeFromNodeSet(nodeSet) # 需要从集合中删掉的边
        edgeSet = list(set(edgeSet) - set(edgesToExclude))
        iterIdx += 1
    return ind

toolbox = base.Toolbox()
toolbox.register('individual', tools.initIterate, creator.Individual ,PrimPredCoding)

# 解码
def decoding(ind):
    '''对给定的染色体编码,解码为生成树(边的集合)
    输入:ind -- 个体实数编码,长度为节点数-1
    输出:generatedTree -- 边的集合,类似于edges,每个元素为形如'i,j'的str
    '''
    generatedTree = []
    geneLen = len(ind)
    for i,j in zip(ind, range(2,2+geneLen)):
        generatedTree.append(str(min(i,j))+','+str(max(i,j)))
    return generatedTree

# 评价个体
weightDict = {
    '1,2': 35, '1,3': 23, '1,4': 26, '1,5': 29, '1,6': 52,
    '2,3': 34, '2,4': 23, '2,5': 68, '2,6': 42,
    '3,4': 23, '3,7': 51, '3,8': 23, '3,9': 64, '3,10': 28,
    '4,5': 54, '4,7': 24, '4,8': 47, '4,9': 53, '4,10': 24,
    '5,6': 56, '5,7': 26, '5,8': 35, '5,9': 63, '5,10': 23,
    '6,7': 27, '6,8': 29, '6,9': 65, '6,10': 24,
    '7,8': 38, '7,11': 52, '7,12': 41,
    '8,9': 62, '8,11': 26, '8,12': 30,
    '9,10': 47, '9,11': 68, '9,12': 33,
    '10,11': 42, '10,12': 26,
    '11,12': 51
}
def evaluate(ind):
    '''对给定的染色体编码,返回给定边的权值之和'''
    generatedTree = decoding(ind)
    weightSum = 0
    for eachEdge in generatedTree:
        weightSum += weightDict[eachEdge]
    return weightSum,

# 交叉操作
def cxPrimPred(ind1, ind2):
    '''给定两个个体,将其边叠加,再根据PrimPred编码方法生成新个体'''
    edges1 = decoding(ind1) # 将个体解码为边
    edges2 = decoding(ind2)
    edgesCombined = list(set(edges1 + edges2))
    return PrimPredCoding(edges=edgesCombined)

# 突变操作
def mutLowestCost(ind, weightDict=weightDict):
    '''给定一个个体,用lowest cost method生成新个体,先将父代染色体中随机删除一条边,将原图
    分为两个互不连通的子图,然后选择连接这两个子图的具有最小权数的边并连接子图'''
    # 将原图分为两个互不连通的子图
    edges = decoding(ind)
    edgeIdx = random.randint(0, len(edges)-1) # 选择一条需要删除的边
    u = int(edges[edgeIdx].split(',')[0])
    v = int(edges[edgeIdx].split(',')[1])
    edges = edges[:edgeIdx] + edges[edgeIdx+1:] # 删除选中的边
    # 将属于两个子图的顶点分别归如两个点集
    A = [0]*(len(ind)+1)
    U = edges
    while U:
        randomEdgeIdx = random.randint(0, len(U)-1) # 随机选择一条边(i,j)
        i = int(U[randomEdgeIdx].split(',')[0])
        j = int(U[randomEdgeIdx].split(',')[1])
        U = U[:randomEdgeIdx] + U[randomEdgeIdx+1:] # 删除选中的边
        if A[i-1] == 0 and A[j-1] == 0:
            l = min(i,j)
            A[i-1] = l
            A[j-1] = l
        elif A[i-1] == 0 and A[j-1] != 0:
            A[i-1] = A[j-1]
        elif A[i-1] != 0 and A[j-1] == 0:
            A[j-1] = A[i-1]
        else:
            if A[i-1] < A[j-1]:
                idx = [A[_]==A[j-1] for _ in range(len(A))]
                A = np.where(idx, A[i-1], A)
            elif A[i-1] > A[j-1]:
                idx = [A[_]==A[i-1] for _ in range(len(A))]
                A = np.where(idx, A[j-1], A)
    nodeSet1 = [_+1 for _ in range(len(A)) if A[_]==A[u-1]] # 注意index和节点编号的关系
    nodeSet2 = [_+1 for _ in range(len(A)) if A[_]==A[v-1]]
    # 选择两个点集中代价最小的边,添进边中
    minCostEdge = None
    minEdgeCost = 1e5
    for vert1 in nodeSet1:
        for vert2 in nodeSet2:
            key = str(min(vert1,vert2)) + ',' + str(max(vert1,vert2))
            if key in weightDict:
                if weightDict[key] < minEdgeCost:
                    minEdgeCost = weightDict[key]
                    minCostEdge = key
    edges = edges + [minCostEdge]
    # 从边还原编码
    return PrimPredCoding(edges)

# 注册工具
toolbox.register('evaluate', evaluate)
toolbox.register('select', tools.selTournament, tournsize=2)
toolbox.register('mate', cxPrimPred)
toolbox.register('mutate', mutLowestCost)

# 迭代数据
stats = tools.Statistics(key=lambda ind:ind.fitness.values)
stats.register('min', np.min)
stats.register('avg', np.mean)
stats.register('std', np.std)

##---------------------------
# 遗传算法参数
toolbox.ngen = 200
toolbox.popSize = 100
toolbox.cxpb = 0.8
toolbox.mutpb = 0.1

# 生成初始族群
toolbox.register('population', tools.initRepeat, list, toolbox.individual)
pop = toolbox.population(toolbox.popSize)

# 遗传算法主程序
hallOfFame = tools.HallOfFame(maxsize=1)
logbook = tools.Logbook()
logbook.header = ['gen', 'nevals'] + (stats.fields if stats else [])

# Evaluate the individuals with an invalid fitness
invalid_ind = [ind for ind in pop if not ind.fitness.valid]
fitnesses = toolbox.map(toolbox.evaluate, invalid_ind)
for ind, fit in zip(invalid_ind, fitnesses):
    ind.fitness.values = fit

hallOfFame.update(pop)

record = stats.compile(pop) if stats else {}
logbook.record(gen=0, nevals=len(invalid_ind), **record)

# Begin the generational process
for gen in range(1, toolbox.ngen + 1):
    # Select the next generation individuals
    offspring = toolbox.select(pop, len(pop))

    # Vary the pool of individuals
    for i in range(1, len(offspring), 2):
        if random.random() < toolbox.cxpb:
            offspring[i - 1][:] = toolbox.mate(offspring[i - 1],
                                                          offspring[i])
            del offspring[i - 1].fitness.values

    for i in range(len(offspring)):
        if random.random() < toolbox.mutpb:
            offspring[i][:] = toolbox.mutate(offspring[i])
            del offspring[i].fitness.values
            
    # Evaluate the individuals with an invalid fitness
    invalid_ind = [ind for ind in offspring if not ind.fitness.valid]
    fitnesses = toolbox.map(toolbox.evaluate, invalid_ind)
    for ind, fit in zip(invalid_ind, fitnesses):
        ind.fitness.values = fit

    # Update the hall of fame with the generated individuals
    hallOfFame.update(offspring)

    # Replace the current population by the offspring
    pop[:] = offspring

    # Append the current generation statistics to the logbook
    record = stats.compile(pop) if stats else {}
    logbook.record(gen=gen, nevals=len(invalid_ind), **record)
print(logbook)

输出结果:

## 输出结果
bestInd = hallOfFame.items[0]
bestFitness = bestInd.fitness.values
bestEdges = decoding(bestInd)
print('最小生成树的边为:'+str(bestEdges))
print('最小生成树的代价为:'+str(bestFitness))

## 画出迭代图
minFit = logbook.select('min')
avgFit = logbook.select('avg')
plt.plot(minFit, 'b-', label='Minimum Fitness')
plt.plot(avgFit, 'r-', label='Average Fitness')
plt.xlabel('# Gen')
plt.ylabel('Fitness')
plt.legend(loc='best')

## 结果:
#最小生成树的边为:['2,4', '1,3', '3,4', '5,10', '6,10', '4,7', '3,8', '9,12', '4,10', '8,11', '10,12']
#最小生成树的代价为:(272.0,)

计算的迭代过程可视化:

MinimumSpanTree_iterations

生成的最小生成树为:

生成树模型结果
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容