给定n本书和m个学生中的书页数。书以页数的升序排列。每个学生都被分配去读一些连续的书。任务是分配图书,以使分配给学生的最大页数最少。
例子 :
Input : pages[] = {12, 34, 67, 90}
m = 2
Output : 113
Explanation:
There are 2 number of students. Books can be distributed
in following fashion :
1) [12] and [34, 67, 90]
Max number of pages is allocated to student
2 with 34 + 67 + 90 = 191 pages
2) [12, 34] and [67, 90]
Max number of pages is allocated to student
2 with 67 + 90 = 157 pages
3) [12, 34, 67] and [90]
Max number of pages is allocated to student
1 with 12 + 34 + 67 = 113 pages
Of the 3 cases, Option 3 has the minimum pages = 113.
这个想法是使用二进制搜索。我们将页数的值固定为当前最小和最大的中间值。我们将最小值和最大值分别初始化为0和所有页面的总和。如果当前的中点可以解决,那么我们在下半部分进行搜索,否则在上半部分进行搜索。
现在出现了问题,如何检查中间值是否可行?基本上,我们需要检查是否可以以最大人数不超过当前人数的方式将页面分配给所有学生。为此,我们按顺序为每个学生分配页面,而当前分配的页面数不超过该值。在此过程中,如果学生人数超过m,则该解决方案不可行。其他可行。
以下是上述想法的实现。
C++
// C++ program for optimal allocation of pages
#include
using namespace std;
// Utility function to check if current minimum value
// is feasible or not.
bool isPossible(int arr[], int n, int m, int curr_min)
{
int studentsRequired = 1;
int curr_sum = 0;
// iterate over all books
for (int i = 0; i < n; i++)
{
// check if current number of pages are greater
// than curr_min that means we will get the result
// after mid no. of pages
if (arr[i] > curr_min)
return false;
// count how many students are required
// to distribute curr_min pages
if (curr_sum + arr[i] > curr_min)
{
// increment student count
studentsRequired++;
// update curr_sum
curr_sum = arr[i];
// if students required becomes greater
// than given no. of students,return false
if (studentsRequired > m)
return false;
}
// else update curr_sum
else
curr_sum += arr[i];
}
return true;
}
// function to find minimum pages
int findPages(int arr[], int n, int m)
{
long long sum = 0;
// return -1 if no. of books is less than
// no. of students
if (n < m)
return -1;
// Count total number of pages
for (int i = 0; i < n; i++)
sum += arr[i];
// initialize start as 0 pages and end as
// total pages
int start = 0, end = sum;
int result = INT_MAX;
// traverse until start <= end
while (start <= end)
{
// check if it is possible to distribute
// books by using mid as current minimum
int mid = (start + end) / 2;
if (isPossible(arr, n, m, mid))
{
// if yes then find the minimum distribution
result = min(result, mid);
// as we are finding minimum and books
// are sorted so reduce end = mid -1
// that means
end = mid - 1;
}
else
// if not possible means pages should be
// increased so update start = mid + 1
start = mid + 1;
}
// at-last return minimum no. of pages
return result;
}
// Drivers code
int main()
{
//Number of pages in books
int arr[] = {12, 34, 67, 90};
int n = sizeof arr / sizeof arr[0];
int m = 2; //No. of students
cout << "Minimum number of pages = "
<< findPages(arr, n, m) << endl;
return 0;
}
Java
// Java program for optimal allocation of pages
public class GFG
{
// Utility method to check if current minimum value
// is feasible or not.
static boolean isPossible(int arr[], int n, int m, int curr_min)
{
int studentsRequired = 1;
int curr_sum = 0;
// iterate over all books
for (int i = 0; i < n; i++)
{
// check if current number of pages are greater
// than curr_min that means we will get the result
// after mid no. of pages
if (arr[i] > curr_min)
return false;
// count how many students are required
// to distribute curr_min pages
if (curr_sum + arr[i] > curr_min)
{
// increment student count
studentsRequired++;
// update curr_sum
curr_sum = arr[i];
// if students required becomes greater
// than given no. of students,return false
if (studentsRequired > m)
return false;
}
// else update curr_sum
else
curr_sum += arr[i];
}
return true;
}
// method to find minimum pages
static int findPages(int arr[], int n, int m)
{
long sum = 0;
// return -1 if no. of books is less than
// no. of students
if (n < m)
return -1;
// Count total number of pages
for (int i = 0; i < n; i++)
sum += arr[i];
// initialize start as 0 pages and end as
// total pages
int start = 0, end = (int) sum;
int result = Integer.MAX_VALUE;
// traverse until start <= end
while (start <= end)
{
// check if it is possible to distribute
// books by using mid is current minimum
int mid = (start + end) / 2;
if (isPossible(arr, n, m, mid))
{
// if yes then find the minimum distribution
result = Math.min(result, mid);
// as we are finding minimum and books
// are sorted so reduce end = mid -1
// that means
end = mid - 1;
}
else
// if not possible means pages should be
// increased so update start = mid + 1
start = mid + 1;
}
// at-last return minimum no. of pages
return result;
}
// Driver Method
public static void main(String[] args)
{
//Number of pages in books
int arr[] = {12, 34, 67, 90};
int m = 2; //No. of students
System.out.println("Minimum number of pages = " +
findPages(arr, arr.length, m));
}
}
Python3
# Python3 program for optimal allocation of pages
# Utility function to check if
# current minimum value is feasible or not.
def isPossible(arr, n, m, curr_min):
studentsRequired = 1
curr_sum = 0
# iterate over all books
for i in range(n):
# check if current number of pages are
# greater than curr_min that means
# we will get the result after
# mid no. of pages
if (arr[i] > curr_min):
return False
# count how many students are required
# to distribute curr_min pages
if (curr_sum + arr[i] > curr_min):
# increment student count
studentsRequired += 1
# update curr_sum
curr_sum = arr[i]
# if students required becomes greater
# than given no. of students, return False
if (studentsRequired > m):
return False
# else update curr_sum
else:
curr_sum += arr[i]
return True
# function to find minimum pages
def findPages(arr, n, m):
sum = 0
# return -1 if no. of books is
# less than no. of students
if (n < m):
return -1
# Count total number of pages
for i in range(n):
sum += arr[i]
# initialize start as 0 pages and
# end as total pages
start, end = 0, sum
result = 10**9
# traverse until start <= end
while (start <= end):
# check if it is possible to distribute
# books by using mid as current minimum
mid = (start + end) // 2
if (isPossible(arr, n, m, mid)):
# if yes then find the minimum distribution
result = min(result, mid)
# as we are finding minimum and books
# are sorted so reduce end = mid -1
# that means
end = mid - 1
else:
# if not possible means pages should be
# increased so update start = mid + 1
start = mid + 1
# at-last return minimum no. of pages
return result
# Driver Code
# Number of pages in books
arr = [12, 34, 67, 90]
n = len(arr)
m = 2 # No. of students
print("Minimum number of pages = ",
findPages(arr, n, m))
# This code is contributed by Mohit Kumar
C#
// C# program for optimal
// allocation of pages
using System;
class GFG
{
// Utility function to check
// if current minimum value
// is feasible or not.
static bool isPossible(int []arr, int n,
int m, int curr_min)
{
int studentsRequired = 1;
int curr_sum = 0;
// iterate over all books
for (int i = 0; i < n; i++)
{
// check if current number of
// pages are greater than curr_min
// that means we will get the
// result after mid no. of pages
if (arr[i] > curr_min)
return false;
// count how many students
// are required to distribute
// curr_min pages
if (curr_sum + arr[i] > curr_min)
{
// increment student count
studentsRequired++;
// update curr_sum
curr_sum = arr[i];
// if students required becomes
// greater than given no. of
// students, return false
if (studentsRequired > m)
return false;
}
// else update curr_sum
else
curr_sum += arr[i];
}
return true;
}
// function to find minimum pages
static int findPages(int []arr,
int n, int m)
{
long sum = 0;
// return -1 if no. of books
// is less than no. of students
if (n < m)
return -1;
// Count total number of pages
for (int i = 0; i < n; i++)
sum += arr[i];
// initialize start as 0 pages
// and end as total pages
int start = 0, end = (int)sum;
int result = int.MaxValue;
// traverse until start <= end
while (start <= end)
{
// check if it is possible to
// distribute books by using
// mid as current minimum
int mid = (start + end) / 2;
if (isPossible(arr, n, m, mid))
{
// if yes then find the
// minimum distribution
result = Math.Min(result, mid);
// as we are finding minimum and
// books are sorted so reduce
// end = mid -1 that means
end = mid - 1;
}
else
// if not possible means pages
// should be increased so update
// start = mid + 1
start = mid + 1;
}
// at-last return
// minimum no. of pages
return result;
}
// Drivers code
static public void Main ()
{
//Number of pages in books
int []arr = {12, 34, 67, 90};
int n = arr.Length;
int m = 2; // No. of students
Console.WriteLine("Minimum number of pages = " +
findPages(arr, n, m));
}
}
// This code is contributed by anuj_67.
PHP
$curr_min)
return false;
// count how many students are
// required to distribute
// curr_min pages
if ($curr_sum + $arr[$i] > $curr_min)
{
// increment student count
$studentsRequired++;
// update curr_sum
$curr_sum = $arr[$i];
// if students required becomes
// greater than given no. of
// students, return false
if ($studentsRequired > $m)
return false;
}
// else update curr_sum
else
$curr_sum += $arr[$i];
}
return true;
}
// function to find minimum pages
function findPages($arr, $n, $m)
{
$sum = 0;
// return -1 if no. of books is
// less than no. of students
if ($n < $m)
return -1;
// Count total number of pages
for ($i = 0; $i < $n; $i++)
$sum += $arr[$i];
// initialize start as 0 pages
// and end as total pages
$start = 0;
$end = $sum;
$result = PHP_INT_MAX;
// traverse until start <= end
while ($start <= $end)
{
// check if it is possible to
// distribute books by using
// mid as current minimum
$mid = (int)($start + $end) / 2;
if (isPossible($arr, $n, $m, $mid))
{
// if yes then find the minimum
// distribution
$result = min($result, $mid);
// as we are finding minimum and
// books are sorted so reduce
// end = mid -1 that means
$end = $mid - 1;
}
else
// if not possible means pages
// should be increased so update
// start = mid + 1
$start = $mid + 1;
}
// at-last return minimum
// no. of pages
return $result;
}
// Driver code
// Number of pages in books
$arr = array(12, 34, 67, 90);
$n = count($arr);
$m = 2; // No. of students
echo "Minimum number of pages = ",
findPages($arr, $n, $m), "\n";
// This code is contributed by Sach_Code
?>
输出 :
Minimum number of pages = 113