题解【[FJOI2018]所罗门王的宝藏】
本题解同步于luogu
emmm切了近年省选题来写题解啦qwq
该题较其他省选题较水吧(否则我再怎么做的出来
思路是图论做法,做法上楼上大佬已经讲的很清楚了,我来谈谈代码实现上的一些细节
\]
\]
\]
\]
\]
\]
\]
\]
时间复杂度为$$O(T\times(KlogK+K)) = O(TNlogN)$$要改进也行,因为我们对于每个点所连边只要边权最小数所以没必要sort,但当我想到这一点时已经AC本题~
\]
#include<iostream>
#include<cstdio>
#include<vector>
#include<algorithm>
#include<cstring>
#define MAXN 1005
using namespace std ;
inline void read(int &x) {
scanf("%d",&x) ;
}
class getsol {
public:
//========data========
vector<pair<int,int> > edge[MAXN*2] ; //pair第一维是边权,第二维是到达边的编号
int n , m , k , onk[MAXN*2] , inq[MAXN*2] , flag ;//inq表示是否被搜到
//========func========
void add(int x,int y,int v) {
edge[x].push_back(make_pair(v,y)) ; //加边
}
bool check(int u,int v,int w) {
//check , 判断v点是否可行
if(onk[u]+onk[v]!=w) return 0 ;
return 1 ;
}
void dfs(int D) {
//cout<<"DFS : START SEARCH IN DOT "<<D<<endl ;
if(flag==0) return ;
inq[D] = 1 ;
for(auto& i : edge[D]) { //对于每个edge[D]中的元素i
///cout<<"DFS : SEARCH IN DOT "<<i.second<<endl ;
if(flag==0) return ;
//cout<<"In dot : "<<i.second<<endl ;
int ver = i.second , edgeval = i.first ;
if(inq[ver]) {
if(flag==1) //如果答案还是"Yes"那么更新,这里是一个优化~
flag = check(D,ver,edgeval) ;
continue ;
} else {
onk[ver] = edgeval-onk[D] ;
dfs(ver) ;
}
}
}
void PRINT(int* arr,int n) {
for(int i=1; i<=n; ++i) {
cout<<"arr["<<i<<"] = "<<arr[i]<<endl ;
}
}
void sol() {
flag = true ;
read(n) , read(m) , read(k) ;
//行的编号为1~n
//列的编号为(n+1)~(2*n)
//喵~
for(int i=1; i<=k; ++i) {
int x,y,v ;
read(x) , read(y) , read(v) ;
add(x,y+n,v) ;
add(y+n,x,v) ;
}
//cerr<<"FINISH READ"<<endl ;
for(int i=1; i<=2*n; ++i) sort(edge[i].begin(),edge[i].end()) ;
//cerr<<"FINISH SORT"<<endl ;
for(int i=1; i<=2*n; ++i) {
if(!inq[i]&&flag&&!edge[i].empty()) { // 注意这里判一下vector是否为空。。因为这个RE了两三次
onk[i] = (*edge[i].begin()).first ;
//cerr<<"SEARCH IN DOT "<<i<<endl ;
dfs(i) ;
}
}
if(flag==1) {
for(int i=1; i<=2*n; ++i)
for(auto& j : edge[i])
if(flag==1) //重新check一遍,以免遗漏
flag = check(i,j.second,j.first) ;
}
if(flag) puts("Yes") ;
else puts("No") ;
//PRINT(onk,2*n) ;
}
void clear() {
for(int i=1; i<=2*n; ++i) edge[i].clear() ;
memset(inq,0,sizeof(onk)) ;
memset(onk,0,sizeof(onk)) ;
n = m = k = flag = 0 ;
}
} ;
getsol M ;
int T ;
int main() {
//freopen("solo3.in" , "rb" , stdin) ;
//freopen("solo3.out", "wb" ,stdout) ;
read(T) ;
while(T--) M.sol() , M.clear() ;
}
注意本代码是使用C++11标准写成,代码中不同不同语法处已标注
题解【[FJOI2018]所罗门王的宝藏】的更多相关文章
- 【BZOJ5470】[FJOI2018]所罗门王的宝藏()
[BZOJ5470][FJOI2018]所罗门王的宝藏() 题面 BZOJ 洛谷 有\(n+m\)个变量,给定\(k\)组限制,每次告诉你\(a_i+b_j=c_k\),问是否有可行解. 题解 一道很 ...
- bzoj5470 / P4578 [FJOI2018]所罗门王的宝藏
P4578 [FJOI2018]所罗门王的宝藏 设第$i$行上的值改变了$r1[i]$,第$j$列上的值改变了$r2[i]$ 显然密码$(i,j,c)=r1[i]+r2[j]$ 对于同一列上的两个密码 ...
- 洛谷4578 & LOJ2520:[FJOI2018]所罗门王的宝藏——题解
https://www.luogu.org/problemnew/show/P4578 https://loj.ac/problem/2520 有点水的. 先转换成图论模型,即每个绿宝石,横坐标向纵坐 ...
- 洛谷P4578 [FJOI2018]所罗门王的宝藏(dfs)
题意 题目链接 Sol 对于每个询问\(x, y, c\) 从在\((x, y)\)之间连一条边权为\(c\)的双向边,然后就是解\(K\)个二元方程. 随便带个数进去找找环就行了 #include& ...
- P4578 [FJOI2018]所罗门王的宝藏
传送门 考虑一个位置答案传递性,如果某个位置的红宝石转动确定了,那么会引起连锁反应: 如图,绿色的转动确定了,那么那两个蓝色的转动也确定了 自己手玩一下,发现如果有解那么随便找一个开始然后一路玩下去最 ...
- luoguP4578_ [FJOI2018]所罗门王的宝藏
题意 一个n*m的矩阵,初始值全为0,每一行每一列操作一次可以加1或者减1,问能否操作得到给定矩阵. 分析 行和列的分别的加减是可以相互抵消的,因此我们只需要考虑行的加和列的减. 对于给定矩阵每一个数 ...
- 【LOJ】 #2520. 「FJOI2018」所罗门王的宝藏
题解 发现似乎相当于问一个2000个元的方程组有没有解-- 然而我懵逼啊-- 发现当成图论,两个点之间连一条边,开始BFS,每个点的值赋成边权减另一个点的点权 如果一个环不合法那么肯定无解 代码 #i ...
- 【题解】P3959 宝藏 - 状压dp / dfs剪枝
P3959 宝藏 题目描述 参与考古挖掘的小明得到了一份藏宝图,藏宝图上标出了 n 个深埋在地下的宝藏屋, 也给出了这 n 个宝藏屋之间可供开发的m 条道路和它们的长度. 小明决心亲自前往挖掘所有宝 ...
- 题解-FJOI2018 领导集团问题
题面 FJOI2018 领导集团问题 给一棵树 \(T(|T|=n)\),每个点有个权值 \(w_i\),从中选出一个子点集 \(P=\{x\in {\rm node}|x\in T\}\),使得 \ ...
随机推荐
- UVALive - 7752 Free Figurines
题意:有n个娃娃,如果大娃娃j直接套小娃娃i,则fa[i] = j.若fa[i] = 0,则该娃娃自由.给出每个娃娃初始的父亲,和改变后的父亲,在满足以下合法操作的条件下,问最少需要多少次变换. 1. ...
- AS-PATH(路径属性)路由路径欺骗术
AS-PATH(路径属性)路由路径欺骗术: ①:抓取感兴趣流量——前缀与访问 ②:创建路由地图 ③:路由地图第一法则——permit 10 ④:在第一法则中,匹配(感兴趣流量) ⑤:设置 路径欺骗术— ...
- STL--迭代器设计原则和萃取机制(Traits)
title: C++ STL迭代器设计原则和萃取机制(Traits) date: 2019-12-23 15:21:47 tags: STL C/C++ categories: STL 迭代器 (it ...
- C#中类的字段或属性不被序列化成JSON或XML
将一个类序列化成JSON或XML时,如果某个字段或属性不想被序列化,则可以使用以下Attribute: 1.[Newtonsoft.Json.JsonIgnore]特性:使用Newtonsoft.Js ...
- POJ 1177/HDU 1828 picture 线段树+离散化+扫描线 轮廓周长计算
求n个图矩形放下来,有的重合有些重合一部分有些没重合,求最后总的不规则图型的轮廓长度. 我的做法是对x进行一遍扫描线,再对y做一遍同样的扫描线,相加即可.因为最后的轮廓必定是由不重合的线段长度组成的, ...
- 吴裕雄--天生自然C++语言学习笔记:C++ 标准库
C++ 标准库可以分为两部分: 标准函数库: 这个库是由通用的.独立的.不属于任何类的函数组成的.函数库继承自 C 语言. 面向对象类库: 这个库是类及其相关函数的集合. C++ 标准库包含了所有的 ...
- axios实现类似form传值的格式,以及实现拦截器功能,response拦截实现权限判断
import axios from 'axios' import Qs from 'qs' // 超时设置 const service = axios.create({ transformReques ...
- js模式-观察者模式
// 主题,接收状态变化,触发每个观察者 class Subject { constructor() { this.state = 0 this.observers = [] } getState() ...
- CodeForces - 402B Trees in a Row (暴力)
题意:给定n个数,要求修改其中最少的数,使得这n个数满足ai + 1 - ai = k. 分析: 暴力,1000*1000. 1.这n个数,就是一个首项为a1,公差为k的等差数列.k已知,如果确定了a ...
- 用Python分析淘宝2000款避孕套,得出这些有趣的结论
数据分析之前我们需要清楚的知道自己想要分析什么东西,也就是先搞清楚我们的目标.在公司可能是公司财报.用户增量变化.产品受欢迎程度.一些报表等等. 那我们今天的目标有哪些呢?我们来看看: ! 分析避孕套 ...