link。

点 \(A\) 与 \((0,0)\),\(B\) 共线的充要条件是 \(\frac{y_A}{x_A}=\frac{y_B}{x_B}\),即 \(k_{OA}=k_{OB}\)。又考虑到题目提出刻画斜率相等双点间的关系,所以不妨把所有斜率相同的点看作一个。再考虑刻画点的移动,由于与共线的点是移动后两者之间的哪一者无妨,所以我们可以在移动后的两点所代表的斜率集合之间连一条边,问题就转化成了在一张无向图中,删除或一条三点二边的链,或一个二点二边的环,询问最多可以删除多少次,并给出可行方案。那么答案中最大值的部分我们可以拿出来,即 \(\lfloor\frac{\text{\# edges}}{2}\rfloor\)。

论删边的顺序,我们可以建出转化后图的任一生成树,并考虑非树边。考虑任一结点 \(x\),设有非树边边 \(\lang x,y\rang\),我们可以将 \(y\) 给“拖下去”,意即新建一个点,并将 \(x\) 连向该点。其正确性并非自明,但是考虑深度可以简单证明。至于答案的求解过程,参见常见 trick 树的最大匹配(但是略有不同,具体见代码)。

说一下如何精简实现,你的代码逻辑可以不是「建出原图 - 得到生成树 - 新建节点 - 求匹配」,更加优秀的逻辑可以是「建出原图并通过动态维护连通性新建节点 - 在求出生成树的同时获得深度信息 - 求匹配」。

#include <bits/stdc++.h>
using namespace std;
using ll = long long;
int n, rn, uset[500100], head[500100], to[1000100], nxt[1000100], ent;
ll gcd(ll x, ll y) {
return y == 0 ? x : gcd(y, x%y);
}
int ldr(int x) {
while (x != uset[x]) x = uset[x] = uset[uset[x]];
return x;
}
bool sm(int x, int y) {
return ldr(x) == ldr(y);
}
void mrg(int x, int y) {
if (ldr(x) != ldr(y)) {
uset[ldr(y)] = ldr(x);
}
}
struct frac {
ll p, q;
frac() {}
explicit frac(ll a, ll b) : p(a), q(b) {
norm(*this);
}
bool operator<(const frac& o) const {
return p < o.p || (p == o.p && q < o.q);
}
void norm(frac& x) {
ll g = gcd(x.p, x.q);
x.p /= g, x.q /= g;
}
};
map<frac, int> mp;
void add(int x, int y) {
to[++ent] = y, nxt[ent] = head[x], head[x] = ent;
}
bool vis[500100];
int dep[500100], fa[500100], eid[500100], mxd;
set<int> ch[500100], sd[500100];
void dfs(int x) {
mxd = max(mxd, dep[x]);
vis[x] = 1;
sd[dep[x]].insert(x);
for (int i = head[x], y; i; i = nxt[i]) {
if (vis[y = to[i]]) {
continue;
}
fa[y] = x, dep[y] = dep[x]+1;
ch[x].insert(y);
eid[y] = (i+1)/2;
dfs(y);
}
}
signed main() {
ios::sync_with_stdio(0);
cin.tie(0);
cin >> n;
for (int i = 1; i <= n; ++i) {
ll a, b, c, d;
cin >> a >> b >> c >> d;
frac cur((a+b)*d, b*c);
if (!mp.count(cur)) {
++rn;
mp[cur] = uset[rn] = rn;
}
int u = mp[cur], v;
cur = frac(a*d, b*(c+d));
if (!mp.count(cur)) {
++rn;
mp[cur] = uset[rn] = rn;
}
v = mp[cur];
if (sm(u, v)) {
v = ++rn;
}
else mrg(u, v);
add(u, v), add(v, u);
}
for (int i = 1; i <= rn; ++i) {
if (!vis[i]) dfs(i);
}
vector<pair<int, int>> ans;
for (int d = mxd; d >= 1; --d) {
for (int x : sd[d]) {
if (int(ch[fa[x]].size()) >= 2) {
ch[fa[x]].erase(x);
int bro = *ch[fa[x]].begin();
ch[fa[x]].erase(bro), sd[dep[bro]].erase(bro);
ans.emplace_back(eid[x], eid[bro]);
}
else if (fa[fa[x]]) {
ch[fa[x]].erase(x);
ch[fa[fa[x]]].erase(fa[x]);
sd[dep[fa[x]]].erase(fa[x]);
ans.emplace_back(eid[x], eid[fa[x]]);
}
}
}
cout << ans.size() << "\n";
for (auto it : ans) {
cout << it.first << " " << it.second << "\n";
}
}

「codeforces - 1519E」Off by One的更多相关文章

  1. 「CodeForces 581D」Three Logos

    BUPT 2017 Summer Training (for 16) #3A 题意 给你三个矩形,需要不重叠不留空地组成一个正方形.不存在输出-1,否则输出边长和这个正方形(A,B,C表示三个不同矩形 ...

  2. 「CodeForces - 50C 」Happy Farm 5 (几何)

    BUPT 2017 summer training (16) #2B 题意 有一些二维直角坐标系上的整数坐标的点,找出严格包含这些点的只能八个方向走出来步数最少的路径,输出最少步数. 题解 这题要求严 ...

  3. 「CodeForces - 598B」Queries on a String

    BUPT 2017 summer training (for 16) #1I 题意 字符串s(1 ≤ |s| ≤ 10 000),有m(1 ≤ m ≤ 300)次操作,每次给l,r,k,代表将r位置插 ...

  4. 「CodeForces - 717E」Paint it really, really dark gray (dfs)

    BUPT 2017 summer training (for 16) #1H 题意 每个节点是黑色or白色,经过一个节点就会改变它的颜色,一开始在1节点.求一条路径使得所有点变成黑色. 题解 dfs时 ...

  5. 「CodeForces 476A」Dreamoon and Stairs

    Dreamoon and Stairs 题意翻译 题面 DM小朋友想要上一个有 \(n\) 级台阶的楼梯.他每一步可以上 \(1\) 或 \(2\) 级台阶.假设他走上这个台阶一共用了 \(x\) 步 ...

  6. 「CodeForces 546B」Soldier and Badges 解题报告

    CF546B Soldier and Badges 题意翻译 给 n 个数,每次操作可以将一个数 +1,要使这 n 个数都不相同, 求最少要加多少? \(1 \le n \le 3000\) 感谢@凉 ...

  7. 「Codeforces 79D」Password

    Description 有一个 01 序列 \(a_1,a_2,\cdots,a_n\),初始时全为 \(0\). 给定 \(m\) 个长度,分别为 \(l_1\sim l_m\). 每次可以选择一个 ...

  8. 「Codeforces 468C」Hack it!

    Description 定义 \(f(x)\) 表示 \(x\) 的各个数位之和.现在要求 \(\sum_{i=l}^rf(i)\bmod a\). 显然 ans=solve(l,r)%a; if(a ...

  9. 「Codeforces 724F」Uniformly Branched Trees

    题目大意 如果两棵树可以通过重标号后变为完全相同,那么它们就是同构的. 将中间节点定义为度数大于 \(1\) 的节点.计算由 \(n\) 个节点,其中所有的中间节点度数都为 \(d\) 的互不同构的树 ...

  10. 「codeforces - 1284G」Seollal

    给定 \(n\times m\) 的网格图,有些格子有障碍,无障碍且相邻的格子之间连边形成图.保证 \((1, 1)\) 无障碍,保证无障碍格子连通. 将网格图黑白染色,相邻格子颜色不同,\((1, ...

随机推荐

  1. 通过redis学网络(1)-用go基于epoll实现最简单网络通信框架

    本系列主要是为了对redis的网络模型进行学习,我会用golang实现一个reactor网络模型,并实现对redis协议的解析. 系列源码已经上传github https://github.com/H ...

  2. RT_object

    以下图片来自"张世争"的微博  

  3. Windows11右键菜单修改为Win10模式的方法

    Windows11右键菜单修改为Win10模式的方法 自述: 更新win11后看着鼠标右键的菜单有些不太舒服,索性就改回了win10的右键菜单的样式 , 下面开始进行操作 第一步 首先以管理员方式打开 ...

  4. 一次oracle行级锁导致的问题

    分析问题:我在plsql/developer是用的system用户连接的数据库,而在crt用  sqlplus / as  sysdba 连接数据库,是sys用户.现在在plsql/developer ...

  5. 基于FFMPEG+SDL的简单的视频播放器分析

    基于FFMPEG+SDL的简单的视频播放器分析 前言 最近看了雷霄骅前辈的博客<最简单的基于FFMPEG+SDL的视频播放器 ver2 (采用SDL2.0)>,参照他的代码,在window ...

  6. Redis缓存同步1-策略介绍

    缓存数据同步策略示意图 在大多数情况下,我们通过浏览器查询到的数据都是缓存数据,如果缓存数据与数据库的数据存在较大差异的话,可能会产生比较严重的后果的.所以,我们应该也必须保证数据库数据.缓存数据的一 ...

  7. linux内核vmlinux的编译过程之 --- vmlinux.o详解(八)

    内核构建系统之所以要在链接 vmlinux 之前,去链接出vmlinux.o.其原因并不是要将 vmlinux.o 链接进 vmlinux,而是要在链接 vmlinux.o 的过程中做完两个动作: e ...

  8. 2023icpc大学生程序设计竞赛-zx

    第一次出市打线下,洛阳师范风景不错,就是比赛的筹备有点波折.题目在这几次xcpc省赛算是比较难的.开始一个二维前缀和板子以及一个小贪心还是顺利拿下,后面那个dp一直是我们的短板wa了几发才过,后面就是 ...

  9. CRM系统化整合从N-1做减法实践

    1 背景 京销易系统已经接入大网.KA以及云仓三个条线商机,每个条线商机规则差异比较大,当前现状是独立实现三套系统分别做支撑. 2 目标 2022年下半年CRM目标是完成9个新条线业务接入,完成销售过 ...

  10. npm install -g 错误集锦

    1.切换源安装:npm config set registry http://registry.cnpmjs.org,参考http://yijiebuyi.com/blog/b12eac891cdc5 ...