松中提高组模拟赛8.24 选数排列 题解

松中提高组模拟赛8.24 选数排列 题解

题目

给出 N N N 个数,我们需要选择其中的 R R R× C C C 个数,,把它们填入一个 R R R× C C C 的矩阵( R R R C C C 列)中。

我们先定义一个函数 D D D( i i i) 代表第 i i i 行中最大的数和最小的数之差。对于整个矩阵,定义 F F F 为矩阵中 D D D( i i i)(1≤ i i i R R R) 的最大值。

我们需要 F F F 的值最少,你能求出最少可能达到的 F F F 值是多少吗?


输入z

第一行给出 3 个整数 N N N, R R R, C C C,对应题目中描述的参数。

接下来一行有 N N N 个整数,代表 N N N 个可以选择的数 P i Pi Pi


输出

输出一行表示最少可能达到的 F F F 值。


样例

input
7 2 3
170 205 225 190 260 225 160

output
30


说明/提示

对于 50% 的数据,1≤ N N N≤1000

对于所有数据,1≤ R R R, C C C≤104, R R R× C C C N N N≤5∗105,0< P i Pi Pi≤109


解题思路

DP 80
二分答案 100
qwq不懂贪心的好,就少了20分(;′⌒`)

二分解法
二分最大值
f[i]意为选到第i个数,可以填满f[i]行
如果满足mid,f[i]=f[i-c+1]+1;
最后f[n]>=r说明最大值还能更小
r左移,否则l右移


代码

DP 80
1 WA
1TLE

#include
#include 
#include
#include
using namespace std;
int n,r,c,a[500010],f[1010][50100],b[500010];
int main()
{
	memset(f,0x7f,sizeof(f));
	scanf("%d%d%d",&n,&r,&c);
	f[1][0]=0;
	for (int i=1;i<=n;i++)
	{
	    f[1][i]=0;
	    scanf("%d",&a[i]);
	}
	sort(a+1,a+n+1);
	for (int i=c;i<=n;i++)
	{
	    b[i]=a[i]-a[i-c+1]; 
        if (f[1][i-1]!=0)
		   f[1][i]=min(f[1][i-1],b[i]);
		   else f[1][i]=b[i];
    }
    for (int i=2;i<=r;i++)
        for (int j=i*c;j<=n;j++)
            f[i][j]=min(f[i][j],min(f[i][j-1],max(b[j],f[i-1][j-c])));
    printf("%d",f[r][n]);
    return 0;
}

二分 100

#include
#include 
#include
#include
using namespace std;
int n,r,c,a[500010],f[500100],b[500010];
bool check(int x)
{
	 memset(f,0,sizeof(f));
	 for (int i=c;i<=n;i++)
	 {
	     f[i]=f[i-1];
	     if (a[i]-a[i-c+1]<=x)  //满足当前最大值
	        f[i]=f[i-c]+1;  //累加行
	 }
	 if (f[n]>=r)  //满足行数
	    return 1;
	    else return 0;
}
int main()
{
	memset(f,0x7f,sizeof(f));
	scanf("%d%d%d",&n,&r,&c);
	for (int i=1;i<=n;i++)
	    scanf("%d",&a[i]);
	sort(a+1,a+n+1);
	int l=0,r=a[n]-a[1];  //初始化
	while (l<r)
	{
		  int mid=(l+r)/2;  //枚举最大值
		  if (check(mid))  //判断
		     r=mid;
		     else l=mid+1;
	}
    printf("%d",l);
    return 0;
}

你可能感兴趣的:(二分)