题目链接:

IOI2018highway

题目大意:给出一张$n$个点$m$条边的无向图,并给出一对未知的起点和终点,每条边都有两种边权$A$和$B$(每条边的$A$和$B$都分别相同),每次你可以设置每条边的边权并向交互库询问,交互库会返回给你当前边权下起点到终点的最短路,你需要在不多于$50$次的询问后找出起点和终点。

我们设起点为$S$,终点为$T$。

首先需要一次询问将边权都设为$A$来知道$S$到$T$的最短路。然后我们可以用二分来找到一个处于$S$到$T$最短路上的点:每次将编号在$[0,mid]$的点的所有出边设为$B$,其他的设为$A$。如果得到的最短路不变,那么显然编号在$[mid+1,n-1]$的点有处于$S$到$T$最短路上的点,反之编号在$[0,mid]$的点有处于$S$到$T$最短路上的点。我们设找到的这个点为$x$,那么$S$与$T$中一定有一个点距离$x$较远,我们设这个点为$S$。从$x$开始$bfs$,二分然后每次将$bfs$序的$[mid+1,n-1]$这些点的所有出边设为$B$,其他边设为$A$,这样就能找到$S$,再从$S$开始$bfs$同样二分$bfs$序找到$T$。这样询问次数是$3*log_{2}^{90000}+1=52$,可以得到$90$分。

既然第一步可以二分找到最短路上的一个点,那么我们同样也可以找到一条边。每次将编号在$[0,mid]$的边设为$B$其他边设为$A$来找到最短路上的一条边,对于这条边的两端点$(u,v)$显然每个点到这两个点的最短距离不同,我们按每个点到这两个点的最短距离将离$u$更近的分为一部分,离$v$更近的分为另一部分,对于每部分还是二分$bfs$序来分别找到$S$和$T$,这样最坏情况询问次数为$1+log_{2}^{130000}+2*log_{2}^{45000}=50$,即可得到满分。

#include"highway.h"
#include<queue>
#include<cstdio>
#include<vector>
#include<algorithm>
#define ll long long
using namespace std;
vector<int>s[90010];
vector<int>to[90010];
int w[130010];
ll path;
queue<int>q;
int dis_u[90010];
int dis_v[90010];
int que_u[90010];
int que_v[90010];
int cnt_u;
int cnt_v;
int S,T;
bool cmp_u(int x,int y)
{
return dis_u[x]<dis_u[y];
}
bool cmp_v(int x,int y)
{
return dis_v[x]<dis_v[y];
}
void find_pair(int n,vector<int> u,vector<int> v,int A,int B)
{
int num=u.size();
for(int i=0;i<num;i++)
{
s[u[i]].push_back(i);
to[u[i]].push_back(v[i]);
s[v[i]].push_back(i);
to[v[i]].push_back(u[i]);
}
path=ask(vector<int>(w,w+num));
int l=0;
int r=num-1;
while(l<r)
{
int mid=(l+r)>>1;
for(int i=0;i<num;i++)
{
w[i]=0;
}
for(int i=0;i<=mid;i++)
{
w[i]=1;
}
ll value=ask(vector<int>(w,w+num));
if(value==path)
{
l=mid+1;
}
else
{
r=mid;
}
}
q.push(u[l]);
dis_u[u[l]]=1;
while(!q.empty())
{
int now=q.front();
q.pop();
int len=to[now].size();
for(int i=0;i<len;i++)
{
if(!dis_u[to[now][i]])
{
dis_u[to[now][i]]=dis_u[now]+1;
q.push(to[now][i]);
}
}
}
q.push(v[l]);
dis_v[v[l]]=1;
while(!q.empty())
{
int now=q.front();
q.pop();
int len=to[now].size();
for(int i=0;i<len;i++)
{
if(!dis_v[to[now][i]])
{
dis_v[to[now][i]]=dis_v[now]+1;
q.push(to[now][i]);
}
}
}
for(int i=0;i<n;i++)
{
if(dis_u[i]<dis_v[i])
{
que_u[++cnt_u]=i;
}
else
{
que_v[++cnt_v]=i;
}
}
sort(que_u+1,que_u+1+cnt_u,cmp_u);
sort(que_v+1,que_v+1+cnt_v,cmp_v);
l=1,r=cnt_u;
while(l<r)
{
int mid=(l+r)>>1;
for(int i=0;i<num;i++)
{
w[i]=0;
}
for(int i=mid+1;i<=cnt_u;i++)
{
int len=s[que_u[i]].size();
for(int j=0;j<len;j++)
{
w[s[que_u[i]][j]]=1;
}
}
ll value=ask(vector<int>(w,w+num));
if(path==value)
{
r=mid;
}
else
{
l=mid+1;
}
}
S=que_u[l];
l=1,r=cnt_v;
while(l<r)
{
int mid=(l+r)>>1;
for(int i=0;i<num;i++)
{
w[i]=0;
}
for(int i=mid+1;i<=cnt_v;i++)
{
int len=s[que_v[i]].size();
for(int j=0;j<len;j++)
{
w[s[que_v[i]][j]]=1;
}
}
ll value=ask(vector<int>(w,w+num));
if(path==value)
{
r=mid;
}
else
{
l=mid+1;
}
}
T=que_v[l];
answer(S,T);
}

[IOI2018]高速公路收费——二分查找+bfs的更多相关文章

  1. HDU 5652 India and China Origins 二分优化+BFS剪枝

    题目大意:给你一个地图0代表可以通过1代表不可以通过.只要能从第一行走到最后一行,那么中国与印度是可以联通的.现在给你q个点,每年风沙会按顺序侵蚀这个点,使改点不可通过.问几年后中国与印度不连通.若一 ...

  2. jvascript 顺序查找和二分查找法

    第一种:顺序查找法 中心思想:和数组中的值逐个比对! /* * 参数说明: * array:传入数组 * findVal:传入需要查找的数 */ function Orderseach(array,f ...

  3. Java实现的二分查找算法

    二分查找又称折半查找,它是一种效率较高的查找方法. 折半查找的算法思想是将数列按有序化(递增或递减)排列,查找过程中采用跳跃式方式查找,即先以有序数列的中点位置为比较对象,如果要找的元素值小 于该中点 ...

  4. 从一个NOI题目再学习二分查找。

    二分法的基本思路是对一个有序序列(递增递减都可以)查找时,测试一个中间下标处的值,若值比期待值小,则在更大的一侧进行查找(反之亦然),查找时再次二分.这比顺序访问要少很多访问量,效率很高. 设:low ...

  5. java实现二分查找

    /** * 二分查找 * @param a * @param n * @param value * @return * @date 2016-10-8 * @author shaobn */ publ ...

  6. 最新IP地址数据库 二分逼近&二分查找 高效解析800万大数据之区域分布

    最新IP地址数据库  来自 qqzeng.com 利用二分逼近法(bisection method) ,每秒300多万, 比较高效! 原来的顺序查找算法 效率比较低 readonly string i ...

  7. c#-二分查找-算法

    折半搜索,也称二分查找算法.二分搜索,是一种在有序数组中查找某一特定元素的搜索算法. A 搜素过程从数组的中间元素开始,如果中间元素正好是要查找的元素,则搜素过程结束: B 如果某一特定元素大于或者小 ...

  8. 【Python】二分查找算法

    二分查找:在一段数字内,找到中间值,判断要找的值和中间值大小的比较.如果中间值大一些,则在中间值的左侧区域继续按照上述方式查找.如果中间值小一些,则在中间值的右侧区域继续按照上述方式查找.直到找到我们 ...

  9. PHP实现文本快速查找 - 二分查找

    PHP实现文本快速查找 - 二分查找法 起因 先说说事情的起因,最近在分析数据时经常遇到一种场景,代码需要频繁的读某一张数据库的表,比如根据地区ID获取地区名称.根据网站分类ID获取分类名称.根据关键 ...

随机推荐

  1. iOS开发简记(9):APPStore审核

    "觅知音"这个APP的第一个版本从提交审核到上架,历时三个星期,其中遇到一些审核上的问题,它的处理或许能帮助到遇到同样问题的小伙伴们,所以这里列举出来,这三个星期如何跟苹果的审核团 ...

  2. nginx 动态添加ssl模块

    一.查看nginx模块 /usr/local/nginx/sbin/nginx -V 二.安装openssl包 yum -y install pcre  pcre-devel zlib  zlib-d ...

  3. vue报错信息

    1.Property or method "xxx" is not defined on the instance but referenced during render. 原因 ...

  4. Python-TXT文本操作

    一.列出IO操作的标识符及描述 标识符 描述 r 以只读方式打开文件.文件的指针将会放在文件的开头.这是默认模式. rb 以二进制格式打开一个文件用于只读.文件指针将会放在文件的开头.这是默认模式. ...

  5. hibernate中实体与数据库中属性对应的类型

    常用的字段及类型,在数据库中字段名称若与实体对应的属性字段名称相同,hibernate可以自动映射,在一些情况下hibernate可能报错这时候有的错误可以通过指定对应的类型避免.下面给出一些常用的 ...

  6. Python—闭包

    闭包的定义:即函数定义和函数表达式位于另一个函数的函数体内(嵌套函数).而且,这些内部函数可以访问它们所在的外部函数中声明的所有局部变量.参数.当其中一个这样的内部函数在包含它们的外部函数之外被调用时 ...

  7. 牛客练习赛38 D 题 出题人的手环 (离散化+树状数组求逆序对+前缀和)

    链接:https://ac.nowcoder.com/acm/contest/358/D来源:牛客网 出题人的手环 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 524288K,其他 ...

  8. 通过Webstorm上传代码到Github、更新代码后同步到github及克隆github代码到本地的方法

    导读: Github做为IT爱好者分享代码的一个知名的平台,广受大家喜欢,那么我们平时该怎么将自己写的代码上传到github上面保存并且提供给其他人参考? 我想方法不外乎如下几个: 1.直接在gith ...

  9. JSP页面的基本元素

    JSP页面元素构成:静态内容.指令.表达式.小脚本.声明.注释. JSP指令包括: page指令:通常位于jsp页面的顶端,同一个页面可以有多个page指令. include指令:将一个外部文件嵌入到 ...

  10. mysql 小数转换成百分数查出(保留两位小数百分数)

    SELECT id as 'ID',GROUP_CONCAT(concat(truncate(royalties *100,2),'%')) as '比例' FROM yser FROM id in( ...