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

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

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

方法思路

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

    import sysfrom sys import stdindef 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/

    你可能感兴趣的文章
    Nginx 反向代理配置去除前缀
    查看>>
    nginx 后端获取真实ip
    查看>>
    Nginx 学习总结(16)—— 动静分离、压缩、缓存、黑白名单、性能等内容温习
    查看>>
    Nginx 学习总结(17)—— 8 个免费开源 Nginx 管理系统,轻松管理 Nginx 站点配置
    查看>>
    Nginx 常用配置清单
    查看>>
    nginx 常用配置记录
    查看>>
    Nginx 我们必须知道的那些事
    查看>>
    Nginx 的 proxy_pass 使用简介
    查看>>
    Nginx 的配置文件中的 keepalive 介绍
    查看>>
    Nginx 负载均衡与权重配置解析
    查看>>
    Nginx 负载均衡详解
    查看>>
    nginx 配置 单页面应用的解决方案
    查看>>
    nginx 配置https(一)—— 自签名证书
    查看>>
    nginx 配置~~~本身就是一个静态资源的服务器
    查看>>
    Nginx 配置解析:从基础到高级应用指南
    查看>>
    nginx+Tomcat性能监控
    查看>>
    Nginx下配置codeigniter框架方法
    查看>>
    nginx报错:the “ssl“ parameter requires ngx_http_ssl_module in /usr/local/nginx/conf/nginx.conf:128
    查看>>
    nginx添加模块与https支持
    查看>>
    Nginx用户认证
    查看>>