题解 AtCoder Beginner Contest 168
小兔的话
欢迎大家在评论区留言哦~
A - ∴ (Therefore)
B - ... (Triple Dots)
C - : (Colon)
D - .. (Double Dots)
E - ∙ (Bullet)
简单题意
小兔捕获了 \(N\) 条不同的沙丁鱼,第 \(i\) 条沙丁鱼的 美味程度 和 香味程度 分别是 \(A_i\) 和 \(B_i\)
她想在这些沙丁鱼中选择 一条 或者 多条 放入冷冻箱;但是必须保证沙丁鱼的选择是合格的
(合格的定义:其中的任意两条沙丁鱼 \(i\) 和 \(j\) 都不满足 \(A_i \times A_j + B_i \times B_j = 0\))
小兔想知道有多少种选择沙丁鱼的方法(选择的沙丁鱼的集合相同,算同一种方法),答案对 \(1e9 +7\) 取模
数据范围
\(1 \leq N \leq 2 \times 10^5\)
\(-10^{18} \leq A_i, B_i \leq 10^{18}\)
知识点
- 数学知识
- 最大公约数 \(\mathrm{gcd}\)
- 快速幂
- STL
map
pair
分析
需要不满足的式子与 \(i\) 和 \(j\) 的关系太大了,不妨化简一下:
\]
我们可以把 \(\frac{A_i}{B_i}\) 相同的分成一组,统计出属于这一组的沙丁鱼的数量,再把 \(\frac{A_i}{B_i}\) 和 \(- \frac{B_j}{A_j}\) 的两组分成一对,这一对肯定是互相满足的(就是 \(C\) 与 \(D\) 是一对,反过来 \(D\) 肯定与 \(C\) 是一对,\(D\) 不会和其它成为一对)
我们计算每一对里的选择方案,把所有的选择方案数乘起来再 减一(排除全部不选的情况),就是最终的答案了
如何计算每一对里的选择方案呢?
可以先计算每一对中每组的选择方案,设属于这一组的沙丁鱼有 \(s\) 条,选择沙丁鱼的方案数就是 \(2^s\)(每条鱼有 被选择 和 不被选择 \(2\) 种情况)
那么每一对的方案数就是 \(s_1 + s_2 - 1\)
- 因为其中的两组是不能同时选的,所以是 \(+\) 而不是 \(\times\)
- 因为在统计 \(s_1\) 被选的时候,\(s_2\) 一定是不选的;同理,在统计 \(s_2\) 被选的时候,\(s_1\) 一定是不选的;需要减去这 \(2\) 种多算的情况;又因为 \(2\) 组都不选也是 \(1\) 种合格的情况,所以又要加上 \(1\) 种情况,所以是 \(+1\)
代码
#include <cstdio>
#include <map>
#include <utility>
using namespace std;
#define int long long
int Gcd(int u, int v) { return (v == 0) ? u : Gcd(v, u % v); }
int Max(int u, int v) { return (u > v) ? u : v; }
int Min(int u, int v) { return (u < v) ? u : v; }
int rint()
{
int x = 0, fx = 1; char c = getchar();
while (c < '0' || c > '9') { fx ^= ((c == '-') ? 1 : 0); c = getchar(); }
while ('0' <= c && c <= '9') { x = (x << 3) + (x << 1) + (c ^ 48); c = getchar(); }
if (!fx) return -x;
return x;
}
int qpow(int u, int v, int Mod)
{
int ans = 1; u %= Mod;
while (v)
{
if (v & 1) ans = ans * u % Mod;
u = u * u % Mod; v >>= 1;
}
return ans;
}
const int MOD = 1e9 + 7;
const int MAX_n = 2e5;
int n, ans = 1, sum = 0;
int A[MAX_n + 5];
int B[MAX_n + 5];
map<pair<int, int>, int> G;
map<pair<int, int>, bool> vis;
signed main()
{
n = rint();
for (int i = 1; i <= n; i++)
{
A[i] = rint(), B[i] = rint();
if (A[i] == 0 && B[i] == 0)
{
++sum; --i; --n; continue;
}
int temp = Gcd(A[i], B[i]);
A[i] /= temp; B[i] /= temp;
if (A[i] < 0) { A[i] = -A[i]; B[i] = -B[i]; }
++G[make_pair(A[i], B[i])];
}
for (int i = 1; i <= n; i++)
{
pair<int, int> now = make_pair(A[i], B[i]);
if (-B[i] < 0) { B[i] = -B[i]; A[i] = -A[i]; }
pair<int, int> other = make_pair(-B[i], A[i]);
if (vis[now] || vis[other]) continue;
vis[now] = vis[other] = true;
ans = ans * ((qpow(2, G[now], MOD) + qpow(2, G[other], MOD) - 1) % MOD) % MOD;
}
printf("%lld\n", (ans - 1 + sum + MOD) % MOD);
return 0;
}
F - . (Single Dot)
题解 AtCoder Beginner Contest 168的更多相关文章
- AtCoder Beginner Contest 168
比赛链接:https://atcoder.jp/contests/abc168/tasks A - ∴ (Therefore) 题意 给出一个由数字组成的字符串 $s$,要求如下: 如果 $s$ 以 ...
- Atcoder Beginner Contest 168 D - .. (Double Dots) (BFS)
题意:有\(n\)个房间,在这些房间中两两连\(m\)次条边,问除了第一个房间,其他房间走到第一个房间的最短路径,输出这个房间所连的上一个房间,如果走不到,输出\(no\). 题解:刚开始我写了一个d ...
- [题解] Atcoder Beginner Contest ABC 270 G Ex 题解
点我看题 G - Sequence in mod P 稍微观察一下就会发现,进行x次操作后的结果是\(A^xS+(1+\cdots +A^{x-1})B\).如果没有右边那一坨关于B的东西,那我们要求 ...
- [题解] Atcoder Beginner Contest ABC 265 Ex No-capture Lance Game DP,二维FFT
题目 首先明确先手的棋子是往左走的,将其称为棋子1:后手的棋子是往右走的,将其称为棋子2. 如果有一些行满足1在2右边,也就是面对面,那其实就是一个nim,每一行都是一堆石子,数量是两个棋子之间的空格 ...
- AtCoder Beginner Contest 154 题解
人生第一场 AtCoder,纪念一下 话说年后的 AtCoder 比赛怎么这么少啊(大雾 AtCoder Beginner Contest 154 题解 A - Remaining Balls We ...
- AtCoder Beginner Contest 153 题解
目录 AtCoder Beginner Contest 153 题解 A - Serval vs Monster 题意 做法 程序 B - Common Raccoon vs Monster 题意 做 ...
- AtCoder Beginner Contest 177 题解
AtCoder Beginner Contest 177 题解 目录 AtCoder Beginner Contest 177 题解 A - Don't be late B - Substring C ...
- KYOCERA Programming Contest 2021(AtCoder Beginner Contest 200) 题解
KYOCERA Programming Contest 2021(AtCoder Beginner Contest 200) 题解 哦淦我已经菜到被ABC吊打了. A - Century 首先把当前年 ...
- AtCoder Beginner Contest 184 题解
AtCoder Beginner Contest 184 题解 目录 AtCoder Beginner Contest 184 题解 A - Determinant B - Quizzes C - S ...
随机推荐
- python的数组
- Java枚举类与注解详解——一篇文章读懂枚举类与注解详
目录 一.枚举类 ① 自定义枚举类 ② enum关键字定义枚举类 ③ enum 枚举类的方法 ④ enum 枚举类实现接口 二.注解 ① 生成文档相关注解 ②注解在编译时进行格式检查 ③注解跟踪代码的 ...
- 危险!水很深,让叔来 —— 谈谈命令查询权责分离模式(CQRS)
多年以前,那时我正年轻,做技术如鱼得水,甚至一度希望自己能当一辈子的一线程序员. 但是我又有两个小愿望想要达成:一个是想多挣点钱:另一个就是对项目的技术栈和架构选型能多有点主动权. 多挣点钱是因为当时 ...
- 十一、.net core(.NET 6)搭建ElasticSearch(ES)系列之ElasticSearch、head-master、Kibana环境搭建
搭建ElasticSearch+Kibana环境 前提条件:已经配置好JDK环境以及Nodejs环境.如果还未配置,请查看我的上一篇博客内容,有详细配置教程. 先下载ElasticSearch(以下文 ...
- Django(48)drf请求模块源码分析
前言 APIView中的dispatch是整个请求生命过程的核心方法,包含了请求模块,权限验证,异常模块和响应模块,我们先来介绍请求模块 请求模块:request对象 源码入口 APIView类中di ...
- Error attempting to get column from result set
当使用mybatis plus3.2.0+springboot2.1.1 报错 Error attempting to get column from result set 1.一般出现这种问题,最简 ...
- 目标检测coco数据集点滴介绍
目标检测coco数据集点滴介绍 1. COCO数据集介绍 MS COCO 是google 开源的大型数据集, 分为目标检测.分割.关键点检测三大任务, 数据集主要由图片和json 标签文件组成. c ...
- 使用Keil语言的嵌入式C编程教程(下)
使用Keil语言的嵌入式C编程教程(下) 用8051单片机进行定时器/计数器的计算与编程 延迟是应用软件开发中的重要因素之一.然而,在实现定时延迟的过程中,正常的延迟并不能给出克服这一问题的宝贵结果. ...
- Python集合:set
集合 集合的描述 set是一个无序不重复的序列,可以用{}或者 set() 函数创建集合,它存放不可变类型(如字符串.数字.元组)数据. 注意:创建一个空集合必须使用set()方法,因为{}是用来生成 ...
- pytest xfail的使用
@pytest.mark.xfail: 期望测试用例是失败的,但是不会影响测试用例的的执行; 如果测试用例执行失败的则结果是xfail(不会额外显示出错误信息); 如果测试用例执行成功的则结果是xpa ...