题目大意

给定N个数,这些数字两两求差构成C(N,2)(即N*(N-1)/2)个数值,求这C(N,2)个数的中位数。N <= 100000.

题目分析

根据数据规模N最大为100000,可知不能直接求每两个数的查找(O(N*N)复杂度),然后排序,再求中位数。考虑使用二分法查找满足要求的中位数: 
    先将原数组排序,然后假设数k为中位数,那么如果这N个数字构成的C(N,2)个差值中小于k的小于C(N,2)/2个,则k需要增加;如果小于等于k的数字大于等于C(N,2),那么k需要减少。最后求出的k是恰好满足C(N,2)个差值中小于等于k的数大于等于C(N,2)个,且差值中存在k 的最小值

实现(c++)

#include<stdio.h>
#include<algorithm>
using namespace std;
#define MAX_N 100005 int an[MAX_N];
int DiffKLessThanHalf(int n, int k){
int sum = 0;
int max_pair = n*(n-1) / 2;
max_pair = max_pair / 2 + max_pair % 2; //小于等于中位数的最少个数
bool flag = false;
for (int i = 0; i < n - 1; i++){
int beg = i + 1, end = n;
while (beg < end){
int mid = (beg + end) / 2;
if (an[mid] - an[i] == k){ //差值中存在k,说明k可以做为中位数
flag = true;
}
if (an[mid] - an[i] <= k)
beg = mid + 1;
else
end = mid;
}
sum += (beg - i - 1); //和 an[i] 的差值小于等于k的个数 为 beg - i - 1
if (sum >= max_pair && flag)//小于等于k的个数 至少为C(N,2) 且 存在差值等于k,说明k可以作为一个中位数,
//但后续需要继续求满足这种条件的最小的k。直接返回
return 2;
}
if (sum < max_pair) //小于等于k的个数 小于 C(N,2),说明中位数 > k
return 0;
return 1;
} int main(){
int n; while (scanf("%d", &n) != EOF){
for (int i = 0; i < n; i++){
scanf("%d", &an[i]);
}
sort(an, an + n);
int beg = 0, end = an[n-1] - an[0] + 1;
int rr = 0;
while (beg < end){
int mid = (beg + end) / 2;
int result = DiffKLessThanHalf(n, mid);
if (result == 0)
beg = mid + 1;
else if (result == 1){
end = mid;
}
else{
end = mid;
rr = mid; //记录下一个可能的中位数值
}
}
printf("%d\n", rr);
}
return 0;
}

poj_3579 二分法的更多相关文章

  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. 【Unity/Kinect】Kinect入门——项目搭建

    本文是Unity Store里的官方Demo包中的ReadMe翻译(别人翻的),介绍了用Unity如何入门搭建起一个Kinect项目工程. 非常感谢下面这位大大的无私奉献! http://www.ma ...

  2. 【C#/WPF】TextBlock/TextBox/Label编辑文字的问题

    标题有点描述不清,就当是为了方便自己用时易于搜索到. 总之需求是:显示用户信息(文字)时,允许用户编辑自己的信息. 效果图如下: 点击[编辑]按钮前: 点击[编辑]按钮后,允许编辑: 别吐槽为甚性别还 ...

  3. am335x Linux kernel DTS pinmux 定义记录

    记录am335x TI PDK3.0 Linux Kernel 设备的pinmux 的配置 在TI 的Linux kernel 设备树里面,有很多关于pinctrl-single,pins 的配置, ...

  4. 工作队列workqueue应用

    工作队列是另一种将工作推后执行的形式,它可以把工作交给一个内核线程去执行,这个下半部是在进程上下文中执行的,因此,它可以重新调度还有睡眠. 区分使用软中断/tasklet还是工作队列比较简单,如果推后 ...

  5. 透明遮罩图层VS高斯模糊滤镜 效果分析

    前端流行布局中最常见的弹出图层有popup, 对话框, tooltip等, 他们都使用了新的图层,但是实现办法各不相同, 有 的是通过半通明的黑白图层实现的, 有的是通过滤镜实现的, 我们来研究一下两 ...

  6. BuddyPress创建组、查看成员信息等找不到页面

    BuddyPress创建组.查看成员信息等找不到页面 http://aoxuangame.com/wordpress/groups/create/ http://aoxuangame.com/word ...

  7. 怎么让一个div 悬浮在另一个div上

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  8. Jquery与.net MVC结合,通过Ajax

    在工作中做了这么一个东西. Html端: @using Test.fh.Project.Storefront.ViewModels @using Test.fh.Project.Storefront. ...

  9. 移动H5功能设计反思 测试用例总结

    一.线上页面滑动流畅性测试 1.减少长动画效果(影响流畅) 2.是否自动跳转或者还是让用户自己操作跳转需要推敲 二.buttom和页面滑动的选择(优劣) 部分手机本身就会滑动不灵敏,大部分时候其实用b ...

  10. nginx反向代理压测问题记录

    使用nginx反向代理压测web程序,100个用户并发时,每隔一段时间loadrunner工具中就会报错,报错信息如下: Continuing after Error -26610: HTTP Sta ...