给定一段文字,已知单词a1, a2, …, an出现的频率分别t1, t2, …, tn。可以用01串给这些单词编码,即将每个单词与一个01串对应,使得任何一个单词的编码(对应的01串)不是另一个单词编码的前缀,这种编码称为前缀码。
使用前缀码编码一段文字是指将这段文字中的每个单词依次对应到其编码。一段文字经过前缀编码后的长度为:
L = a1的编码长度 × t1 + a2的编码长度 × t2 + … + an的编码长度 × tn。
定义一个前缀编码为字典序编码,指对于1 ≤ i < n,ai的编码(对应的01串)的字典序在ai+1编码之前,即a1,a2, …, an的编码是按字典序升序排列的。
例如,文字E A E C D E B C C E C B D B E中, 5个单词A、B、C、D、E出现的频率分别为1, 3, 4, 2, 5,则一种可行的编码方案是A:000, B:001, C:01, D:10, E:11,对应的编码后的01串为1100011011011001010111010011000111,对应的长度L为3×1+3×3+2×4+2×2+2×5=34。
在这个例子中,如果使用哈夫曼(Huffman)编码,对应的编码方案是A:000, B:01, C:10, D:001, E:11,虽然最终文字编码后的总长度只有33,但是这个编码不满足字典序编码的性质,比如C的编码的字典序不在D的编码之前。
在这个例子中,有些人可能会想的另一个字典序编码是A:000, B:001, C:010, D:011, E:1,编码后的文字长度为35。
请找出一个字典序编码,使得文字经过编码后的长度L最小。在输出时,你只需要输出最小的长度L,而不需要输出具体的方案。在上面的例子中,最小的长度L为34。
显然,不能对已给出的频率进行排序了,因为一旦这样,就不能满足编码字典序的要求。
正如网上的很多题解所说,这是一道DP问题。
dp[i][j]表示:对[i,j]这一段上的字符进行合并,从而合并成一棵二叉树,进而得到编码,这段文字编码后的长度(并且这种合并方案是[i,j]上进行的所有合并方案中,对应的长度最小的)。
sum[i][j]表示:freq[i] + freq[i+1] + …… + freq[j];
#include<cstdio>
#include<cstring>
#include<algorithm>
#define MAXN 1005
#define INF 0x7fffffff
using namespace std;
typedef long long ll;
ll dp[MAXN][MAXN],sum[MAXN];
//状态转移方程:dp[i][j]=min(dp[i][k],dp[k+1][j]) + sum[i][j] ; (k= i to j-1)
int n;
int main()
{
scanf("%d",&n);
sum[]=;
for(int i=;i<=n;i++)
{
ll tmp;
scanf("%lld",&tmp);
sum[i]=sum[i-]+tmp;
}
/*
使用三重循环
最外重为每次归并的堆数的循环,从2到n;
第二重为dp[i][j]中的i的循环;
最内重为 k= i to j-1 的循环;
*/
for(int i=;i<=n;i++) dp[i][i]=; //初始化dp[i][i]
for(int v=;v<=n;v++)
{
for(int i=;i<=n-(v-);i++)
{
int j=i+(v-);
dp[i][j]=INF;
for(int k=i;k<=j-;k++)
{
dp[i][j]=min(dp[i][j],dp[i][k]+dp[k+][j]);
}
dp[i][j]+=sum[j]-sum[i-];
}
}
if(n==) printf("%d\n",sum[]);
else printf("%lld\n",dp[][n]);
}