练习 3C Tree

C. Infected Tree

Problem - C - Codeforces

问题描述:一个二叉树,根节点被感染了,可以通过删除一个节点使这个树分成两个部分,从而使分离开的子树无法被感染,求这样操作,有最多多少个节点是未被感染的(删除的节点不算未被感染的)。

思路:树形dp。f[u]表示为u被感染后他的子树有最多多少个的节点未被感染。如果u是叶子节点,那么就是0;如果u只要一个节点,那么就是siz[v] - 1;如果有两个节点,那么就是max(siz[v1] - 1 + f[v2], siz[v2] - 1 + f[v1])
F ( u ) = { 叶子节点 0 一个子节点 s i z [ v ] − 1 两个子节点 m a x ( s i z [ v 1 ] − 1 + f [ v 2 ] , s i z [ v 2 ] − 1 + f [ v 1 ] ) F(u) = \begin{cases} 叶子节点 \quad 0 \\ 一个子节点 \quad siz[v] - 1 \\ 两个子节点 \quad max(siz[v1]- 1 + f[v2], siz[v2] - 1 + f[v1]) \end{cases} F(u)=⎩ ⎨ ⎧叶子节点0一个子节点siz[v]−1两个子节点max(siz[v1]−1+f[v2],siz[v2]−1+f[v1])

边界:无。

目标:
F ( 1 ) F(1) F(1)

注意事项:u的子节点不是g[u].size(),由于建树是双向边,因此多一个<u, fu>这条边。而且,也不能单纯直接令v1 = g[u][0], v2 = g[u][1],原因同上。但是可以用两个变量去处理v1 v2

代码:

cpp 复制代码
void solve() {
    int n; cin>>n;
    vector<vector<int>> g(n + 1);
    for(int i = 1; i < n; ++i) {
        int u,v; cin>>u>>v;
        g[u].push_back(v);
        g[v].push_back(u);
    }
    vector<int> f(n + 1), siz(n + 1);
    auto dfs = [&](auto &&dfs, int u, int fu) -> void {
        siz[u] = 1;
        int cnt = 0;
        PII son = {-1,-1};
        for(auto y: g[u]) {
            if(y == fu) continue;
            dfs(dfs, y,u);
            cnt++;
            if(son.vf == -1) son.vf = y;
            else son.vs = y;
            siz[u] += siz[y];
        }
        if(cnt == 0) {
            f[u] = 0;
        }
        if(cnt == 1) {
            f[u] = siz[ son.vf] - 1;
        }
        if(cnt == 2) {
            int v1 = son.vf, v2 = son.vs;
            f[u] = max(siz[v1] - 1 + f[v2], siz[v2] - 1 + f[v1]);
        }
    };
    dfs(dfs,1,-1);
    cout<<f[1]<<endl;
}

C. Kefa and Park

Problem - C - Codeforces

问题描述:一个树,叶子节点有餐馆,从根节点出发到餐馆求有多少条路。限制条件是在一条路中,不能走连续k个点权为1的点。

思路:记录一条路径上连续的1的个数,如果连续的1的个数大于k,则后面怎么都到达不了,否则就有可能。之后叶子节点能否到达即可。

代码:

cpp 复制代码
void solve() {
    int n,k; cin>>n>>k;
    vector<int> vw(n + 1),lnow(n + 1), leaf(n + 1);
    vector<vector<int>> g(n + 1);
    for(int i = 1; i <= n; ++i) cin>>vw[i];
    for(int i = 1; i < n; ++i) {
        int u,v; cin>>u>>v;
        g[u].push_back(v);
        g[v].push_back(u);
    }
    auto dfs = [&] (auto &&dfs, int u, int fu) -> void {
        for(auto y: g[u]) {
            if(y == fu) continue;
            if(vw[y] == 0) {
                lnow[y] = 0;
            } else {
                lnow[y] = lnow[u] + 1;
            }
            if(lnow[y] > k) continue;
            dfs(dfs, y,u);
        }
        if(g[u].size() == 1) leaf[u] = 1;
    };
    // 对根节点连续1进行处理
    lnow[1] = vw[1];
    dfs(dfs,1,-1);
    int ans = 0;
    for(int i = 2; i <= n; ++i) if(leaf[i]) ans += 1;
    cout<<ans;
}

C. k-Tree

Problem - C - Codeforces

问题描述:一个满k叉树,对于一个节点来说,它到它k个节点的边权分为1...k。求从根节点出发,到路径和为n且存在一条大于等于d的边权的个数。

思路:像是数字三角形的那个dp,数据范围很小。而且只有一个特殊状态,是否大于d了。设状态为从根节点到一个节点的路径和为i时,存在了大于d的边和不存在大于d的边。

状态转移方程:
F ( i , j ) = { w < d F ( i , 0 ) + = F ( i − w , 0 ) ; F ( i , 1 ) + = F ( i − w , 1 ) w ≥ d F ( i , 1 ) + = F ( i − w , 1 ) + F ( i − w , 0 ) w 为 < u , v > 的边权 F(i,j) = \begin{cases} w < d \quad F(i,0) += F(i - w,0); \quad F(i,1) += F(i - w,1) \\ w \ge d \quad F(i,1) += F(i-w,1) + F(i-w,0) \\ \end{cases} \\ w为<u,v>的边权 F(i,j)={w<dF(i,0)+=F(i−w,0);F(i,1)+=F(i−w,1)w≥dF(i,1)+=F(i−w,1)+F(i−w,0)w为<u,v>的边权

边界:
F ( 0 , 0 ) = 1 F(0,0) = 1 F(0,0)=1

目标:
F ( n , 1 ) F(n,1) F(n,1)

注意: 如果w > i那这个i - w就是负数,不合法,因此遍历的应该是min(w,i)

代码:

cpp 复制代码
const LL mod = 1000000007;
LL f[N][N];
void inpfile();
void solve() {
    int n,k,d; cin>>n>>k>>d;
    f[0][0] = 1;
    for(int i = 1; i <= n; ++i) {
        for(int j = 1; j <= min(i,k); ++j) {
            if(j < d) f[i][0] += f[i - j][0], f[i][1] += f[i-j][1];
            else f[i][1] += f[i-j][1] + f[i-j][0];
            f[i][0] %= mod;
            f[i][1] %= mod;
        }
    }
    cout<<f[n][1];
}
相关推荐
CoovallyAIHub13 分钟前
避开算力坑!无人机桥梁检测场景下YOLO模型选型指南
深度学习·算法·计算机视觉
YouQian77217 分钟前
问题 C: 字符串匹配
c语言·数据结构·算法
yanxing.D23 分钟前
408——数据结构(第二章 线性表)
数据结构·算法
艾莉丝努力练剑1 小时前
【LeetCode&数据结构】二叉树的应用(二)——二叉树的前序遍历问题、二叉树的中序遍历问题、二叉树的后序遍历问题详解
c语言·开发语言·数据结构·学习·算法·leetcode·链表
YuTaoShao1 小时前
【LeetCode 热题 100】51. N 皇后——回溯
java·算法·leetcode·职场和发展
1 小时前
3D碰撞检测系统 基于SAT算法+Burst优化(Unity)
算法·3d·unity·c#·游戏引擎·sat
Tony沈哲1 小时前
OpenCV 图像调色优化实录:基于图像金字塔的 RAW / HEIC 文件加载与调色实践
opencv·算法
我就是全世界2 小时前
Faiss中L2欧式距离与余弦相似度:究竟该如何选择?
算法·faiss
boyedu2 小时前
比特币运行机制全解析:区块链、共识算法与数字黄金的未来挑战
算法·区块链·共识算法·数字货币·加密货币
KarrySmile3 小时前
Day04–链表–24. 两两交换链表中的节点,19. 删除链表的倒数第 N 个结点,面试题 02.07. 链表相交,142. 环形链表 II
算法·链表·面试·双指针法·虚拟头结点·环形链表