• [BZOJ4813][CQOI2017]小Q的棋盘(DP,贪心)


    4813: [Cqoi2017]小Q的棋盘

    Time Limit: 10 Sec  Memory Limit: 512 MB
    Submit: 804  Solved: 441
    [Submit][Status][Discuss]

    Description

    小Q正在设计一种棋类游戏。在小Q设计的游戏中,棋子可以放在棋盘上的格点中。某些格点之间有连线,棋子只能
    在有连线的格点之间移动。整个棋盘上共有V个格点,编号为0,1,2…,V-1,它们是连通的,也就是说棋子从任意格
    点出发,总能到达所有的格点。小Q在设计棋盘时,还保证棋子从一个格点移动到另外任一格点的路径是唯一的。
    小Q现在想知道,当棋子从格点0出发,移动N步最多能经过多少格点。格点可以重复经过多次,但不重复计数。

    Input

    第一行包含2个正整数V,N,其中V表示格点总数,N表示移动步数。
    接下来V-1行,每行两个数Ai,Bi,表示编号为Ai,Bi的两个格点之间有连线。
    V,N≤ 100, 0 ≤Ai,Bi<V 

    Output

    输出一行一个整数,表示最多经过的格点数量。

    Sample Input

    5 2
    1 0
    2 1
    3 2
    4 3

    Sample Output

    3
    从格点 0 出发移动 2 步。经过 0, 1, 2 这 3 个格点。

    HINT

    Source

    [Submit][Status][Discuss]

    普及题?数据范围这么小?

    f[i][j]表示以i为根的子树走j步最多能走几个点(不要求返回i节点),g[i][j]要求返回,直接递推即可。

    考虑贪心,首先一棵子树能走完肯定会尽量走完,因为除了最后一步的那条链之外每个点走的步数都为2(因为要返回)。

    直接枚举最后在哪里停下,设停下的深度为$d[x]$(根节点深度为0),则答案为$d[x]+1+frac{m-d[x]}{2}=frac{d[x]+m+2}{2}$

    DP:

     1 #include<iostream>
     2 #include<cstdio>
     3 #include<cstring>
     4 #define rep(i,l,r) for (int i=l; i<=r; i++)
     5 typedef long long ll;
     6 using namespace std;
     7 
     8 const int N=110;
     9 int u,v,f[N][N],g[N][N],cnt,n,m,ans,to[N<<1],nxt[N<<1],h[N];
    10 char ch;
    11 void add(int u,int v){ to[++cnt]=v; nxt[cnt]=h[u]; h[u]=cnt; }
    12 
    13 void dfs(int x,int fa){
    14     f[x][0]=g[x][0]=1;
    15     for (int i=h[x],k; i; i=nxt[i])
    16         if ((k=to[i])!=fa){
    17             dfs(k,x);
    18             for (int j=m; j; j--)
    19                 for (int l=0; l<j; l++){
    20                     if (l<j-1) f[x][j]=max(f[x][j],f[k][l]+f[x][j-l-2]),
    21                                   g[x][j]=max(g[x][j],f[k][l]+g[x][j-l-2]);
    22                     g[x][j]=max(g[x][j],g[k][l]+f[x][j-l-1]);
    23                 }
    24         }
    25 }
    26 
    27 int main(){
    28     freopen("chessboard.in","r",stdin);
    29     freopen("chessboard.out","w",stdout);
    30     scanf("%d%d",&n,&m);
    31     rep(i,1,n-1) scanf("%d%d",&u,&v),add(u+1,v+1),add(v+1,u+1);
    32     dfs(1,0);
    33     rep(i,0,m) ans=max(ans,g[1][i]);
    34     printf("%d
    ",ans);
    35     return 0;
    36 }

    贪心:

    #include<cstdio>
    #include<algorithm>
    #define rep(i,l,r) for (int i=l; i<=r; i++)
    using namespace std;
    
    const int N=107;
    int n,m,u,v,ans,to[N<<1],nxt[N<<1],h[N],cnt,dep[N];
    void add(int u,int v){ to[++cnt]=v; nxt[cnt]=h[u]; h[u]=cnt; }
    
    void dfs(int x,int fa){
        if (dep[x]>m) return;
        ans=max(ans,(dep[x]+m+2)>>1);
        for (int i=h[x],k; i; i=nxt[i]) if ((k=to[i])!=fa) dep[k]=dep[x]+1,dfs(k,x);
    }
    
    int main(){
        freopen("chessboard.in","r",stdin);
        freopen("chessboard.out","w",stdout);
        scanf("%d%d",&n,&m);
        rep(i,2,n) scanf("%d%d",&u,&v),u++,v++,add(u,v),add(v,u);
        dfs(1,0); printf("%d
    ",min(ans,n));
        return 0;
    }
  • 相关阅读:
    SVM理论之最优超平面
    回归系列之L1和L2正则化
    Logistic回归明明称呼为回归但为什么是分类算法?
    LTE无线接入三层协议体系结构
    80 道大厂算法高频面试题
    linux内核调试技术之printk
    计算机视觉岗常见面试题
    偏差(Bias)和方差(Variance)——机器学习中的模型选择
    正态分布x/y轴
    Python time strftime()方法
  • 原文地址:https://www.cnblogs.com/HocRiser/p/8692238.html
Copyright © 2020-2023  润新知