题解-Quantifier Question
有长度为 \(n\) 的序列 \(x\{n\}\),有 \(m\) 个条件 \((j_i,k_i)\)。有 \(n\) 个待定的条件符 \(Q_i\in\{\forall,\exists\}\),使
\[Q_1x_1,Q_2x_2,...,Q_nx_n,(x_{j_1}<x_{k_1})∧(x_{j_2}<x_{k_2})∧\cdots∧(x_{j_m}<x_{k_m})
\]求 \(\forall\) 最多的方案。如果没有满足方案,输出 \(-1\)。否则输出 \(\forall\) 的数量及整个条件符串。
数据范围:\(2\le n\le 2\cdot 10^5\),\(1\le m\le 2\cdot 10^5\),\(1\le j_i,k_i\le n,j_i\not=k_i\)。
一句话题解:拓扑排序,有环输出 \(-1\),否则将每个不被前后节点影响的节点置为 \(\forall\)。
首先理解一下,不同序号的 \(Q_i\) 是有顺序的,不是平等的。如 \(x_1<x_2\),\(x_1<x_3\),令 \(Q_1=\exists,Q_2=\forall,Q_3=\forall\) 是不可以的(如选定 \(x_1\) 后取 \(x_2=x_1-1\) 就爆了);但是如果 \(x_3<x_1,x_3<x_2\),令 \(Q_1=\exists,Q_2=\forall,Q_3=\forall\) 是可以的。
先用差分约束的思想,把条件转化为边,即连 \(j_i\to k_i\)。然后拓扑排序。如果拓扑排序序列长度 \(<n\),说明有环,无解,输出 \(-1\)。否则存下这个拓扑序列。
设 \(u<v\),因为先发挥 \(u\) 的条件符再发挥 \(v\) 的条件符。所以如果 \(u\) 可以走到 \(v\) 或者 \(v\) 可以走到 \(u\),那么 \(v\) 点就不能选 \(\forall\) 了;反之,如果对于 \(v\),不存在 \(u<v\) 可以走到它或被它走到,这样的节点条件符选 \(\forall\) 是最优且没有后效性的。
然后用类似 \(\texttt{dp}\) 的方法,求出每个节点的最小前驱(可以走到它)和最小后继(它可以走到),然后如果最小前驱和最小后继都不小于该节点编号,那么该节点的条件符选 \(\forall\)。
时间复杂度 \(\Theta(n+m)\)。
代码:
#include <bits/stdc++.h>
using namespace std;
//Start
typedef long long ll;
typedef double db;
#define mp(a,b) make_pair(a,b)
#define x(a) a.first
#define y(a) a.second
#define b(a) a.begin()
#define e(a) a.end()
#define sz(a) int((a).size())
#define pb(a) push_back(a)
const int inf=0x3f3f3f3f;
const ll INF=0x3f3f3f3f3f3f3f3f;
//Data
const int N=2e5;
int n,m,d[N+7],f[N+7],b[N+7],sm,ans[N+7];
vector<int> e[N+7],g[N+7];
//Bfs
int Bfs(vector<int>&q){ //就是拓扑排序
//将图的拓扑序存入q,如果形成了环就返回0。
q.clear();
for(int i=1;i<=n;i++)if(!d[i]) q.pb(i);
for(int i=0;i<sz(q);i++)
for(int to:e[q[i]])if(!--d[to]) q.pb(to);
return sz(q)==n;
}
//Main
int main(){
scanf("%d%d",&n,&m);
for(int i=1,u,v;i<=m;i++) scanf("%d%d",&u,&v),e[u].pb(v),g[v].pb(u);
for(int i=1;i<=n;i++) d[i]=sz(g[i]); //d为入度
vector<int> tp;
if(!Bfs(tp)) return puts("-1"),0;
iota(f+1,f+n+1,1),iota(b+1,b+n+1,1); //将fi:=i,bi:=i。
for(int i=0;i<=sz(tp)-1;i++)
for(int to:g[tp[i]]) f[tp[i]]=min(f[tp[i]],f[to]); //最小前驱
for(int i=sz(tp)-1;i>=0;i--)
for(int to:e[tp[i]]) b[tp[i]]=min(b[tp[i]],b[to]); //最小后继
for(int i=1;i<=n;i++)if(min(f[i],b[i])==i) sm++,ans[i]=1; //讲解中说了
printf("%d\n",sm);
for(int i=1;i<=n;i++) putchar("EA"[ans[i]]);puts("");
return 0;
}
祝大家学习愉快!
题解-Quantifier Question的更多相关文章
- CF R639 div 2 E Quantifier Question 数学 dfs 图论
LINK:Quantifier Question 题面过长 引起不适 读题花了好长时间 对于 和 存在符合不是很熟练 导致很懵逼的做完了. 好在还算很好想.不过wa到了一个坑点上面 自闭一大晌 还以为 ...
- HDU 5793 A Boring Question (逆元+快速幂+费马小定理) ---2016杭电多校联合第六场
A Boring Question Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others ...
- noip2007提高组题解
题外话:这一年的noip应该是最受大众关心的,以至于在百度上输入noip第三个关键字就是noip2007.主要是由于这篇文章:http://www.zhihu.com/question/2110727 ...
- 多校6 1001 HDU5793 A Boring Question (推公式 等比数列求和)
题解:http://bestcoder.hdu.edu.cn/blog/ 多校6 HDU5793 A Boring Question // #pragma comment(linker, " ...
- 2016多校第六场题解(hdu5793&hdu5794&hdu5795&hdu5800&hdu5802)
这场就做出一道题,怎么会有窝这么辣鸡的人呢? 1001 A Boring Question(hdu 5793) 很复杂的公式,打表找的规律,最后是m^0+m^1+...+m^n,题解直接是(m^(n+ ...
- hdu_5793_A Boring Question(打表找规律)
题目链接:hdu_5793_A Boring Question 题意: 自己看吧,说不清楚了. 题解: 打表找规律 #include<cstdio> typedef long long l ...
- LeetCode题解33.Search in Rotated Sorted Array
33. Search in Rotated Sorted Array Suppose an array sorted in ascending order is rotated at some piv ...
- leetcode & lintcode 题解
刷题备忘录,for bug-free 招行面试题--求无序数组最长连续序列的长度,这里连续指的是值连续--间隔为1,并不是数值的位置连续 问题: 给出一个未排序的整数数组,找出最长的连续元素序列的长度 ...
- HDU 5793 A Boring Question (找规律 : 快速幂+乘法逆元)
A Boring Question Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others ...
随机推荐
- rgw前端替换civetweb为beast
前言 ceph的rgw现在提供了两种前端, civetweb和beast 配置 修改配置文件 rgw_frontends = civetweb port=7481 为 rgw frontends = ...
- Python_爬虫_基础
1.urllib 和 Xpath的区别与联系 from urllib import request from lxml import etree from bs4 import BeautifulS ...
- mysql之用户
1.通过Navicat For Mysql可以查看目前的用户情况 2.创建用户 create user 'Fqq'@'127.0.0.1' IDENTIFIED by '123'; -- 创建一个用户 ...
- 7、Spring Boot检索
1.ElasticSearch简介 Elasticsearch是一个分布式搜索服务,提供Restful API,底层基于Lucene,采用多shard(分片)的方式保证数据安全,并且提供自动resha ...
- 面试阿里,字节跳动90%会被问到的Java异常面试题集,史上最全系列!
Java异常架构与异常关键字 Java异常简介 Java异常是Java提供的一种识别及响应错误的一致性机制. Java异常机制可以使程序中异常处理代码和正常业务代码分离,保证程序代码更加优雅,并提高程 ...
- FL Studio 插件使用教程 —— 3x Osc(上)
在FL Studio20 中,3x Osc是继TS404插件之后资历最老的插件之一,也是FL Studio20 中最重要.使用率最高的插件之一.相比别的FL Studio20内置插件,3x Osc 相 ...
- Vegas教程:教你制作抖音热门人物穿越门窗特效
抖音上经常会有很多特效视频,例如换妆.分镜.合拍.放大等,合适的特效总是会让视频更加出彩.这些特效,除了一部分是抖音自带以外,很多都是用的其他视频特效软件制作而成.这些视频编辑软件操作简单易上手,强大 ...
- [python学习手册-笔记]003.数值类型
003.数值类型 ❝ 本系列文章是我个人学习<python学习手册(第五版)>的学习笔记,其中大部分内容为该书的总结和个人理解,小部分内容为相关知识点的扩展. 非商业用途转载请注明作者和出 ...
- Pytest自动化测试 - 必知必会的一些插件
Pytest拥有丰富的插件架构,超过800个以上的外部插件和活跃的社区,在PyPI项目中以" pytest- *"为标识. 本篇将列举github标星超过两百的一些插件进行实战演示 ...
- Java基础教程——异常处理详解
异常处理 好程序的特性 可重用性 可维护性 可扩展性 鲁棒性 |--|--Robust的音译 |--|--健壮.强壮之意 |--|--指在异常和危险情况下系统依然能运行,不崩溃 Java中,写下如下代 ...