A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© orgcheng 中级黑马   /  2015-9-29 10:04  /  853 人查看  /  1 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

问题详情

有n本书和k个抄写员。要求n本书必须连续地分配给这k个抄写员抄写。也就是说前a1本书分给第一个抄写员,接下来a2本书分给第二个抄写员,如此类推(a1,a2需要你的算法来决定)。给定n,k和每本书的页数p1,p2..pn,假定每个抄写员速度一样(每分钟1页),k个抄写员同时开始抄写,问最少需要多少时间能够将所有书全部抄写完工?(提示:本题有很多种算法可以在不同的时间复杂度下解决,需要尽可能的想到所有的方法)
解答

解法1:动态规划

设f[j]代表前i本书分给j个抄写员抄完的最少耗时。答案就是f[n][k]。状态转移方程f[j] = min{max(f[x][j-1], sum(x+1, i)), j<x<i}。其中x是在枚举第j个抄写员是从哪本书开始抄写。

时间复杂度O(n^2*k)


解法2;动态规划+决策单调。

同上一解法,但在x的枚举上进行优化,设s[j]为使得f[j]获得最优值的x是多少。有s[j+1]>=s[j]>=s[i-1][j]。因此x这一层的枚举不再是每次都是n而是总共加起来n。

时间复杂度O(n*k)


解法3:二分答案

二分答案,然后尝试一本本的加进来,加满了就给一个抄写员。看最后需要的抄写员数目是多余k个还是少于k个,然后来决定是将答案往上调整还是往下调整。

时间复杂度O( n log Sum(pi) )


面试官角度:

该问题的考点在于算法能力。需要一定的算法积累,如对动态规划和二分法的算法积累。面试官不一定会需要你答出所有的解法(根据职位要求和招聘名额来看了),但是你答出多少就能够大概知道你在算法能力上的水平是多少。一般来讲至少需要答出动态规划的解法,因为只要稍微做过一点动态规划的训练,都是可以想出来的。

按解法一的代码如下:有待优化
  1. package com.itheima;

  2. public class Main {
  3.         public static void main(String[] args) {
  4.                 int n = 6;
  5.                 int p = 3;
  6.                 // pages.length == n+1,其中pages[0]是自己添加的
  7.                 int[] pages = { 0, 1, 100, 23, 46, 78, 41 };

  8.                 int[][] dp = new int[n + 1][p + 1];

  9.                 // 初始化-- begin
  10.                 for (int i = 1; i <= n; ++i)
  11.                         for (int j = 1; j <= p; ++j)
  12.                                 dp[i][j] = Integer.MAX_VALUE;
  13.                 for (int i = 1; i <= n; ++i)
  14.                         dp[i][1] = dp[i - 1][1] + pages[i];
  15.                 for (int j = 1; j <= p; ++j)
  16.                         dp[1][j] = pages[1];
  17.                 // 初始化-- end

  18.                 // 辅助数组sum[i]=pages[1]+pages[2]+...+pages[i]
  19.                 int[] sum = new int[n + 1];
  20.                 sum[0] = 0;
  21.                 for (int i = 1; i <= n; ++i)
  22.                         sum[i] = sum[i - 1] + pages[i];
  23.                
  24.                 // 核心计算部分
  25.                 for (int j = 2; j <= p; ++j) { // 枚举抄书员
  26.                         for (int i = 2; i <= n; ++i) { // 枚举书的总数
  27.                                 dp[i][j] = Math.min(dp[i][j], dp[i][j - 1]);
  28.                                 for (int x = j - 1; x < i; ++x) {        // 枚举第j个人处理的书本数 0 ~ i-j
  29.                                         dp[i][j] = Math.min(dp[i][j], Math.max(dp[x][j - 1], sum[i] - sum[x]));
  30.                                 }
  31.                         }
  32.                 }

  33.                 for (int i = 1; i <= n; ++i) {
  34.                         for (int j = 1; j <= p; ++j) {
  35.                                 System.out.print(dp[i][j] + "\t");
  36.                         }
  37.                         System.out.println();
  38.                 }
  39.                 System.out.println("结果为" + dp[n][p]);
  40.         }
  41. }
复制代码

1 个回复

倒序浏览
这字体,我看着也是晕了
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马