题目链接

题目

题目描述

现有一个传动系统,包含了N个组合齿轮和M个链条。每一个链条连接了两个组合齿轮u和v,并提供了一个传动比x : y。

即如果只考虑这两个组合齿轮,编号为u的齿轮转动x圈,编号为v的齿轮会转动y圈。传动比为正表示若编号为u的齿轮顺时针转动,则编号为v的齿轮也顺时针转动。传动比为负表示若编号为u的齿轮顺时针转动,则编号为v的齿轮会逆时针转动。若不同链条的传动比不相容,则有些齿轮无法转动。我们希望知道,系统中的这N个组合齿轮能否同时转动。

输入描述

有多组数据,第一行给定整数T,表示总的数据组数,之后依次给出T组数据。

每一组数据的第一行给定整数N和M,表示齿轮总数和链条总数。

之后有M行,依次描述了每一个链条,其中每一行给定四个整数u,v,x和y,表示只考虑这一组联动关系的情况下,编号为u的齿轮转动x圈,编号为v的齿轮会转动y圈。

请注意,x为正整数,而y为非零整数,但是y有可能为负数。

T ≤ 32,N ≤ 1000,M ≤ 10000且x与y的绝对值均不超过100

输出描述

输出T行,对应每一组数据。首先应该输出标识这是第几组数据,参见样例输出。之后输出判定结果,如果N个组合齿轮可以同时正常运行,则输出Yes,否则输出No。

示例1

输入

2
3 3
1 2 3 5
2 3 5 -7
1 3 3 -7
3 3
1 2 3 5
2 3 5 -7
1 3 3 7

输出

Case #1: Yes
Case #2: No

题解

方法一

知识点:并查集。

用带权并查集维护齿轮之间的关系,用权值表示根节点转一圈,这个节点转的圈数。

路径压缩,将父节点到根节点的权值乘以自己的权值。

集合合并,对于节点 \(x\) 和 \(y\) 以及其根节点 \(rx\) 和 \(ry\) ,将 \(rx\) 合并到 \(ry\) 需要得到 \(rx\) 的新权值,即得到 \(rx\) 和 \(ry\) 的传动比,有:

\[w_{rx}' = \frac{1}{w_{x}} \cdot \frac{x}{y} \cdot \frac{w_{y}}{1} \cdot 1
\]

即四个齿轮的三个传动比相乘得到 \(rx\) 和 \(ry\) 的传动比再将 \(ry\) 圈数设为 \(1\) ,于是 \(w_{rx}\) 就等于 \(rx\) 与 \(ry\) 的传动比乘以 \(ry\) 的圈数 \(1\) 。

如果新加关系的两个点已经在一个关系集合中,那就检验是否合法:

\[\frac{w_{x}}{w_{y}} = \frac{x}{y}
\]

两者传动比是否相等,相等则合法。

注意精度问题,相等用小于误差表示。

时间复杂度 \(O(n + m\log n)\)

空间复杂度 \(O(n)\)

方法二

知识点:DFS,图论。

和并查集思路差不多,但建图时要建无向图,因为需要直到可能遍历时走的是反向的。给起点权值赋为 \(1\) ,其他节点根据传动比赋值,上一个节点乘以这个方向的传动比的倒数的结果即是这个点实际转多少圈。

如果遇到遍历到一个访问过的节点,那就判断实际权值和目前算出来的权值是否相等。

时间复杂度 \(O(n+m)\)

空间复杂度 \(O(n+m)\)

代码

方法一

#include <bits/stdc++.h>
#define ll long long using namespace std; int n, m;
int fa[10007];
double w[10007]; int find(int x) {
if (fa[x] == x) return x;
int pre = fa[x];
fa[x] = find(fa[x]);
w[x] *= w[pre];
return fa[x];
} bool merge(int x, int y, double r) {
int rx = find(x);
int ry = find(y);
if (rx == ry)
return abs(w[x] / w[y] - r) < 1e-6;
fa[rx] = ry;
w[rx] = 1 / w[x] * r * w[y];
return true;
} bool solve() {
cin >> n >> m;
for (int i = 1;i <= n;i++) fa[i] = i, w[i] = 1;
bool flag = true;
for (int i = 0;i < m;i++) {
int u, v;
double x, y;
cin >> u >> v >> x >> y;
if (!flag) continue;
flag &= merge(u, v, x / y);
}
return flag;
} int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int t = 1;
cin >> t;
for (int i = 1;i <= t;i++) {
cout << "Case #" << i << ": ";
if (!solve()) cout << "No" << '\n';
else cout << "Yes" << '\n';
}
return 0;
}

方法二

#include <bits/stdc++.h>
#define ll long long using namespace std; int n, m;
struct edge {
int to, nxt;
double w;
}e[10007 << 1];
int h[1007], cnt;
double vis[1007]; void add(int u, int v, double w) {
e[cnt].to = v;
e[cnt].w = w;
e[cnt].nxt = h[u];
h[u] = cnt++;
} bool dfs(int u) {
for (int i = h[u];~i;i = e[i].nxt) {
int v = e[i].to;
if (vis[v]) {
if (abs(vis[v] - vis[u] / e[i].w) > 1e-6) return false;
}
else {
vis[v] = vis[u] / e[i].w;
if (!dfs(v)) return false;
}
}
return true;
} bool solve() {
cin >> n >> m;
for (int i = 1;i <= n;i++) h[i] = -1, vis[i] = 0;
cnt = 0;
for (int i = 0;i < m;i++) {
int u, v;
double x, y;
cin >> u >> v >> x >> y;
add(u, v, x / y);
add(v, u, y / x);
}
vis[1] = 1;
return dfs(1);
} int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int t = 1;
cin >> t;
for (int i = 1;i <= t;i++) {
cout << "Case #" << i << ": ";
if (!solve()) cout << "No" << '\n';
else cout << "Yes" << '\n';
}
return 0;
}

NC20583 [SDOI2016]齿轮的更多相关文章

  1. [Sdoi2016]齿轮

    4602: [Sdoi2016]齿轮 Time Limit: 10 Sec  Memory Limit: 512 MB Submit: 613  Solved: 324 [Submit][Status ...

  2. BZOJ 4602: [Sdoi2016]齿轮 dfs

    4602: [Sdoi2016]齿轮 题目连接: http://www.lydsy.com/JudgeOnline/problem.php?id=4602 Description 现有一个传动系统,包 ...

  3. BZOJ4602 Sdoi2016 齿轮 【带权并查集】*

    BZOJ4602 Sdoi2016 齿轮 Description 现有一个传动系统,包含了N个组合齿轮和M个链条.每一个链条连接了两个组合齿轮u和v,并提供了一个传动比x : y.即如果只考虑这两个组 ...

  4. bzoj 4602: [Sdoi2016]齿轮

    4602: [Sdoi2016]齿轮 Description 现有一个传动系统,包含了N个组合齿轮和M个链条.每一个链条连接了两个组合齿轮u和v,并提供了一个传动比x  : y.即如果只考虑这两个组合 ...

  5. BZOJ4602:[SDOI2016]齿轮(并查集)

    Description 现有一个传动系统,包含了N个组合齿轮和M个链条.每一个链条连接了两个组合齿轮u和v,并提供了一个传动比x  : y.即如果只考虑这两个组合齿轮,编号为u的齿轮转动x圈,编号为v ...

  6. 【bzoj4602】[Sdoi2016]齿轮 BFS

    题目描述 给出一张n个点m条边的有向图,每条边 (u,v,x,y) 描述了 u 的点权乘 x 等于 v 的点权乘 y (点权可以为负).问:是否存在满足条件的图. 输入 有多组数据,第一行给定整数T, ...

  7. BZOJ4602 SDOI2016齿轮(搜索)

    dfs一遍给每个齿轮随便标个值看是否矛盾就行了. #include<iostream> #include<cstdio> #include<cmath> #incl ...

  8. bzoj4602 [Sdoi2016]齿轮

    传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=4602 [题解] 对于每组齿轮(u, v)连边,权值为y/x(反向边x/y) 那么直接dfs计 ...

  9. [bzoj4602][Sdoi2016]齿轮——dfs

    题目 现有一个传动系统,包含了N个组合齿轮和M个链条.每一个链条连接了两个组合齿轮u和v,并提供了一个传动比x : y.即如果只考虑这两个组合齿轮,编号为u的齿轮转动x圈,编号为v的齿轮会转动y圈.传 ...

随机推荐

  1. 利用expect批量修改Linux服务器密码

    一个执着于技术的公众号 背景 修改Linux系统密码,执行passwd即可更改密码.可如果有成千上百台服务器呢,通过ssh的方式逐一进行修改,对我们来说,工作量是非常大,且效率非常低下.因此采用批量修 ...

  2. 在SpringBoot中使用logback优化异常堆栈的输出

    一.背景 在我们在编写程序的过程中,无法保证自己的代码不抛出异常.当我们抛出异常的时候,通常会将整个异常堆栈的信息使用日志记录下来.通常一整个异常堆栈的信息是比较多的,而且存在一些没用的信息.那么我们 ...

  3. Python生成GIF动态图

    python生成摸头GIF 本篇教程演示了如何使用python的PIL库生成GIF图片 源码已经贴在文中,自行取用 效果演示 运行代码,会让你选择要制作的图片 运行完成后,会在同路径下生成dem.gi ...

  4. 1┃音视频直播系统之浏览器中通过WebRTC访问摄像头

    一.WebRTC的由来 对于前端开发小伙伴而言,如果用 JavaScript 做音视频处理 在以前是不可想象的,因为首先就要考虑浏览器的性能是否跟得上音视频的采集 但是 Google 作为国际顶尖科技 ...

  5. Java学习笔记-基础语法Ⅶ-集合

    集合 集合类特点:提供一种存储空间可变的存储模型,存储的数据容量可以随时发生改变 这里需要回顾一下,因为数组和字符串一旦创建,就不可改变,需要区分一下 import java.util.ArrayLi ...

  6. CTF简介

    最近在学习渗透测试,后来发现CTF很有趣,发现对学习有所帮助,于是找了几个网站,下面推荐几个我觉得不错的网站 https://www.ctfhub.com/#/index https://adworl ...

  7. arthas学习图文记录

    Arthas 是阿里开源的 Java 诊断工具.在线排查问题,无需重启:动态跟踪 Java 代码:实时监控 JVM 状态.Arthas 支持 JDK 6+,支持 Linux/Mac/Windows,采 ...

  8. 初始 Django

    Python 知识点:函数,面向对象 前端开发:HTML,CSS,JavaScript,jQuery,BootStrap MySQL 数据库 Python 的 WEB 框架 Flask:轻量化,第三方 ...

  9. 好客租房6-React脚手架的应用(细节)

    3.2使用react脚手架初始化 npx命令介绍 npm v5.2.0引入的一条命令 目的:提升包内提供的命令行工具的使用体验 原先:先安装脚手架包 再使用这个包中提供的命令 现在无需安装脚手架包 就 ...

  10. 『忘了再学』Shell基础 — 21、变量的测试与内容置换

    目录 1.什么是变量的测试与内容置换 2.变量的测试与内容置换 3.示例 例1: 例2: 例3: 1.什么是变量的测试与内容置换 我们之前说过,在Shell中,一个变量未定义,和一个变量为空值的输出效 ...