HOJ-1005

这题一看就像是动归,但我就是想不出来怎么动归。。。

纠结了n天之后,果断百度,果然是动归

a[i][j]表示前i个饭店如果需要j个depot

c[i][j]表示如果从第i个到第j个饭店都由一个depot负责,的最短距离

于是就有了神奇的动态转移方程

  a[i][j] = min{a[i][j], a[i-k][j-1]+c[i-k+1][i]} (k = 1, 2, …, i-1)

嗯,然后就益母聊染了

 1 #include<cstdio>
 2 #include<cstdlib>
 3 #include<cstring>
 4
 5 #define min(a,b) (((a)>(b))?(b):(a))
 6
 7 #define MAXN 300
 8
 9 int d[MAXN];//distance between restautants i and headquarter
10 int a[MAXN][MAXN];//the lowest distance while the first i restautants need j depots
11 int c[MAXN][MAXN];//the lowest distance while restautant i to restaurant j need 1 depots
12
13 int main(void)
14 {
15   int n, k;
16   int num = 1;
17   scanf("%d%d", &n, &k);
18   while(n > 0 && k > 0){
19     for(int i = 1; i <= n; ++i) scanf("%d", &d[i]);
20     memset(c, 0, sizeof(c));
21     for(int i = 1; i <= n; ++i){
22       for(int j = i+1; j <= n; ++j)
23     for(int k = i, p = (i+j)/2; k <= j; ++k)
24       c[i][j] += abs(d[k] - d[p]);
25     }
26     for(int i = 0; i <= n; ++i)
27       for(int j = 0; j <= n; ++j)
28     a[i][j] = 0xFFFFFFF;
29     a[1][1] = 0;
30     for(int i = 2; i <= n; ++i){
31       a[i][1] = c[1][i];
32       for(int j = 2; j <= i && j <= k; ++j)
33     for(int k = 1; k < i; ++k)
34       if(a[i][j] > a[i-k][j-1]+c[i-k+1][i]){
35         a[i][j] = a[i-k][j-1]+c[i-k+1][i];
36         /*for(int i = 0; i < n; ++i) printf("%12d", i);
37         printf("\n");
38         for(int i = 0; i <= n; ++i){
39           printf("%2d: ", i);
40           for(int j = 0; j <= n; ++j) printf("%12d", a[i][j]);
41           printf("\n");
42         }*/
43         //printf("a[%d][%d] = a[%d-%d][%d-1] + c[%d+1][%d]\n", i, j, i, k, j, k, i);
44         //printf("%d = %d + %d\n", a[i][j], a[i-k][j-1], c[k+1][i]);
45       }
46     }
47     printf("Chain %d\n", num++);
48     printf("Total distance sum = %d\n\n", a[n][k]);
49
50     scanf("%d%d", &n, &k);
51   }
52   return 0;
53 }
时间: 2024-10-16 14:04:59

HOJ-1005的相关文章

HOJ 题目分类

转自:http://blog.sina.com.cn/s/blog_65f3869301011a1o.html ******************************************************************************* 简单题(包括枚举,二分查找,(复杂)模拟,基础数据结构(栈.队列),杂题等 ****************************************************************************

CSUFT 1005 Coffin Tiles

1005: Coffin Tiles Time Limit: 1 Sec      Memory Limit: 128 MB Submit: 2      Solved: 2 Description The Pumpkin King has a great idea for this Christmas: Personalized coffins for all the good little boys and girls! To make them extra special, Jack ha

PAT 乙级 1005 继续(3n+1)猜想 (25) C++版

1005. 继续(3n+1)猜想 (25) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 作者 CHEN, Yue 卡拉兹(Callatz)猜想已经在1001中给出了描述.在这个题目里,情况稍微有些复杂. 当我们验证卡拉兹猜想的时候,为了避免重复计算,可以记录下递推过程中遇到的每一个数.例如对n=3进行验证的时候,我们需要计算3.5.8.4.2.1,则当我们对n=5.8.4.2进行验证的时候,就可以直接判定卡拉兹猜想的真伪,而不需要重

杭电女生赛1001 1002 1003 1005 1008 hdu6023 6024 6025 6027 6030

代码先贴这里 #include "iostream" #include "string.h" #include "stack" #include "queue" #include "string" #include "vector" #include "set" #include "map" #include "algorithm&quo

HDU - 1005 Number Sequence(简单矩阵快速幂)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1005 题意:f(1) = 1, f(2) = 1, f(n) = (A * f(n - 1) + B * f(n - 2)) mod 7. 就是这道题目,然而找了一晚上的错误 \("▔□▔)/\("▔□▔)/\("▔□▔)/. 1 #include <iostream> 2 #include <cstring> 3 using namespace std;

1005 生日礼物

1005 生日礼物 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题目描述 Description 9月12日是小松的朋友小寒的生日.小松知道小寒特别喜欢蝴蝶,所以决定折蝴蝶作为给小寒的生日礼物.他来到了PK大学最大的一家地下超市,在超市里,小松找到了n种可以用来折纸的本子.每种类型的本子里有若干不同颜色的纸若干张,当然同种类型的本子一定是完全一样的,而不同种类型的本子不一定完全不一样.他统计了一下,这里总共有n种不同类型的可以用来折纸的本子,每种本子各有bi

PAT甲级1005 Spell It Right

题目:PAT甲级 1005 题解:水题.看到题目的第一时间就在想一位一位的mod,最后一加一转换就完事了.结果看到了N最大为10的100的次方,吓得我赶紧放弃这个想法... 发现碰到这种情况用字符串十分好用,这道题应该考察的就是这一点.大致思路就是把数字的每一位放到字符串中,然后通过ASCII码得到每一位的相加结果num,然后把num一位一位的放到stack中,使用stack是因为它先进先出的特性,最后输出就行了. 代码: 1 #include<cstdio> 2 #include<qu

HOJ 1797 Red and Black

传送门  http://acm.hit.edu.cn/hoj/problem/view?id=1797 总体的思路是遍历可以到达的' . ',将其对应的vis数组化为1,然后统计所有为1的vis项; ①常用的加边法,防止越界 ②初始化,不然两次相同的输入得到的结果会不同,由于是二维数组,能力有限,只好在结尾初始化,为下次输入做准备 ③结束条件,递归函数一定要先写好结束条件,不然会炸 ④上下左右四向移动 ⑤直接递归函数调用自身 1 #include <stdio.h> 2 #include &l

hdu 1005

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1005 思路:找规律题 1 #include<stdio.h> 2 main() 3 { 4 int s[100]; 5 int a,b,n,i; 6 while(scanf("%d%d%d",&a,&b,&n)!=EOF&&(a||b||n)) 7 { 8 s[1]=1; 9 s[2]=1; 10 for(i=3;i<=49;i++

hoj 2662 状态压缩dp

题意: 给定mxn的棋盘,要往其中放d枚棋子,其中,一枚棋子的上下左右四个位置不能再放棋子.求所有合法的放置状态 ( 0<m,n<=80 , mxn<=80 ) 棋盘位置可多达80,爆搜目测超时,我用的状态压缩dp. 棋盘中的一个格子,放棋用1,不放用0 压缩其中一维,每个状态数用一个二进制数表示. 每个状态都是棋盘一行的放置方法 假设,m >= n , 这个递推棋盘就有 m行 ,每行至多有 2^n-1个状态数.其实,没有这么多!按照题意,每个1的上下左右不能有1.对于一行,只知左