挺复杂的一题,用了多个dp数组来保存过程量
Given a sequence of words, and a limit on the number of characters that can be put in one line (line width). Put line breaks in the given sequence such that the lines are printed neatly. Assume that the length of each word is smaller than the line width.
The word processors like MS Word do task of placing line breaks. The idea is to have balanced lines. In other words, not have few lines with lots of extra spaces and some lines with small amount of extra spaces.
The extra spaces includes spaces put at the end of every line except the last one. The problem is to minimize the following total cost. Cost of a line = (Number of extra spaces in the line)^3 Total Cost = Sum of costs for all lines For example, consider the following string and line width M = 15 "Geeks for Geeks presents word wrap problem" Following is the optimized arrangement of words in 3 lines Geeks for Geeks presents word wrap problem The total extra spaces in line 1, line 2 and line 3 are 0, 2 and 3 respectively. So optimal value of total cost is 0 + 2*2 + 3*3 = 13
Please note that the total cost function is not sum of extra spaces, but sum of cubes (or square is also used) of extra spaces. The idea behind this cost function is to balance the spaces among lines. For example, consider the following two arrangement of same set of words:
1) There are 3 lines. One line has 3 extra spaces and all other lines have 0 extra spaces. Total extra spaces = 3 + 0 + 0 = 3. Total cost = 3*3*3 + 0*0*0 + 0*0*0 = 27.
2) There are 3 lines. Each of the 3 lines has one extra space. Total extra spaces = 1 + 1 + 1 = 3. Total cost = 1*1*1 + 1*1*1 + 1*1*1 = 3.
Total extra spaces are 3 in both scenarios, but second arrangement should be preferred because extra spaces are balanced in all three lines. The cost function with cubic sum serves the purpose because the value of total cost in second scenario is less.
Method 1 (Greedy Solution)
The greedy solution is to place as many words as possible in the first line. Then do the same thing for the second line and so on until all words are placed. This solution gives optimal solution for many cases, but doesn’t give optimal solution in all cases. For example, consider the following string “aaa bb cc ddddd” and line width as 6. Greedy method will produce following output.
aaa bb cc ddddd
Extra spaces in the above 3 lines are 0, 4 and 1 respectively. So total cost is 0 + 64 + 1 = 65.
But the above solution is not the best solution. Following arrangement has more balanced spaces. Therefore less value of total cost function.
aaa bb cc ddddd
Extra spaces in the above 3 lines are 3, 1 and 1 respectively. So total cost is 27 + 1 + 1 = 29.
Despite being sub-optimal in some cases, the greedy approach is used by many word processors like MS Word and OpenOffice.org Writer.
Method 2 (Dynamic Programming)
The following Dynamic approach strictly follows the algorithm given in solution of Cormen book. First we compute costs of all possible lines in a 2D table lc[][]. The value lc[i][j] indicates the cost to put words from i to j in a single line where i and j are indexes of words in the input sequences. If a sequence of words from i to j cannot fit in a single line, then lc[i][j] is considered infinite (to avoid it from being a part of the solution). Once we have the lc[][] table constructed, we can calculate total cost using following recursive formula. In the following formula, C[j] is the optimized total cost for arranging words from 1 to j.
The above recursion has overlapping subproblem property. For example, the solution of subproblem c(2) is used by c(3), C(4) and so on. So Dynamic Programming is used to store the results of subproblems. The array c[] can be computed from left to right, since each value depends only on earlier values.
To print the output, we keep track of what words go on what lines, we can keep a parallel p array that points to where each c value came from. The last line starts at word p[n] and goes through word n. The previous line starts at word p[p[n]] and goes through word p[n] – 1, etc. The function printSolution() uses p[] to print the solution.
In the below program, input is an array l[] that represents lengths of words in a sequence. The value l[i] indicates length of the ith word (i starts from 1) in theinput sequence.
另外可以参考wiki的这篇:https://zh.wikipedia.org/wiki/%E8%87%AA%E5%8A%A8%E6%8D%A2%E8%A1%8C
在TeX中使用的,则是另一个算法,旨在将行尾空格数的平方最小化,以产生一个更加美观的结果。以上的算法不能完成这一目标,例如:
aaa bb cc ddddd
如果惩罚函数定义为行尾剩余空格数的平方,则贪婪算法会得到一个次优解(为了简化起见,不妨假设采用定宽字体):
------ 一行的宽度为6 aaa bb 剩余的空格数:0,平方=0 cc 剩余的空格数:4,平方=16 ddddd 剩余的空格数:1,平方=1
总计代价17,而最佳的解决方案是这样的:
------ 一行的宽度为6 aaa 剩余空格数:3 平方=9 bb cc 剩余空格数:1 平方=1 ddddd 剩余空格数:1 平方=1
请注意,第一行在bb前断开了,相对于在bb后断开的解法,可以得到更好的右边界和更低的代价11。
为了解决这个问题,我们需要定义一个惩罚函数,用于计算包含单词到单词的一行的代价:
其中通常为或。另外,有一些特殊的情况值得考虑:如果结果为负(即单词串不能全部放在一行里),惩罚函数需要反映跟踪或压缩文本以适应一行的代价;如果这是不可能的,则返回
最优解的代价可以用以下的递归式定义:
这可以利用动态规划来高效地实现,时间和空间复杂度均为
package DP; import java.util.Arrays; public class WordWrap { public static void main(String[] args) { int[] l = {3, 2, 2, 5}; int n = l.length; int M = 6; solveWordWrap(l, n, M); } // l[] represents lengths of different words in input sequence. For example, // l[] = {3, 2, 2, 5} is for a sentence like "aaa bb cc ddddd". n is size of // l[] and M is line width (maximum no. of characters that can fit in a line) // Time Complexity: O(n^2) Auxiliary Space: O(n^2) public static void solveWordWrap(int[] l, int n, int M){ // For simplicity, 1 extra space is used in all below arrays // extras[i][j] will have number of extra spaces if words from i // to j are put in a single line // extras[i][j]记录把单词序号i到j放在一行后剩余的空格数目 int[][] extras = new int[n+1][n+1]; // lc[i][j] will have cost of a line which has words from i to j // lc[i][j]记录把单词序号i到j放在一行后导致的花费cost int[][] lc = new int[n+1][n+1]; // c[i] will have total cost of optimal arrangement of words from 1 to i // c[i]记录了把单词序号从1到i放在多行后导致的总的最小花费 int[] c = new int[n+1]; // p[] is used to print the solution. // p[n]记录了单词编号p[n]到n被放在了同一行 int[] p = new int[n+1]; // calculate extra spaces in a single line. The value extra[i][j] // indicates extra spaces if words from word number i to j are // placed in a single line for(int i=1; i<=n; i++){ extras[i][i] = M-l[i-1]; // 如果只有一个单词,则剩余空格数为M扣除这个单词的长度 for(int j=i+1; j<=n; j++){ // 否则剩余空格数为上一次剩余空格数扣除接下来用的单词长度 extras[i][j] = extras[i][j-1]-l[j-1]-1; // 因为单词之间要有空格隔开,所以多扣除一个1 } } // print(extras); // Calculate line cost corresponding to the above calculated extra // spaces. The value lc[i][j] indicates cost of putting words from // word number i to j in a single line for(int i=1; i<=n; i++){ // 起始单词序号 for(int j=i; j<=n; j++){ // j:结尾单词序号,j肯定大于i,因为j指向的单词在i之后 if(extras[i][j] < 0){ // 空格数为负说明,放不下所以设为最大值 lc[i][j] = Integer.MAX_VALUE; }else if(j==n && extras[i][j]>=0){ // 最后一个单词,后面有空格也无所谓 lc[i][j] = 0; }else{ // 其余情况cost为空格数的平方 lc[i][j] = extras[i][j]*extras[i][j]; } } } // Calculate minimum cost and find minimum cost arrangement. // The value c[j] indicates optimized cost to arrange words // from word number 1 to j. c[0] = 0; for(int j=1; j<=n; j++){ // 依次计算单词数目为j时的最小cost c[j] = Integer.MAX_VALUE; for(int i=1; i<=j; i++){ // i: 把i...j放在一行,然后计算剩下1...i的放法 if(c[i-1]!=Integer.MAX_VALUE && lc[i][j]!=Integer.MAX_VALUE // 防止溢出 && c[i-1]+lc[i][j]<c[j]){ c[j] = c[i-1] + lc[i][j]; // 记录更小的花费 p[j] = i; // 单词数目为j时,最优放法是1...i放在前几行,把i...j放在最后一行 } } } printSolution(p, n); // System.out.println(Arrays.toString(p)); } // 递归返回出最后一行的行号k, 每一行的单词序号总是由p[n]开始到n结尾 public static int printSolution(int[] p, int n){ int k; if(p[n] == 1){ k = 1; }else{ k = printSolution(p, p[n]-1)+1; // 上一行行号加一,p[n]为开始的单词序号,所以p[n]-1就是上一行的结尾单词序号 } System.out.format("Line number %d: From word no. %d to %d \n", k, p[n], n); return k; } public static void print(int[][] A){ for(int i=0; i<A.length; i++){ for(int j=0; j<A[0].length; j++){ System.out.print(A[i][j] + "\t"); } System.out.println(); } } }