博客
关于我
P2704 [NOI2001]炮兵阵地(状压dp)
阅读量:461 次
发布时间:2019-03-05

本文共 3151 字,大约阅读时间需要 10 分钟。

为了解决这个问题,我们需要在给定的网格地图上部署尽可能多的炮兵部队,同时确保任何两支炮兵部队之间不能互相攻击。攻击范围是每个炮兵部队横向和纵向各两格,包括其所在的位置。

方法思路

  • 预处理攻击范围:首先,我们需要确定每个位置的攻击范围。对于每个位置,攻击范围包括其左边、右边、上方和下方的相邻位置。
  • 动态规划状态定义:使用动态规划来记录每一行的状态。状态dp[i][mask]表示在第i行,当前行的状态是mask时的最大炮兵数目。
  • 状态转移:对于每一行的每个状态mask,检查它是否与上一行的状态prev_mask有任何攻击范围的重叠。如果没有重叠,则更新当前行的状态。
  • 预处理允许的状态列表:预先计算每个状态mask允许的上一行的状态prev_mask,以避免在动态规划中进行费时的冲突检查。
  • 解决代码

    import sys
    from sys import stdin
    def main():
    n, m = map(int, stdin.readline().split())
    grid = []
    for _ in range(n):
    line = stdin.readline().strip()
    grid.append(line)
    # 预处理attak_matrix
    attak_matrix = [[False for _ in range(m)] for _ in range(m)]
    for x in range(m):
    for y in range(m):
    if (x == y + 1 and x != y) or (y == x + 1 and y != x):
    attak_matrix[x][y] = True
    elif (x == y - 1 and x != y) or (y == x - 1 and y != x):
    attak_matrix[x][y] = True
    # 预处理allowed_masks
    allowed_masks = [set() for _ in range(1 << m)]
    for current_mask in range(1 << m):
    current_attack = 0
    for x in range(m):
    if (current_mask >> x) & 1:
    current_attack |= (1 << x)
    for prev_mask in range(1 << m):
    prev_attack = 0
    for y in range(m):
    if (prev_mask >> y) & 1:
    prev_attack |= (1 << y)
    conflict = False
    for x in range(m):
    if (current_mask >> x) & 1:
    for y in range(m):
    if (prev_mask >> y) & 1 and attak_matrix[x][y]:
    conflict = True
    break
    if conflict:
    break
    if not conflict:
    allowed_masks[current_mask].add(prev_mask)
    # 初始化动态规划
    dp = [ [-1 for _ in range(1 << m)] for _ in range(n+1)]
    dp[0][0] = 0
    for i in range(1, n+1):
    for current_mask in range(1 << m):
    # 检查当前mask是否有炮兵在H的位置
    valid = True
    for x in range(m):
    if (current_mask >> x) & 1 and grid[i-1][x] == 'H':
    valid = False
    break
    if not valid:
    continue
    current_count = bin(current_mask).count('1')
    max_prev = 0
    for prev_mask in allowed_masks[current_mask]:
    if dp[i-1][prev_mask] != -1:
    if dp[i-1][prev_mask] + current_count > max_prev:
    max_prev = dp[i-1][prev_mask] + current_count
    if max_prev > 0:
    dp[i][current_mask] = max_prev
    else:
    dp[i][current_mask] = current_count if current_count > 0 else 0
    max_result = 0
    for mask in range(1 << m):
    if dp[n][mask] > max_result:
    max_result = dp[n][mask]
    print(max_result)
    if __name__ == "__main__":
    main()

    代码解释

  • 输入处理:读取输入的网格地图,确定每个位置是否是山地(H)或平原(P)。
  • 攻击范围矩阵预处理:创建一个二维数组attak_matrix,记录每个位置之间是否在攻击范围内。
  • 允许的状态预处理:预先计算每个状态mask允许的上一行的状态prev_mask,以避免在动态规划中进行费时的冲突检查。
  • 动态规划初始化和状态转移:初始化动态规划表dp,并逐步计算每一行的状态,确保每一行的状态与上一行的状态不冲突。
  • 结果计算:遍历所有可能的状态,找到最大的炮兵数目并输出结果。
  • 转载地址:http://qwvzz.baihongyu.com/

    你可能感兴趣的文章
    Node.js安装和入门 - 2行代码让你能够启动一个Server
    查看>>
    node.js安装方法
    查看>>
    Node.js官网无法正常访问时安装NodeJS的方法
    查看>>
    node.js模块、包
    查看>>
    node.js的express框架用法(一)
    查看>>
    Node.js的交互式解释器(REPL)
    查看>>
    Node.js的循环与异步问题
    查看>>
    Node.js高级编程:用Javascript构建可伸缩应用(1)1.1 介绍和安装-安装Node
    查看>>
    nodejs + socket.io 同时使用http 和 https
    查看>>
    NodeJS @kubernetes/client-node连接到kubernetes集群的方法
    查看>>
    NodeJS API简介
    查看>>
    Nodejs express 获取url参数,post参数的三种方式
    查看>>
    nodejs http小爬虫
    查看>>
    nodejs libararies
    查看>>
    nodejs npm常用命令
    查看>>
    nodejs npm常用命令
    查看>>
    Nodejs process.nextTick() 使用详解
    查看>>
    NodeJS yarn 或 npm如何切换淘宝或国外镜像源
    查看>>
    nodejs 中间件理解
    查看>>
    nodejs 创建HTTP服务器详解
    查看>>