hihocoder 1169 猜数字
描述
你正在和小冰玩一个猜数字的游戏。小冰首先生成一个长为N的整数序列 $A_1, A_2, \dots , A_N$。在每一轮游戏中,小冰会给出一个区间范围 $[L, R]$,然后你要猜一个数 $K$。如果 $K$ 在 $A_L, A_{L+1}, \dots , A_R$ 中,那么你获胜。
在尝试了几轮之后,你发现这个游戏太难(无聊)了。小冰决定给你一些提示,你每猜一次,小冰会告诉你 $K$ 与 $A_L, A_{L+1}, \dots, A_R$ 中最接近的数的绝对差值,即 $\min(|A_i - K|), L \le i \le R$。
现在,请你实现这个新功能。
输入
第一行为一个整数 $T$,表示数据组数。
每组数据的第一行为两个整数 $N$ 和 $Q$。
第二行为 $N$ 个由空格分开的整数,分别代表 $A_1, A_2, \dots, A_N$。
接下来 $Q$ 行,每行三个由空格隔开的整数 $L、R、K$。
输出
每组数据的先输出一行"Case #X:",X为测试数据编号。
接下来对每个询问输出一行,每行为一个整数,即为所求的值。
数据范围
1 ≤ T ≤ 20
0 ≤ Ai, K ≤ 109
1 ≤ L ≤ R ≤ N
小数据
1 ≤ N, Q ≤ 1000
大数据
1 ≤ N, Q ≤ 200000
输入数据量较大,推荐使用scanf / BufferedReader等IO方法。
- 样例输入
- 
1 
 9 3
 1 8 3 4 9 2 7 6 5
 1 9 10
 3 7 9
 5 6 5
- 样例输出
- Case #1:
- 1
- 0
- 3
Solution
- 先定义一个概念
- 给定数组 $a_{1},\dots,a_{n}$,数 $k$ 在区间 $[l,r]$ 上的Rank——记作Rank(k, l, r)——定义为:
- a[l..r]上小于k的数字的个数
- 对于每组询问 $L, R, K$,先求出Rank(K, L, R),剩下的问题就是求(静态)区间第 $k$ 小。
- 为了方便表述,将 $a[l \dots r]$ 上第 $k$ 小的数记作 least(k, l, r),ans表示每个查询的答案,分三种情况:
- (1) Rank(K, L, R) = 0, ans = least(Rank(K, L, R)+1, L, R) - K
- (2) Rank(K, L, R) = R-L+1, ans = K - least(Rank(K, L, R), L, R)
- (3) otherwise, ans = min(least(Rank(K, L, R)+1, L, R) - K, K - least(Rank(K, L, R), L, R))
- Rank(K, L, R)与least(K, L, R)都可用划分树实现,单次查询的时间复杂度都是O(log n),而且二者代码非常相似。
- 划分树的空间复杂度是 $O(n \log n)$。
- 划分树 (partition tree)
- 划分树是线段树的一种,用于维护数组 $A[1 \dots n]$。它的每个节点u代表数组 $A$ 的一些元素。
- 假设内部(即非叶子)节点 $u$ 表示元素是 $u[L..R] \subset A$,定义节点 $u$ 的长度 $u.length=R-L+1$,记 $u$ 的左右儿子分别为 $v, w$,令
- \[mid\equiv \frac{L+R}{2}\]
- 则
- \[v.length=mid-L+1, w.length=R-mid\]
- 或者表示成
- \[ v = v[L, mid], w=w(mid,R] \]
Implmentation
#include <bits/stdc++.h>
using namespace std; const int N(2e5+); int a[][N], toleft[][N], sa[N]; void build(int lev, int l, int r){
if(l==r) return;
int mid=(l+r)>>, &tar=sa[mid], nl=mid-l+;
for(int i=l; i<=r; i++) if(a[lev][i]<tar) nl--;
for(int i=l, lp=l, rp=mid+; i<=r; i++){
if(a[lev][i]<tar) a[lev+][lp++]=a[lev][i];
else if(a[lev][i]>tar) a[lev+][rp++]=a[lev][i];
else nl?a[lev+][lp++]=a[lev][i],nl--:a[lev+][rp++]=a[lev][i];
toleft[lev][i]=toleft[lev][l-]+lp-l;
}
build(lev+, l, mid); build(lev+, mid+, r);
}
//Rank(l, r, k):区间[l, r]内比k小的数的个数
//满足区间加法
int Rank(int lev, int L, int R, int l, int r, int val){
if(L==R) return a[lev][L]<val; int nl=toleft[lev][r]-toleft[lev][l-], nr=r-l+-nl, mid=(L+R)>>; if(sa[mid]>=val){
if(nl){
l=L+toleft[lev][l-]-toleft[lev][L-];
r=l+nl-;
return Rank(lev+, L, mid, l, r, val);
}
return ; //error-prone
}
else{
if(nr){
r+=toleft[lev][R]-toleft[lev][r];
l=r-nr+;
return nl+Rank(lev+, mid+, R, l, r, val);
}
return nl;
}
} int Query(int lev, int L, int R, int l, int r, int k){
if(L==R) return a[lev][L]; int nl=toleft[lev][r]-toleft[lev][l-], nr=r-l+-nl, mid=(L+R)>>; if(nl>=k){
l=L+toleft[lev][l-]-toleft[lev][L-];
r=l+nl-;
return Query(lev+, L, mid, l, r, k);
}
r+=toleft[lev][R]-toleft[lev][r];
l=r-nr+;
return Query(lev+, mid+, R, l, r, k-nl);
} int main(){
int T; scanf("%d", &T);
for(int n, q, cs=; T--;){
scanf("%d%d", &n, &q);
printf("Case #%d:\n", ++cs); for(int i=; i<=n; i++) scanf("%d", sa+i), a[][i]=sa[i];
sort(sa+, sa+n+); build(, , n); for(int l, r, k, rk, res; q--;){
scanf("%d%d%d", &l, &r, &k);
rk=Rank(, , n, l, r, k);
//printf("%d\n", rk);
if(rk==) res=Query(, , n, l, r, rk+)-k;
else if(rk==r-l+) res=k-Query(, , n, l, r, rk);
else res=min(Query(, , n, l, r, rk+)-k, k-Query(, , n, l, r, rk));
printf("%d\n", res);
}
}
}
问题解决了,但代码不是可以写得再短一些?
Rank和Query可否合并到一起呢?
hihocoder 1169 猜数字的更多相关文章
- C语言猜数字游戏
		猜数字游戏,各式各样的实现方式,我这边提供一个实现方式,希望可以帮到新手. 老程序猿就不要看了,黑呵呵 源代码1 include stdio.h include stdlib.h include ti ... 
- 不一样的猜数字游戏 — leetcode 375. Guess Number Higher or Lower II
		好久没切 leetcode 的题了,静下心来切了道,这道题比较有意思,和大家分享下. 我把它叫做 "不一样的猜数字游戏",我们先来看看传统的猜数字游戏,Guess Number H ... 
- java 猜数字游戏
		作用:猜数字游戏.随机产生1个数字(1~10),大了.小了或者成功后给出提示. 语言:java 工具:eclipse 作者:潇洒鸿图 时间:2016.11.10 >>>>> ... 
- 【原创Android游戏】--猜数字游戏Version 0.1
		想当年高中时经常和小伙伴在纸上或者黑板上或者学习机上玩猜数字的游戏,在当年那个手机等娱乐设备在我们那还不是很普遍的时候是很好的一个消遣的游戏,去年的时候便写了一个Android版的猜数字游戏,只是当时 ... 
- 【原创Android游戏】--猜数字游戏V1.1 --数据存储,Intent,SimpleAdapter的学习与应用
		--------------------------------------------------------------- V0.1版本 上次做完第一个版本后,发现还有一些漏洞,并且还有一些可以添 ... 
- python学习笔记 ——python写的猜数字游戏 002
		from sys import exit import random def Arrfor(str): #CONTST = CONTST + 1 artificial = input("请输 ... 
- Python小游戏之猜数字
		最近师兄师姐毕业,各种酒席,酒席上最常玩的一个游戏就是猜数字,游戏规则如下: 出题人在手机上输入一个0-100之间的数字,其它人轮流猜这个数字,如果你不幸猜中则要罚酒一杯.每次猜数字,出题人都要缩小范 ... 
- 【Qt】2.4 做一个“猜数字”的游戏
		使用对话框和Qt设计师来实现一个相当简单的小游戏.同时将通过这个程序来看布局的隐藏和显示是如何来影响窗口界面的变化的. 新建一个Qt项目,把Qt Creator默认给的mainwindow.h.mai ... 
- python写的第一个简单小游戏-猜数字
		#Filename:game1.py guess=10 running=True while running: try: answer=int(raw_input('Guess what i thin ... 
随机推荐
- Android Studio如何设置代码自动提示
			在用Eclipse时候,你可以进行设置,设置成不管你输入任何字母,都能进行代码的提示,在Android Studio中也可以 设置,而且比Eclipse设置来的简单.当然如果你觉得代码自动提示会降低你 ... 
- AMAP
			ViewController.m #import "ViewController.h" //地图显示需要的头文件 #import <MAMapKit/MAMapKit.h&g ... 
- 使用AdapterTypeRender对不同类型的item数据到UI的渲染
			要实现聊天功能中的发送不同类型的信息,比如纯文本.图片.语音.图文混排多媒体的数据等(具体效果看微信). 这里使用AdapterTypeRender在BaseTypeAdapter(这个之后会讲到)中 ... 
- 从Python爬虫到SAE云和微信公众号:二、新浪SAE上搭建微信服务
			目的:用PHP在SAE上搭建一个微信公众号的服务器. 1.申请一个SAE云账号 SAE申请地址:http://sae.sina.com.cn/ 可以使用微博账号登陆,SAE是新浪的云服务,时间也比较 ... 
- 用python简单处理图片(1):打开\显示\保存图像
			一提到数字图像处理,可能大多数人就会想到matlab,但matlab也有自身的缺点: 1.不开源,价格贵 2.软件容量大.一般3G以上,高版本甚至达5G以上. 3.只能做研究,不易转化成软件. 因此, ... 
- java.util.ConcurrentModificationException 解决办法
			在使用iterator.hasNext()操作迭代器的时候,如果此时迭代的对象发生改变,比如插入了新数据,或者有数据被删除. 则使用会报以下异常:Java.util.ConcurrentModific ... 
- polya计数定理在ACM-icpc中的应用
			[数学公式] PG(x1,x2,...,xn) = 1/|G| * ∑π∈G x1^b1 * x2^b2*...*bn^bn 其中π是1^b12^b2...n^bn型轮换 然后一般染色情况下x1= ... 
- Jenkins进阶系列之——11修改Jenkins用户的密码
			说明:本方法仅适用于jdk6+.tomcat6+和Jenkins专有用户数据库的Jenkins! 很多童鞋在使用jenkins的时候忘记密码了,然后各种蛋疼.最近闲着无事,折腾了下.好了,闲话少扯. ... 
- WPF  MVVM 写一个健壮的INotifyPropertyChanged基类
			当我们用MVVM的时候要实现INotifyPropertyChanged,如果你是基于.net4.5以下的framework(.net4.5已有新特性我这里就不说了) 你很可能会这么写 public ... 
- Android四大布局及其主要属性
			布局: <LinearLayout></LinearLayout> <RelativeLayout></RelativeLayout> <Fram ... 
