ZOJ Bizarre Routine 2013杭州网赛B题
题目意思:
给定n, expect, a, b 要求你构造一组array[],存放一个1..n的排列,使的下面的程序能输出YES
题目所示代码:
bool less_than(x, y) {
T++;
return x < y;
}
void work(array[], l, r) {
if (l >= r) return;
swap(array[(l * A + r * B) / (A + B)], array[r]);
int index = l;
for (i = l; i < r; i++)
if (less_than(array[i], array[r]))
swap(array[index++], array[i]);
swap(array[r], array[index]);
work(array, l, index - );
work(array, index + , r);
}
void main() {
T = ;
Input(n, expect, A, B, array[]);
work(array, , n - );
if (T == expect)
Output("YES");
else
Output("NO");
}
sort
壕无疑问这是一个快排!
n个数排列,T的上限是sigma(1..n-1) = n*(n-1)/2
而下限则可以通过递推得到
mi[x] = 0 for x =0, 1
mi[x] = x-1 + min(mi[i] + mi[x-i-1]) (0 <=i < x) for x > 1
当然根据经验得到,i应该是取x/2的时候最好,跑了一下,mi数组也确实没变。。当然跑n^2的应该时间也够。
而且实际上 mi[i] + mi[x-i-1] 在i = 0..x/2呈现非递增的性质,解法用到了这个性质。
当expext在上下限之间的时候,可以通过递归构造
void work(int l, int r, int c)
构造区间[l,r] 使得这个区间的T = c
显然这个区间要消耗r - l次
设对这个区间的第一次排序后中间数在m位置,还要消耗区间[l,m - 1], [m+1, r]
当m = l .. (l+r)/2 (Ps: 当 m > (l+r)/2时 和<= (l+r)/2的情况是等价的)
假如[l,m - 1], [m+1, r]均取最小值,则数值呈现非递减
假如[l,m - 1], [m+1, r]均取最大值,则数值呈现非递减
由于这个区间一定存在m符合条件,那么取第一个使得区间消耗最小值mi(l,m - 1) + mi(m+1, r) <= c - (r-l) 的必然符合要求(Ps: 把最小值和最大值在纸上画一下函数就明了)
让左区间取消耗最小值, 然后递归构造
work(l,m-1,mi[l-m]) work(m1,r,c - (r-l) - mi[l-m])
我们希望区间[l, r]第一次sort后,p[l..r] = [ p[l..m-1], m ,p[m+1..r] ]
那么,通过给定的a, b计算得到m原本的下标,手动回滚他程序中的两次swap。
构造完毕,代码如下:
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
const int N = ;
int a,b,p[N];
int mi[N];
inline bool can(int l,int r,int c){
int n = r-l+;
return c>=mi[n] && c<=n*(n-)/;
}
inline void mysort(int l,int m,int r,int id){
p[m] = m;
swap(p[m],p[r]);
swap(p[r],p[id]);
}
void work(int l,int r,int c){
if(l>r) return;
if(l == r){
p[l] = l;
return;
}
int lt = l,rt = (l+r)/+, m;
c -= r-l;
while(lt<rt){
m = (lt + rt)>>;
if(mi[m-l]+mi[r-m] > c) lt = m + ;
else rt = m;
}
m = lt;
work(l,m-,mi[m-l]);
work(m+,r,c-mi[m-l]);
mysort(l,m,r,(l*a + r*b)/(a+b));
}
int n,m;
int main(){
//freopen("in.txt", "r", stdin);
for(int i=;i<N;i++){
int x = i/, y = i-x-;
mi[i] = mi[x]+mi[y]+i-;
}
//for(int i=0;i<=20;i++)cout<<i<<" "<<mi[i]<<endl;
while(~scanf("%d%d%d%d",&n,&m,&a,&b)){
if(!can(,n-,m)){
puts("NOWAY");
continue;
}
work(,n-,m);
for(int i=;i<n;i++) printf(i==n-?"%d\n":"%d ",p[i]+);
}
return ;
}
ZOJ Bizarre Routine 2013杭州网赛B题的更多相关文章
- HDU 4746 HDOJ Mophues 2013杭州网赛I题
比赛的时候就预感到这题能出,但是会耗时比较多.结果最后是出了,但是有更简单的题没出. 是不是错误的决策呢?谁知道呢 题目意思: 定义f(x) = x分解质因数出来的因子个数 如 x = p0 * p0 ...
- HDU 4747 Mex (2013杭州网络赛1010题,线段树)
Mex Time Limit: 15000/5000 MS (Java/Others) Memory Limit: 65535/65535 K (Java/Others)Total Submis ...
- HDU 4741 Save Labman No.004 (2013杭州网络赛1004题,求三维空间异面直线的距离及最近点)
Save Labman No.004 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Other ...
- HDU 4739 Zhuge Liang's Mines (2013杭州网络赛1002题)
Zhuge Liang's Mines Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Othe ...
- HDU 4738 Caocao's Bridges (2013杭州网络赛1001题,连通图,求桥)
Caocao's Bridges Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- 2013杭州现场赛B题-Rabbit Kingdom
杭州现场赛的题.BFS+DFS #include <iostream> #include<cstdio> #include<cstring> #define inf ...
- 2013杭州网赛 1001 hdu 4738 Caocao's Bridges(双连通分量割边/桥)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4738 题意:有n座岛和m条桥,每条桥上有w个兵守着,现在要派不少于守桥的士兵数的人去炸桥,只能炸一条桥 ...
- 2013杭州网络赛D题HDU 4741(计算几何 解三元一次方程组)
Save Labman No.004 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Other ...
- 2013杭州网络赛C题HDU 4640(模拟)
The Donkey of Gui Zhou Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/O ...
随机推荐
- DCMTK读取DICOM文件头信息的三种方法
Howto: Load File Meta-Header Here's an example that shows how to load the File Meta Information Head ...
- java的有用基础知识(2013-05-02-bd 写的日志迁移
JDK 是整个Java的核心,包括了Java运行环境.Java工具和Java基础类库.是java开发工具包 jre是java的运行环境(如果不做开发就不用安装jdk单独安装jre就可以运行java程序 ...
- iar注释快捷键
选中多行后注释快捷键:Ctrl+K 取消多行注释快捷键:Ctrl+Shift+K
- linux-shell——01
没有什么好的标题,只是一些随笔.我用的是linux虚拟机,red hat 7 一:nat模式使得虚拟机可以访问外网,但是这种模式下只可以访问外网,但外面的不能访问里面 首先将虚拟机的网络连接改为nat ...
- POJ:1703-Find them, Catch them(并查集好题)(种类并查集)
Find them, Catch them Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 49867 Accepted: 153 ...
- [Uva11178]Morley's Theorem(计算几何)
Description 题目链接 Solution 计算几何入门题 只要求出三角形DEF的一个点就能推出其他两个点 把一条边往内旋转a/3度得到一条射线,再做一条交点就是了 Code #include ...
- 1 web应用
web应用 Web应用程序是一种可以通过Web访问的应用程序,程序的最大好处是用户很容易访问应用程序,用户只需要有浏览器即可,不需要再安装其他软件.应用程序有两种模式C/S.B/S.C/S是客户端/服 ...
- css3心形 perspective transform
CSS3挺有趣的,能实现不少动画,以下为娱乐内容 <!DOCTYPE html> <html> <head> <meta charset="UTF- ...
- 算法:枚举法---kotlin
枚举法:效率低,循环所有的情况,找到正确答案 用于解决数学问题,还是很简单的. 比如,奥数里面: 算 法 描 述 题X题=题题题题题题 其中 算法描述题每一个为一个数字,请写出正确的数字. ok,我们 ...
- 关于代码通过API操作阿里云RDS的巨坑
由于项目原因,要通过API操作阿里云的数据库,于是简单研究了一下阿里云提供的相关文档,发现官方提供了.NET的SDK,而且还提供了github开源代码,这个要为阿里点赞! 于是到github上弄了一份 ...