在办公楼的电梯里高层期间,每层都有人上下。电梯呢只要每层有一个人都有停一次。这样比较麻烦,对于比较低的楼层(6层),每次电梯从一层往上走时,我们只允许电梯停在其中的某一层,然后所有的乘客爬楼梯到自己的目的层。在一楼时,每一位乘客选择自己的楼层,电梯根据每层的人数来计算出应该停的楼层.
问:电梯停在哪一层,能够保证这次乘坐电梯的所以乘客爬楼梯层数之和最少.
分析与解法:
该问题本质是一个优化问题。首先为这个问题找一个合适的抽象模型。从问题中可以看出,有两个因素会影响到最后的结果:乘客的数目及需要停的目的楼层。因此我们可以统计到达各层的乘客数目开始分析。
假设楼层总共有N层,电梯停在第X层,要去第i层的乘客数目总数为Tot[i],这样,所以爬楼梯的总数就是∑Ni=1 {Tot[i]*|i-x|}的值最少.
解法一:
首选考虑简单解法.
可以从第1层开始枚举x一直到N层,然后在计算出如果电梯在x层停的话,所以乘客总共有爬多少层。这是最直接的一种方法。
这个需要一个双重循环来完成计算:时间复杂度为O(N2)
解法二:
进一步分析,假设电梯停在第i层,显然我们可以计算出所有乘客总共要爬的楼梯层数Y。如果有N1个乘客目的楼层在第i层楼以下,有N2个乘客在第i层楼,还有N3个乘客在第i层楼以上。这个时候,如果改停在i-1层楼,所有目的地在第i层及以上的乘客都需要多爬1层,总共需要多爬N2+N3层,而所有目的地在i-1层及以下的乘客可以少爬1层,总共可以少爬N1层。因此,乘客总共需要爬的层数为Y-N1+(N2+N3)=Y-(N1-N2-N3)层。
反这,如果电梯在i+1层,那么乘客总共需要爬的层数为Y+(N1+N2-N3)层。由此可见,当N1>N2+N3时,电梯在第i-1层停更好,乘客走的楼层数减少(N1-N2-N3);
而当N1+N2>N3时,电梯停在i+1层停更好;其他情况下,电梯停在第i层最好。
根据这个规律,我们从第一层开始考察,计算各乘客需要爬楼梯的数目。然后根据上面的策略进行调整,直到找到最佳楼层。总的时间复杂度降为O(N)
package com.floor;
/***
* 电梯优 化方法
*
* */
public class Floor {
public static void main(String[] args) {
getMinFloors(4, 1);
betterFloors();
}
// 双层循环法,计算所以乘客要爬多少层楼梯。。时间复杂度为o(N^2)
public static void getMinFloors(int nTargetFloor, int nMinFloor) {
// 以下是要去往电梯各层的人数
int[] persons = { 0, 2, 0, 1, 3, 5, 8, 4, 6, 0 };
int N = 9;// 电梯一共有多少层
for (int i = 1; i <= N; i++) {
int nFloor = 0;
for (int j = 1; j < i; j++) {
nFloor += persons[j] * (i - j);// 计算往下爬的楼层数
}
for (int j = i + 1; j <= N; j++) {
nFloor += persons[j] * (j - i);// 计算往上爬的楼层数
}
if (nTargetFloor == -1 || nMinFloor > nFloor) {
nMinFloor = nFloor;
nTargetFloor = i;
}
System.out.println("如果停在第" + i + "层需要爬" + nFloor + ":趟电梯");
}
}
// 更优方法
public static void betterFloors() {
// 以下是要去往电梯各层的人数
int[] persons = { 0, 2, 0, 1, 3, 5, 8, 4, 6, 0 };
int N = 9;// 电梯一共有多少层
int nTargetFloor = 1;
int nMinFloor = 0;
int N1 = 0, N2 = persons[1], N3 = 0;
for (int i = 2; i <= N; i++) {
N3 += persons[i];
nMinFloor += persons[i] * (i - 1);//先计算出所以应该要爬的楼层总和
}
for (int i = 2; i <= N; i++) {
if (N1 + N2 < N3) {
nTargetFloor = i;
nMinFloor += (N1 + N2 - N3);
N1 += N2;
N2 = persons[i];
N3 -= persons[i];
} else {
break;
}
}
System.out.println("优化方法结果:停在第" + nTargetFloor + "层要爬:" + nMinFloor
+ "次");
}
}
<!--EndFragment-->