【刷题】动态规划——树形DP:皇宫看守

2023-11-09 01:50

本文主要是介绍【刷题】动态规划——树形DP:皇宫看守,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述题目链接

战略游戏的加强版,只要保证每个点自身被选或其相邻的点被选就可以。

由于相邻的点包括父亲和孩子,所以只考虑孩子状态不够,还要考虑父亲,因此设3个状态:

设当前结点是点 i i i,孩子结点是 s 1 , s 2 , . . . , s n s_1, s_2, ..., s_n s1,s2,...,sn,父亲结点是 p p p
1、f[i, 0] p p p有守卫,那么 i i i的孩子可以有守卫、也可以没有(状态1、2);但是 i i i没有守卫,也就是 i i i孩子的父亲没有守卫(状态0不行)。
f[i, 0] = min(f[s1, 1], f[s1, 2]) + min(f[s2, 1], f[s2, 2]) + ... + min(f[sn, 1], f[sn, 2])
2、f[i, 1] i i i有守卫,那么 i i i的孩子什么状态都行
f[i, 1] = w[i] + min(f[s1,0],f[s1,1],f[s1,2]) + min(f[s2,0],f[s2,1],f[s2,2]) + ... + min(f[sn,0],f[sn,1],f[sn,2])
3、f[i, 2]:孩子之一有守卫,先枚举哪个孩子有守卫(状态1),然后其他孩子可以有守卫也可以没有(状态1、2);但是 i i i没有守卫,也就是 i i i孩子的父亲没有守卫(状态0不行)。
f[i, 2] = Min{f[sk, 1] + min(f[s1,1], f[s1,2])+...+min(f[sj,1], f[sj,2]) +...+ min(f[sn,1], f[sn,2])} sj != sk。

#include <iostream>
#include <cstring>
using namespace std;const int N = 1505;
const int INF = 1e9;int n, f[N][3], val[N];
int nxt[N], lnk[N], head[N], idx;
bool is_son[N];void add(int a, int b) {lnk[idx] = b;nxt[idx] = head[a];head[a] = idx ++ ;
}void dfs(int node) {f[node][1] = val[node];for (int i = head[node]; i != -1; i = nxt[i]) {int son = lnk[i];dfs(son);f[node][0] += min(f[son][1], f[son][2]);f[node][1] += min(f[son][0], min(f[son][1], f[son][2]));}f[node][2] = INF;for (int i = head[node]; i != -1; i = nxt[i]) {int son1 = lnk[i];int t = f[son1][1];for (int j = head[node]; j != -1; j = nxt[j]) {int son2 = lnk[j];if (son2 == son1) continue;t += min(f[son2][1], f[son2][2]);}f[node][2] = min(f[node][2], t);}
}int main() {scanf("%d", &n);memset(head, -1, sizeof head);int a, b, k, m;for (int i = 1; i <= n; i ++ ) {scanf("%d%d%d", &a, &k, &m);val[a] = k;for (int j = 1; j <= m; j ++ ) {scanf("%d", &b);add(a, b);is_son[b] = 1;}}int root; for (int i = 1; i <= n; i ++ ) {if (!is_son[i]) root = i;}dfs(root);printf("%d\n", min(f[root][1], f[root][2]));return 0;
}

再看第三种情况:
3、f[i, 2]:孩子之一有守卫,先枚举哪个孩子有守卫(状态1),然后其他孩子可以有守卫也可以没有(状态1、2);但是 i i i没有守卫,也就是 i i i孩子的父亲没有守卫(状态0不行)。
f[i, 2] = Min{f[sk, 1] + min(f[s1,1], f[s1,2])+...+min(f[sj,1], f[sj,2]) +...+ min(f[sn,1], f[sn,2])},sj != sk。

其实第三种状态枚举的时候,不需要挨个求min(f[sj,1], f[sj,2]),可以先求个所有 s j s_j sjmin(f[sj,1], f[sj,2])的和,再扣掉 s k s_k skmin(f[sk,1], f[sk,2])、加上f[sk,1]
而所有 s j s_j sjmin(f[sj,1], f[sj,2])的和就是f[i, 0]

因此上面31~36行可改成下面的14行

void dfs(int node) {f[node][1] = val[node];for (int i = head[node]; i != -1; i = nxt[i]) {int son = lnk[i];dfs(son);f[node][0] += min(f[son][1], f[son][2]);f[node][1] += min(f[son][0], min(f[son][1], f[son][2]));}f[node][2] = INF;for (int i = head[node]; i != -1; i = nxt[i]) {int son = lnk[i];int t = f[node][0] - min(f[son][1], f[son][2]) + f[son][1];f[node][2] = min(f[node][2], t);}
}

这篇关于【刷题】动态规划——树形DP:皇宫看守的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/373513

相关文章

前端 CSS 动态设置样式::class、:style 等技巧(推荐)

《前端CSS动态设置样式::class、:style等技巧(推荐)》:本文主要介绍了Vue.js中动态绑定类名和内联样式的两种方法:对象语法和数组语法,通过对象语法,可以根据条件动态切换类名或样式;通过数组语法,可以同时绑定多个类名或样式,此外,还可以结合计算属性来生成复杂的类名或样式对象,详细内容请阅读本文,希望能对你有所帮助...

Nginx实现动态封禁IP的步骤指南

《Nginx实现动态封禁IP的步骤指南》在日常的生产环境中,网站可能会遭遇恶意请求、DDoS攻击或其他有害的访问行为,为了应对这些情况,动态封禁IP是一项十分重要的安全策略,本篇博客将介绍如何通过NG... 目录1、简述2、实现方式3、使用 fail2ban 动态封禁3.1 安装 fail2ban3.2 配

Vue3中的动态组件详解

《Vue3中的动态组件详解》本文介绍了Vue3中的动态组件,通过`component:is=动态组件名或组件对象/component`来实现根据条件动态渲染不同的组件,此外,还提到了使用`markRa... 目录vue3动态组件动态组件的基本使用第一种写法第二种写法性能优化解决方法总结Vue3动态组件动态

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

Java使用POI-TL和JFreeChart动态生成Word报告

《Java使用POI-TL和JFreeChart动态生成Word报告》本文介绍了使用POI-TL和JFreeChart生成包含动态数据和图表的Word报告的方法,并分享了实际开发中的踩坑经验,通过代码... 目录前言一、需求背景二、方案分析三、 POI-TL + JFreeChart 实现3.1 Maven

Java导出Excel动态表头的示例详解

《Java导出Excel动态表头的示例详解》这篇文章主要为大家详细介绍了Java导出Excel动态表头的相关知识,文中的示例代码简洁易懂,具有一定的借鉴价值,有需要的小伙伴可以了解下... 目录前言一、效果展示二、代码实现1.固定头实体类2.动态头实现3.导出动态头前言本文只记录大致思路以及做法,代码不进

vue基于ElementUI动态设置表格高度的3种方法

《vue基于ElementUI动态设置表格高度的3种方法》ElementUI+vue动态设置表格高度的几种方法,抛砖引玉,还有其它方法动态设置表格高度,大家可以开动脑筋... 方法一、css + js的形式这个方法需要在表格外层设置一个div,原理是将表格的高度设置成外层div的高度,所以外层的div需要

SpringBoot实现动态插拔的AOP的完整案例

《SpringBoot实现动态插拔的AOP的完整案例》在现代软件开发中,面向切面编程(AOP)是一种非常重要的技术,能够有效实现日志记录、安全控制、性能监控等横切关注点的分离,在传统的AOP实现中,切... 目录引言一、AOP 概述1.1 什么是 AOP1.2 AOP 的典型应用场景1.3 为什么需要动态插

VUE动态绑定class类的三种常用方式及适用场景详解

《VUE动态绑定class类的三种常用方式及适用场景详解》文章介绍了在实际开发中动态绑定class的三种常见情况及其解决方案,包括根据不同的返回值渲染不同的class样式、给模块添加基础样式以及根据设... 目录前言1.动态选择class样式(对象添加:情景一)2.动态添加一个class样式(字符串添加:情

SpringCloud配置动态更新原理解析

《SpringCloud配置动态更新原理解析》在微服务架构的浩瀚星海中,服务配置的动态更新如同魔法一般,能够让应用在不重启的情况下,实时响应配置的变更,SpringCloud作为微服务架构中的佼佼者,... 目录一、SpringBoot、Cloud配置的读取二、SpringCloud配置动态刷新三、更新@R