题目大意

给定区间[0,L],在区间内给定N个数,加上区间的端点总共N+2个值。这N+2个数相邻的两个数之间有一个差值delta[i],现在可以从除去端点之外的这N个数中删除M个,使得剩余的N+2-M个数相邻数的最小差值最大。求这N+2-M个数相邻数的最小差值的最大值。

题目分析

典型的最大化最小值问题,最大化/最小化 最大值/最小值/平均值的问题,通常可以考虑使用二分法。如果对于一个数x,判断x是否满足题目要求的时间复杂度不高,那么可以尝试枚举x(当然如果x满足要求,那么所有比x大(或者所有比x小)的数都满足要求,则可以使用二分法枚举)。 
    题目中位置0和位置L处的数字不能删除,假设从N个数中删除M个,可以使得相邻数字的最小差值为x。那么就从位置0开始考虑,寻找下一个和当前数字差值小于x的数a,将a删除... 直到删除数字为M个或者到达N+1的位置停止。若还没删除M个就已经到达了N+1的位置,则说明必定可以使得相邻数字的最小差值大于等于x;若已经删除了M个,则考虑剩余的那些数字中的相邻差值是否小于x,若小于,则说明不满足要求。 
    此题关键的一点是位置0处的数字不能删除,这样就可以从位置0开始考虑,用类似贪心的方法,通过最小距离大于等于x来覆盖到N+1。

实现(c++)

#include<stdio.h>
#include<cmath>
#include<algorithm>
using namespace std;
#define MAX_N 50005
int gDist[MAX_N];
//判断能否从n个数字中删除m个,使得相邻最小距离大于等于 min_dist
bool CanSet(int n, int m, int min_dist){
int last_i = 0;
int i = 1;
int c = 0;
while (i <= n && c < m){
if (gDist[i] - gDist[last_i] < min_dist){
c++;
}else
last_i = i;
i++;
} if (c == m){
if (gDist[i] - gDist[last_i] < min_dist) //注意考虑 i和 last_i
return false; for (; i <= n; i++)
if (gDist[i + 1] - gDist[i] < min_dist)
return false;
}
return true;
}
int main(){
int l, n, m;
while (scanf("%d %d %d", &l, &n, &m) != EOF){
gDist[0] = 0;
gDist[n + 1] = l;
for (int i = 1; i <= n; i++){
scanf("%d", gDist + i);
}
sort(gDist, gDist + n + 2);
int beg = 1, end = l + 1; //二分法,左闭右开!! 故用 l+1
while (beg < end){
int mid = (beg + end) / 2;
if (CanSet(n, m, mid))
beg = mid + 1;
else
end = mid;
}
printf("%d\n", beg - 1);
}
return 0;
}

poj_3258 二分法的更多相关文章

  1. C语言两种查找方式(分块查找,二分法)

    二分法(必须要保证数据是有序排列的):   分块查找(数据有如下特点:块间有序,块内无序):    

  2. poj3122-Pie(二分法+贪心思想)

    一,题意: 有f+1个人(包括自己),n块披萨pie,给你每块pie的半径,要你公平的把尽可能多的pie分给每一个人 而且每个人得到的pie来自一个pie,不能拼凑,多余的边角丢掉.二,思路: 1,输 ...

  3. 二分法&三分法

    ural History Exam    二分 #include <iostream> #include <cstdlib> using namespace std; //二分 ...

  4. [No000087]Linq排序,SortedList排序,二分法排序性能比较

    using System; using System.Collections; using System.Collections.Generic; using System.Diagnostics; ...

  5. [PHP]基本排序(冒泡排序、快速排序、选择排序、插入排序、二分法排序)

    冒泡排序: function bubbleSort($array){ $len=count($array); //该层循环控制 需要冒泡的轮数 for($i=1;$i<$len;$i++){ / ...

  6. iOS常见算法(二分法 冒泡 选择 快排)

    二分法: 平均时间复杂度:O(log2n) int halfFuntion(int a[], int length, int number)  { int start = 0; int end = l ...

  7. java简单的二分法排序

    二分法排序的思路:数据元素要按顺序排列,对于给定值 x,从序列的中间位置开始比较,如果当前位置值等于 x,则查找成功:若 x 小于当前位置值,则在数列的前半段中查找:若 x 大于当前位置值则在数列的后 ...

  8. 使用二分法查找mobile文件中区号归属地

    #!/usr/bin/env python #coding:utf-8 ''' Created on 2015年12月8日 @author: DL @Description: 使用二分法查找mobil ...

  9. Atitit 迭代法  “二分法”和“牛顿迭代法 attilax总结

    Atitit 迭代法  "二分法"和"牛顿迭代法 attilax总结 1.1. ."二分法"和"牛顿迭代法"属于近似迭代法1 1. ...

随机推荐

  1. 常用PHP文件操作函数

    注:文件操作函数的行为受到 php.ini 中设置的影响. 当在 Unix 平台上规定路径时,正斜杠 (/) 用作目录分隔符.而在 Windows 平台上,正斜杠 (/) 和反斜杠 (\) 均可使用. ...

  2. SQL Server 查询分析器键盘快捷方式

    下表列出 SQL Server 查询分析器提供的所有键盘快捷方式. 活动 快捷方式 书签:清除所有书签. CTRL-SHIFT-F2 书签:插入或删除书签(切换). CTRL+F2 书签:移动到下一个 ...

  3. MySQL 常用语法 之 DISTINCT

    DISTINCT作用很简单就是去除重复行的数据. 具体看下面列子 表A数据[两条 nami 99] nameA   scoreA robin    98 nami    99 saber  98 lu ...

  4. e667. 在给定图像中创建缓冲图像

    An Image object cannot be converted to a BufferedImage object. The closest equivalent is to create a ...

  5. Nodejs入门手记 (01):Hello World的WEB程序

    声明:本文为原创文章,如需转载,请注明来源并保留原文链接Allong,谢谢! “滚滚长江东逝水,浪花淘尽英雄.是非成败转头空.” - <临江仙·杨慎·明> 很熟悉的旋律,鸡汤了一下:高考是 ...

  6. c++ define的用法(转)

    #define是C语言中提供的宏定义命令,其主要目的是为程序员在编程时提供一定的方便,并能在一定程度上提高程序的运行效率,但学生在学习时往往不能 理解该命令的本质,总是在此处产生一些困惑,在编程时误用 ...

  7. 适配器模式(Adapter Pattern)----------结构型模式

    对象适配器模式的缺点是:与类适配器模式相比,要在适配器中置换适配着类的某些方法比较麻烦.如果一定要置换掉适配者类的一个或多个方法,可以先做一个适配者类的子类,在子类中将适配者类的方法置换掉,然后再把适 ...

  8. v$Session详解

    从Oracle10gR1开始,Oracle在V$SESSION中增加关于等待事件的字段,实际上也就是把原来V$SESSION_WAIT视图中的所有字段全部整合到了V$SESSION视图中,开始的时候我 ...

  9. ini 文件操作记要(2): 使用 TMemIniFile

    unit Unit1; interface uses  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Form ...

  10. 如何使用ChemDraw绘制自由基符号

    ChemDraw软件是一款全球领先的化学绘图工具,能够绘制各种复杂的化学符号和化学结构图形.ChemDraw汉化版结合了中国用户的使用习惯,可以帮助国内化学行业工作者更加轻松快捷地绘制化学图形.本教程 ...