题解|《算法竞赛进阶指南》没有上司的舞会

没有上司的舞会

https://ac.nowcoder.com/acm/contest/1044/A

我又回来惹!

这次发的还是dp(逃

众所周知,dp分类极为广泛,最基础的有背包dp,线性dp,然后就是dp与各种算法的结合,例如与图论结合的DAG上的dp以及树性dp,与倍增结合的倍增dp,与位运算结合的状压dp...
这次我要发惹就是树性dp的一道板子题“没有上司的舞会”

首先,我们来明确一下树形dp的定义。树形dp分为树形以及dp两方面,简单来说即为在树上跑dp,dp最重要的几大性质之一就是dp一般都可以分为若干个阶段,而在树形图上设计dp就较为简单,我们一般都以节点从深到浅(子树从小到大)的顺序作为dp的‘阶段’,又由于树形dp由深入浅的性质,我们一般用深搜来完成dp的状态转移。

以上就是树形dp的大致定义惹,下面讲一下树形dp的解题思路:
首先,我们要观察题面,确定题面同时符合树形和dp的性质,确定后我们先建一个树形图,然后根据题意写出本题的状态转移方程,最后就是把代码写出来,如果是在比赛可以再手打一个暴力进行对拍,要不然就直接交上去,我在这里预祝大家ac(逃

然后我们就该开始这次的重点惹,没有上司的舞会就是一道极为经典的树形dp。
首先,根据我们前面说的,我们将职员编号看作dp状态的一维,但很明显,一维是不够的,再看题目,我们发现有一个限制条件:“不过,没有职员愿意和直接上司一起参会。”
根据这个限制,我们可以很轻松的得出dp状态的第二维为根节点是否参加,我们用0和1来表示

此时我们设f(x,0)表示x不参加舞会,那么就可以从以x为根的子树中邀请一部分职员参加:

图片说明

同样的,我们设f(x,1)表示x参加,那么所有s都无法参加,那么:

图片说明

(我们设x为当前节点,s为x的所有直接子节点,而son(x)则为x的子节点的集合)

这样,本题的状态转移方程就求出来了,最后还要注意一点,由于本题输入的是一个有根树,所以我们还要先搜一遍找到校长(即根节点),而作为根节点,dp目标为max(f(root,0),f(root,1));

本题时间复杂度为O(N),可以轻松跑过6000.

my code

#include <bits/stdc++.h>
#define re() read ()

const int N = 6010;
using namespace std;

inline int read () {
    int f = 1, x = 0; char ch = getchar ();
    while (ch < '0' || ch > '9') {
        if (ch == '-') {
            f = -1;
        }
        ch = getchar ();
    }
    while (ch >= '0' && ch <= '9') {
        x = (x << 3) + (x << 1) + ch - '0';
        ch = getchar ();
    }
    return x * f;
}

vector<int>fa[N];

int root;
int n, happy[N], f[N][2];
bool head[N];

void dfs (int x) {
    f[x][0] = 0;
    f[x][1] = happy[x];
    for (int i = 0; i < fa[x].size(); i++) {
        int y = fa[x][i];
        dfs (y);
        f[x][0] += max (f[y][0], f[y][1]);
        f[x][1] += f[y][0];
    }
}

int main () {
    n = re();
    for (int i = 1; i <= n; i++) {
        happy[i] = re(); 
    }
    for (int i = 1; i <= n-1; i++) {
        int son, father;
        son = re(), father = re();
        head[son] = 1;
        fa[father].push_back(son);
    }
    for (int i = 1; i <= n; i++) {
        if (!head[i]) {
            root = i;
            break;
        }
    }
    dfs (root);
    cout << max (f[root][0], f[root][1]) << endl;
    return 0;
} 
全部评论
自己拿自己的前排
点赞 回复 分享
发布于 2019-08-31 10:13

相关推荐

09-16 14:43
已编辑
江娱互动_研发_客户端开发
背景&nbsp;双一流本硕&nbsp;双非大圆满&nbsp;只找游戏开发相关的岗位。&nbsp;8&nbsp;月初开始秋招到现在&nbsp;投了四五十家吧,&nbsp;目前两&nbsp;offer,&nbsp;不打算继续投了,把剩下的流程走完就开始沉淀了。目前两&nbsp;offer&nbsp;一个是网易互娱测开&nbsp;base&nbsp;广州,一个是江娱互动客户端开发&nbsp;base&nbsp;北京。应该确定网易这个了,说实话北京这个我挺想去的,这家的产品和工作氛围我了解了也不错,是那种踏实做事的,可惜我是广东人。网易的测开是调剂的二志愿,看了下有内部转岗机会,所以打算后面找个时间提前实习,沉淀下再做一个&nbsp;demo&nbsp;作品,写一些&nbsp;shader,增强下图形学渲染的能力,再学点编辑器开发。看到时候内部转岗或者春招继续投客户端开发这样。后面还能再动摇的话应该就灵犀或者腾子了吧(假如这两家确认的是客户端开发岗的话)。-----------------------补下timeline网易互娱&nbsp;测开&nbsp;8.2笔试&nbsp;&nbsp;8.21&nbsp;技术面&nbsp;&nbsp;8.29&nbsp;leader&amp;HRBP面(终面)&nbsp;9.8&nbsp;录用审核(之前一直显示面试中)9.14&nbsp;oc江娱互动&nbsp;客户端开发&nbsp;8.29主程面&nbsp;9.3&nbsp;制作人面&nbsp;9.5&nbsp;BOSS面&nbsp;9.11&nbsp;口头OC&nbsp;9.15&nbsp;正式offer后面考虑了一下&nbsp;&nbsp;感觉还是能走开发就开发吧,测开不太感兴趣,要内部活水转岗还要满1年才能申请。。
点赞 评论 收藏
分享
皮格吉:不,有的厂子面试无手撕,可以试试。都是一边学一边面。哪有真正准备好的时候,别放弃
无实习如何秋招上岸
点赞 评论 收藏
分享
评论
3
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务