Description

  

  链接

  

Solution

  

  问题其实就是从一个点出发,每次可以走与其曼哈顿距离恰好为一个常数\(d\)的点

  

  显然不可能一一走完所有的边,这样复杂度下界至少是\(O(ans)\)

  

  我们采用折中方式:间接统计

  

  (1)找出从起始点能到达哪一些点

  (2)统计对于这些点之中的每一个点,与其距离为d的点有多少,求和除二就是总边数

  

  首先考虑第一步,如果我们通过枚举边的思路进行广搜,又要触及边的数目过多这一限制。考虑距离一个点\((x_0,y_0)\)曼哈顿距离为\(d\)的点\((x,y)\)应该满足什么特征,分四类:左上左下右上右下。左上右下的限制都是形如\(x-y=x_0\pm y_0\pm d\)且\(x\)在一定范围内的点,右上坐下的限制都是形如\(x+y=x_0\pm y_0\pm d\)的点

  

  枚举四种情况时,两个符号可以直接定下来,关键是对于每个\(a\),组织起满足\(x+y=a\)或\(x-y=a\)的点,并按\(x\)大小维护。由于我们不需要统计\(x\)在某个范围内的数具体有多少个,而仅仅是需要迭代枚举功能,我们马上想到内层应该要用一个set维护这一些点。那么外层是一个\(x+y\)或\(x-y\)的索引,关于索引,用map

  

  所以用两个map套set,维护\(x+y\)和\(x-y\)的点的信息。对于一个点,它在两个map中都有且仅有一个存在

  

  那么广搜时只需在map上查询所需的特征对应的set,并确定\(x\)值范围,在set上迭代枚举即可

  

  如果这样,我们发现这个复杂度下界至少是边数。我们第一步是找出所有联通点,而不是统计边数,每个点显然入一次队即可。所以如果一个点入队,我们就把它在两个map中的存在删除。这样就能保证复杂度与点数相关

  

  接下来是第二步,由于是无序边,我们想到顺序往右扫、并单向往左统计连边

  

  对于关键点按\(x\)排序,顺序右扫。考虑先前加入的点与当前点能连多少边。那么我们还是需要用\(x+y\)和\(x-y\)作为两个特征储存先前的点。与(1)不一样的是,这回我们要统计等于某个特征值、\(x\)在某个范围内的点有多少个。这回我们不用set,而用map套vector,这样就可以通过二分来确定某个范围内有多少个点。由于\(x\)递增、我们要查询的范围也和\(x\)有关,所以当前点加入信息时,直接将其pushback到每个vector后即可

  

Code

#include <cstdio>
#include <map>
#include <vector>
#include <set>
#include <queue>
#include <algorithm>
#define pb push_back
using namespace std;
typedef long long ll;
const int N=100005;
int n,sa,sb;
int stdDis;
struct Point{
int x,y,id;
void read(int _id){
scanf("%d%d",&x,&y);
id=_id;
}
};
Point p[N],a[N];
struct CompairByX{
bool operator() (const int &a,const int &b){
return p[a].x<p[b].x;
}
}tcmp;
map<int,set<int,CompairByX>> keyp,keyd; // x+y/x-y
int acnt;
inline int abs(int x){
return x>=0?x:-x;
}
int calc(Point &a,Point &b){
return abs(a.x-b.x)+abs(a.y-b.y);
}
void readData(){
scanf("%d%d%d",&n,&sa,&sb);
for(int i=1;i<=n;i++)
p[i].read(i);
stdDis=calc(p[sa],p[sb]);
}
void addPoint(Point &a){
keyp[a.x+a.y].insert(a.id);
keyd[a.x-a.y].insert(a.id);
}
void remPoint(Point &a){
keyp[a.x+a.y].erase(a.id);
keyd[a.x-a.y].erase(a.id);
}
void findConnectedSet(){
// store into a[acnt]
acnt=0;
for(int i=1;i<=n;i++)
addPoint(p[i]);
static queue<int> q;
static bool vis[N];
q.push(sa);
vis[sa]=true;
remPoint(p[sa]);
while(!q.empty()){/*{{{*/
int u=q.front(),v,cur;
q.pop();
a[++acnt]=p[u];
cur=p[u].x+p[u].y+stdDis;
for(auto i=keyp[cur].lower_bound(u),j=i;i!=keyp[cur].end()&&p[*i].x<=p[u].x+stdDis;i=j){ //right
j++;
v=(*i);
if(!vis[v]){
vis[v]=true;
q.push(v);
remPoint(p[v]);
}
}
cur=p[u].x+p[u].y-stdDis;
p[n+1]=(Point){p[u].x-stdDis,0};
for(auto i=keyp[cur].lower_bound(n+1),j=i;i!=keyp[cur].end()&&p[*i].x<p[u].x;i=j){ // left
j++;
v=(*i);
if(!vis[v]){
vis[v]=true;
q.push(v);
remPoint(p[v]);
}
}
cur=p[u].x-p[u].y+stdDis;
for(auto i=keyd[cur].lower_bound(u),j=i;i!=keyd[cur].end()&&p[*i].x<=p[u].x+stdDis;i=j){ //right
j++;
v=(*i);
if(!vis[v]){
vis[v]=true;
q.push(v);
remPoint(p[v]);
}
}
cur=p[u].x-p[u].y-stdDis;
for(auto i=keyd[cur].lower_bound(n+1),j=i;i!=keyd[cur].end()&&p[*i].x<p[u].x;i=j){ // left
j++;
v=(*i);
if(!vis[v]){
vis[v]=true;
q.push(v);
remPoint(p[v]);
}
}
}/*}}}*/
}
bool cmpByX(const Point &a,const Point &b){
return a.x<b.x;
}
map<int,vector<int>> xp,xd;
void calcAnswer(){
ll ans=0;
sort(a+1,a+1+acnt,cmpByX);
for(int i=1;i<=acnt;i++){
int cur=a[i].x-a[i].y-stdDis,left;
left=lower_bound(xd[cur].begin(),xd[cur].end(),a[i].x-stdDis)-xd[cur].begin();
ans+=xd[cur].size()-left;
cur=a[i].x+a[i].y-stdDis;
left=lower_bound(xp[cur].begin(),xp[cur].end(),a[i].x-stdDis)-xp[cur].begin();
if(left<xp[cur].size()&&xp[cur][left]==a[i].x-stdDis)
left++;
ans+=xp[cur].size()-left;
xd[a[i].x-a[i].y].pb(a[i].x);
xp[a[i].x+a[i].y].pb(a[i].x);
}
printf("%lld\n",ans);
}
int main(){
readData();
findConnectedSet();
calcAnswer();
return 0;
}

【ARC065E】??的更多相关文章

  1. 【Luogu4931】情侣?给我烧了! 加强版(组合计数)

    [Luogu4931]情侣?给我烧了! 加强版(组合计数) 题面 洛谷 题解 戳这里 忽然发现我自己推的方法是做这题的,也许后面写的那个才是做原题的QwQ. #include<iostream& ...

  2. 【Luogu4921】情侣?给我烧了!(组合计数)

    [Luogu4921]情侣?给我烧了!(组合计数) 题面 洛谷 题解 很有意思的一道题目. 直接容斥?怎么样都要一个平方复杂度了. 既然是恰好\(k\)对,那么我们直接来做: 首先枚举\(k\)对人出 ...

  3. 【5min+】 什么?原来C#还有这两个关键字

    系列介绍 简介 [五分钟的dotnet]是一个利用您的碎片化时间来学习和丰富.net知识的博文系列.它所包含了.net体系中可能会涉及到的方方面面,比如C#的小细节,AspnetCore,微服务中的. ...

  4. 【iOS】Swift ?和 !(详解)

    Swift语言使用var定义变量,但和别的语言不同,Swift里不会自动给变量赋初始值, 也就是说变量不会有默认值,所以要求使用变量之前必须要对其初始化 .如果在使用变量之前不进行初始化就会报错: [ ...

  5. 【app】Hybrid?Native?不知道你就out了!

    Hybrid?是个啥? 相信大家在平常生活中也会经常见到这个词,比如现在比较火的hybrid汽车(混合动力汽车) 那如果是针对于App而言呢? 那就要从App的分类说起了 目前主流应用程序大体分为三类 ...

  6. 为什么要用lock 【readonly】object?为什么不要lock(this)?

    一. 为什么要用lock,lock了什么? 当我们使用线程的时候,效率最高的方式当然是异步,即各个线程同时运行,其间不相互依赖和等待.但当不同的线程都需要访问某个资源的时候,就需要同步机制了.也就是说 ...

  7. 【Scratch】编程?一节课就教会你!其实我们不用一个个学习如何使用代码。

    第199篇文章 老丁的课程 在很多教程里面,大家都喜欢把模块拿出来一个个讲述其功能. 这样做的好处是,可以把每个代码模块的功能讲的很清楚.但最最讨厌的问题也随之而来…… 举个例子,当你学习英语的时候, ...

  8. 【BZOJ-4726】Sabota? 树形DP

    4726: [POI2017]Sabota? Time Limit: 20 Sec  Memory Limit: 128 MBSec  Special JudgeSubmit: 128  Solved ...

  9. CodeForces 【20C】Dijkstra?

    解题思路 heap+Dijkstra就能过.注意边是双向边,要用long long. 附上代码 #include <iostream> #include <queue> #in ...

随机推荐

  1. Android自动化测试之Monkeyrunner使用方法及实例

    目前Android SDK里自带的现成的测试工具有monkey 和 monkeyrunner两个.大家别看这俩兄弟名字相像,但其实是完完全全不同的两个工具,应用在不同的测试领域.总的来说,monkey ...

  2. Luogu P3953 逛公园

    不管怎么说,这都是一道十分神仙的NOIp题 你可以说它狗,但不可以否认它就是NOIp的难度 首先这道题很显然是道图论题还是一道图论三合一(最短路+拓扑+图上DP) 先考虑最短路,我们分别以\(1\)和 ...

  3. PHP生成QRCode二维码

    php生成QRCode二维码示例 <?php //引入 phpqrcode 类库 //phpqrcode下载地址:https://github.com/t0k4rt/phpqrcode //或从 ...

  4. [开源 .NET 跨平台 Crawler 数据采集 爬虫框架: DotnetSpider] [四] JSON数据解析

    [DotnetSpider 系列目录] 一.初衷与架构设计 二.基本使用 三.配置式爬虫 四.JSON数据解析与配置系统 五.如何做全站采集 场景模拟 接上一篇, JD SKU对应的店铺信息是异步加载 ...

  5. Linux系统下本地yum镜像源环境部署-完整记录

    之前介绍了Linux环境下本地yum源配置方法,不过这个是最简单最基础的配置,在yum安装的时候可能有些软件包不够齐全,下面说下完整yun镜像源系统环境部署记录(yum源更新脚本下载地址:https: ...

  6. Python 工程管理及 virtualenv 的迁移

    virtualenv 是管理 python 工程的利器,它可以很好的帮你维护项目中的依赖,使用 virtualenv,还能保持 global 库的干净.不会被不同项目中的第三方库所污染. virtua ...

  7. Echarts学习求教

    有没有人用过百度的Echarts?刚开始接触,下面这段代码怎么理解啊,新手求指教: myChart.showLoading();$.get('data/asset/data/les-miserable ...

  8. Linux 第八周实验 进程的切换和系统的一般执行过程

    姬梦馨 原创作品 <Linux内核分析>MOOC课程:http://mooc.study.163.com/course/USTC-1000029000 第八讲 进程的切换和系统的一般执行过 ...

  9. CMake系列之四:多个源文件-多个目录

    多个源文件,多个目录 现在进一步将MathFunctions.c和MathFunctions.h文件移到math目录下: ./Demo3 | +--- main.c | +--- math/ | +- ...

  10. MySQL: Connection Refused,调整 mysql.ini中的 max_connections

    连接相同的结构的MySQL数据库,一套库Tomcat启动正常,另一套库一直报Connection Refused. 可以断定是连接数太小了.查找mysql.ini中的 max_connections, ...