标签归档:dynamic programming

小胖守皇宫(Tree dynamic)

问题描述
huyichen世子事件后,xuzhenyi成了皇上特聘的御前一品侍卫。

皇宫以午门为起点,直到后宫嫔妃们的寝宫,呈一棵树的形状;某些宫殿间可以互相望见。大内保卫森严,三步一岗,五步一哨,每个宫殿都要有人全天候看守,在不同的宫殿安排看守所需的费用不同。
可是xuzhenyi手上的经费不足,无论如何也没法在每个宫殿都安置留守侍卫。
帮助xuzhenyi布置侍卫,在看守全部宫殿的前提下,使得花费的经费最少。

输入格式 Input Format

输入文件中数据表示一棵树,描述如下:
第1行 n,表示树中结点的数目。
第2行至第n+1行,每行描述每个宫殿结点信息,依次为:该宫殿结点标号i(0 对于一个n(0 < n <= 1500)个结点的树,结点标号在1到n之间,且标号不重复。
输出格式 Output Format
输出文件仅包含一个数,为所求的最少的经费。

样例输入 Sample Input
6
1 30 3 2 3 4
2 16 2 5 6
3 5 0
4 4 0
5 11 0
6 5 0

样例输出 Sample Output

25

注释 Hint

如图

算法分析
树型动态规划
本题既可以用有向树来做,也可以用无向图随机选根结点

分析可见,对于当前结点i,它有num个子结点,它有它有3种状态:

1:在当前位放置守卫的情况(被自己所控制)
2:不在当前位放置守卫,但是它已经被它的子结点所控制
3:不在当前位放置守卫,它还没有被它的子结点所影响(即被其父结点控制)

用f[i,x]表示结点i的第x种情况:

1情况的值由其子结点的1,2,3情况得到最优
2情况的值由其子结点的1,2情况得到最优
3情况的值只可由其2情况求和.

第2种情况要特别注意,要求它的子结点中必须有一个是1状况的,所以可以先将最小值求和,
然后加上这num个子结点中每个的1情况与最优情况的f值之差–most

方程:
f[i,1]:=Σ(min{f[son[j],1],f[son[j],2],f[son[j],3]})+a[i]
f[i,2]:=Σ(min{f[son[j],1],f[son[j],2]})+most
f[i,3]:=Σ(f[son[j],2]),           1<=j<=num

还有要注意的一点就是对于极限数据,树被拉成了一条链.
有向图解法如果每次递归都开一个数组那么内存会爆,对此,可以不用数组记录子结点,而直接用链表就
可以大大节约空间了.甚至把数组改成全局变量也是可行的,就是注意递归调用要在DP之前.

【代码】

#include 
#define MAXN 1501
#define MAXINT 2000000000
typedef struct
{
        int v, num;
        int child[MAXN];
}node;
 
node a[MAXN];
int f[MAXN][4];
int min2(int x, int y)
{
       return x

乐队

问题描述
你刚刚继承了流行的“破锣摇滚”乐队录制的尚未发表的N(1 <= N <= 600)首歌的版权。你打算从中精选一些歌曲,发行M(1 <= M <= 600)张CD。 每一张CD最多可以容纳T(1 <= T <= 10000)分钟的音乐,一首歌不能分装在 两张CD中。 不巧你是一位古典音乐迷,不懂如何判定这些歌的艺术价值。于是你决定根据以下标准进行选择: · 歌曲必须按照创作的时间顺序在CD盘上出现。 · 选中的歌曲数目尽可能地多。 Input

第一行: 三个整数:N, T, M.;

第二行: N个整数,分别表示每首歌的长度,按创作时间顺序排列。

Output

一个整数,表示可以装进M张CD盘的乐曲的最大数目。

Sample Input

4 5 2

4 3 4 2

Sample Output

3

算法分析
动态规划。

定义数组f[MAXN]。f[i]表示前i张光盘可以装下的音乐的最小长度

代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
#include <stdio.h>
#include <string.h>
 
#define MAXN 605
#define inf 1000000001
int main()
{
 
       int f[MAXN]; //f[i]表示前i张光盘可以装下的音乐的长度
       int n, m, i, j, temp, max, ans, t, s;
 
       while (scanf("%d%d%d", &n, &t, &m) != EOF)
       {
              max = 0;
              ans = 0;
              m = m * t;
              memset(f, 0, sizeof(f));
 
              for (i = 0; i < n; i++)     //对于每首歌曲
              {
                     scanf("%d", &temp);
                     if (temp <= t) //当一首歌曲的长度大于光盘的长度时,这首歌曲就不能用了.
                     {
                            max++;
                            f[max] = inf;
 
                            for (j = max; j >= 1; j--)
                            {
                                   if (t - f[j - 1] % t < temp)    //如果这张光盘无法装下这首歌曲,那么就添加一张光盘  
                                          s = (f[j - 1] / t  + 1) * t + temp;
                                   else                                            //否则直接放在这张光盘上
                                          s = f[j - 1] + temp;
 
                                   if (s < f[j])
                                          f[j] = s;
 
                                   if (f[j] <= m && j > ans)             //更新结果
                                          ans = j;
                            }
                     }
              }
 
              printf("%d\n", ans);
       }
 
       return 0;
}

方法2:

算法分析:

首先把一盘CD装不下的歌全部踢掉!!!

剩下的工作就是DP了。用f[i,j]表示在前i首歌中选出j首在CD上占的最小时间。这里说的时间包括除最后一盘外的CD上浪费的时间。f[i,j]=min{f[i-1,j],sum(f[i-1,j-1],第i首歌的长度)}。这里的sum的含义是:如果最后一盘CD剩下的时间正好可以放下第i首歌,那么直接相加即可,否则要再加上最后一盘CD剩下的时间(这些时间已被浪费了)。找一个最大的j使f[n,j]<=t*m,这个j就是答案。 【代码】

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
#include <stdio.h>
 
#define MAXN 605
#define inf 1000000001
#define min(a,b) a<b?a:b
int a[MAXN], f[MAXN][MAXN];
int main()
{
 
       int n, m, i, j, k, s, t, ans, temp;
       while (scanf("%d%d%d", &n, &t, &m) != EOF)
       {
              s = 0;
              for (i = 1; i <= n; i++)
              {
                     scanf("%d", &temp);
                     if (temp <= t)
                            a[++s] = temp;
              }
 
              for (i = 0; i <= s; i++)   //init
              {
                     for (j = 0; j <= s; j++)
                            f[i][j] = inf;
              }
 
              f[0][0] = 0;
 
              for (i = 1; i <= s; i++)   //for everyone
              {
                     for (j = 1; j <= i; j++)
                     {
                            if (t - f[i - 1][j - 1] % t < a[i])      //if the last one is not enough for a[i]
                                   k = (f[i - 1][j - 1] / t + 1) * t + a[i];
                            else
                                   k = f[i - 1][j - 1] + a[i];
 
                            f[i][j] = min(f[i - 1][j], k);    //select the min one
                     }
              }
 
              ans = 0;
 
              for (i = 1; i <= s; i++)          //search the ans
              {
                     if (f[s][i] <= t * m && i > ans)
                            ans = i;
              }
 
              printf("%d\n", ans);       //output
       }
 
       return 0;
}

打包

【问题描述】
假如你现在拿到了许多的礼物,你要把这些礼物放进袋子里。你只有一个最多能装下V 体积物品的袋子,你不能全部放进去。因为你拿不动那么重的东西。你估计你能拿的最大重量为 G。现在你了解每一个物品的完美值、重量和体积,你当然想让袋子中装的物品的完美值总和最大,你又得计划一下了。
【输入】
第一行:V 和 G 表示最大重量和体积。
第二行:N 表示拿到 N 件礼物。
第三到N+2行:每行3个数 Ti Vi Gi 表示各礼物的完美值、重量和体积
【输出】
输出共一个数,表示可能获得的最大完美值。
【输入输出样例】
输入(pack.in):
6 5
4
10 2 2
20 3 2
40 4 3
30 3 3
输出(pack.out):
50

【数据范围】
对于20%的数据 N,V,G,Ti,Vi,Gi≤10
对于50%的数据 N,V,G,Ti,Vi,Gi≤100
对于80%的数据 N,V,G,Ti,Vi,Gi≤300
80%到100%的数据是N,V,G,Ti,Vi,Gi≤380 的离散随机数据。

算法描述:动态规划。
看到这道题,首先想到的就是0/1背包问题模型。然而这里有两个限制条件,所以会有一定的困难。转变思想,发现这题同样也可以用装箱问题模型来做。不同的是这里要开一个二维的数F[V][G]。表示V的体积G的重量可以达到的最大完美值。具体做法见程序代码。

【代码】

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
#include <stdio.h>
#include <string.h>
#define max(a,b) a>b?a:b
typedef struct 
{
	int v, t, g;
}node;//用结构体存储每个礼物。
int main()
{
	node a[400];
	int f[400][400];
	int i, j, n, v, g, t, max, k;
	scanf("%d%d", &v, &g);
	scanf("%d", &n);
	for (i = 1; i <= n; i++)
		scanf("%d%d%d", &a[i].t, &a[i].v, &a[i].g);
	memset(f, 0, sizeof(f));
	f[0][0] = 1;
	for (i = 1; i <= n; i++) //对于每个礼物
	{
		for (j = v - a[i].v; j >= 0; j--)//对于总体积小于v的
		{
			for (k = g - a[i].g; k>= 0; k--)//对于总重量小于g的
			{
				if (f[j][k] > 0)//如果存在这种情况,找出最大值。
				{
					f[j + a[i].v][k + a[i].g] = max(f[j + a[i].v][k + a[i].g], f[j][k] + a[i].t);
				}
			}
		}
	}
	max = 0;
	for (j = v; j >= 0; j--)
	{
		for (k = g; k >= 0; k--)
			if (f[j][k] > max)
				max = f[j][k];
	}
	printf("%d\n", max - 1);//由于开始的时候初始化f[0][0]为1,所以这里要减去1.
	return 0;
}