zoukankan      html  css  js  c++  java
  • POJ 1155 树形背包

    题意:从一个发射站发射电视,只有叶子节点是用户,收到一部分费用,所有的边都有花费,求在不亏本的情况下,最多可以让多少用户(叶子结点)收看到电视。

    分析:树形背包。

    状态定义: dp(i,j) : 以 i 为根的,让 j 个用户看到电视,最大获益(可以为负数)。那么sz不再是原来的定义了。

    最后遍历 j,第一个不为负数的就是答案。

    状态转移:树形背包,dp(i,j) = max(d(i,j) , dp(i)(k)+dp(son,j-k)-w);

    #include <algorithm>
    #include <vector>
    #include <cstring>
    #include <cstdio>
    
    
    using namespace std;
    
    const int maxn = 3500;
    
    struct Edge
    {
        int u,v,cost;
    };
    
    vector<Edge> G[maxn];
    
    int n,m;
    
    int dp[maxn][maxn],tmp[maxn];
    int sz[maxn];
    
    void dfs(int u)
    {
    
        for(int i=0; i<G[u].size(); i++)
        {
            int v = G[u][i].v;
            int cc = G[u][i].cost;
            dfs(v);
    
            for(int j = 0; j<=sz[u]; j++)
                tmp[j] = dp[u][j];
    
            for(int j=0; j<=sz[u]; j++)
            {
                for(int k=1; k<=sz[v]; k++)
                {
                    dp[u][j+k] = max(dp[u][j+k],tmp[j]+dp[v][k]-cc);
                }
            }
    
            sz[u]+=sz[v];
        }
    }
    
    int main()
    {
        //freopen("in.txt","r",stdin);
        scanf("%d%d",&n,&m);
        for(int i=1; i<=n-m; i++)
        {
            int p;
            scanf("%d",&p);
            for(int j=0; j<p; j++)
            {
                int v,c;
                scanf("%d%d",&v,&c);
                G[i].push_back((Edge)
                {
                    i,v,c
                });
            }
        }
    
        for(int i=1;i<=n;i++)
            for(int j=1;j<=m;j++)
                dp[i][j] = -0x3f3f3f3f;
        memset(sz,0,sizeof(sz));
        for(int i=n-m+1; i<=n; i++)
        {
            int x;
            scanf("%d",&x);
            sz[i] = 1;
            dp[i][1] = x;
        }
    
        dfs(1);
        for(int i = m; i>=0; i--)
        {
            if(dp[1][i]>=0)
            {
                printf("%d
    ",i);
                break;
            }
        }
        return 0;
    }
    View Code
  • 相关阅读:
    android: 记录及回复lisView的位置
    android获取屏幕尺寸、密度
    iphone:蓝牙传输
    android 线程 进程
    android 首次使用app时的使用教程的功能的实现
    android 启动界面
    iphone:数组的反序
    android:onKeyDown
    iphone: 可编辑的tableView Move&Delete
    iphone:类似path的抽屉式导航效果的demo总结
  • 原文地址:https://www.cnblogs.com/TreeDream/p/7266827.html
Copyright © 2011-2022 走看看