CF1504D Flip the Cards(找规律+贪心)

题目大意:给你n张牌,正反面都有数字,保证所有牌上的数字在$[1,2n]$内且互不相同。你可以翻转任意张牌,接下来需要把牌按正面的数字从小到大排序,需要保证排序后牌背面的数字是从大到小。给出初始时牌的状态,问最少需要多少次翻转才能符合要求,如果一定达不到要求则输出-1。 n=1e5

限制关系是环,难以从前到后直接处理这个长度为2n的序列

套路1:把序列砍成一半

我们定义$a[i]$表示数$i$的牌背面的数字是多少。

多画几个例子容易发现,如果一张牌正反面都$\le n$或正反面都$>n$,一定不合法!因为数字各不相同,排的时候剩余的空不够!假设有一张牌的数字都$\le n$,那么在$[1,n]$内可用的其他位置为$n-2$个,却还有$n-1$张牌要占位置。都$>n$的情况是同一个道理

现在只考虑$[1,n]$,我们需要把$a[i]$划分成两个下降子序列!第一个序列i正面$a[i]$负面,第二个序列从后到前,a[i]反面i正面。

套路2:观察性质

如果$i$满足$min_{j\le i}(a[j]) > max_{j>i}(a[j])$,我们称$i$和$i+1$的间隔是一个间断,间断把序列分成数段,段和段之间的答案不会相互影响,因为段头可以接在任意一个子序列的末尾

单个段具有特殊性质!如果这段划分成两个下降子序列,那么划分的方式是唯一的

证明:

假设这一段是$[l,r]$,那么只有唯一的一个$x\in(l,r],a[x]>a[l]$

对于$i\in (l,r]$

1.要么存在$l\le j<i$,$a[j]<a[i]$,$i$和$j$一定不在一组

2.要么存在$i<k\le r$,$a[k]>a[i]$,$k$和$i$不在一组

对于$(l,x)$的位置,一定和l划分到一组。在这之后,l开头一组,x开头一组。现在需要证明$(x,r]$之间的划分方式唯一

如果$a[i]$小于l的序列的末尾元素,似乎可以把i分到l或者x的末尾。但它一定不满足性质1,则性质2成立,它必须和k不在同一组。如果k满足性质2,对应的位置是k',那么i,k,k'都不在同一组,一定不合法,因此k只能满足性质1,与i不在同一组。如果a[i]大于l的末尾元素,则只可能放在x的末尾。

当遍历到x+1是,l组的末尾(也就是x-1)满足性质2,x-1对应的k一定不和l一组,只能和x一组。归纳到(x,r]的其他位置,唯一确定了这一段的划分方式

综上,我们只需要关心l是正还是反就行了,正反两种情况讨论一下取最小值即可

 1 const int maxn=400000,N1=maxn+5; const int inf=0x3f3f3f3f;
2
3 int n;
4 template <typename _T> void read(_T &ret)
5 {
6 ret=0; _T fh=1; char c=getchar();
7 while(c<'0'||c>'9'){ if(c=='-') fh=-1; c=getchar(); }
8 while(c>='0'&&c<='9'){ ret=ret*10+c-'0'; c=getchar(); }
9 ret=ret*fh;
10 }
11 int oa[N1],ob[N1],a[N1], type[N1];
12 int sma[N1],smi[N1];
13
14 int mi[2];
15 int check(int l,int r,int p)
16 {
17 mi[p]=a[l]; mi[p^1]=inf; int ans=type[l]^p;
18 for(int i=l+1;i<=r;i++)
19 {
20 if(a[i]>max(mi[p],mi[p^1])) return -1;
21 if(mi[p]<mi[p^1]){
22 if(a[i]<mi[p]) mi[p]=a[i], ans+=type[i]^p;
23 else mi[p^1]=a[i], ans+=type[i]^p^1;
24 }else{
25 if(a[i]<mi[p^1]) mi[p^1]=a[i], ans+=type[i]^p^1;
26 else mi[p]=a[i], ans+=type[i]^p;
27 }
28 }
29 return ans;
30 }
31 int calc(int l,int r)
32 {
33 return min(check(l,r,1),check(l,r,0));
34 }
35 int solve()
36 {
37 int ans=0, tmp;
38 for(int i=1,j=1;i<=n;i++)
39 {
40 for(j=i;j<n&&smi[j]<=sma[j+1];j++);
41 tmp=calc(i,j); if(tmp==-1) return -1;
42 ans+=tmp;
43 i=j;
44 }
45 return ans;
46 }
47
48 int main()
49 {
50 scanf("%d",&n);
51 for(int i=1;i<=n;i++)
52 {
53 read(oa[i]), read(ob[i]);
54 if( (oa[i]<=n && ob[i]<=n) || (oa[i]>n && ob[i]>n) ){ puts("-1"); return 0; }
55 if(oa[i]<=n) a[oa[i]]=ob[i]; else a[ob[i]]=oa[i], type[ob[i]]=1;
56 }
57 smi[0]=inf;
58 for(int i=1;i<=n;i++) smi[i]=min(smi[i-1],a[i]);
59 for(int i=n;i>=1;i--) sma[i]=max(sma[i+1],a[i]);
60 int ans=solve();
61 printf("%d\n",ans);
62 return 0;
63 }

CF1504X Codeforces Round #712的更多相关文章

  1. Codeforces Round #712 (Div. 2)

    A. Déjà Vu 题意:就是问能否加上字母a,使得字符串不中心对称 思路:只有一种情况不能加入,就是全部是a,剩下的都可以满足,找a的位置就找哪个字母不是a,然后让它的对称位置是新加的这个a 代码 ...

  2. Codeforces Round #366 (Div. 2) ABC

    Codeforces Round #366 (Div. 2) A I hate that I love that I hate it水题 #I hate that I love that I hate ...

  3. Codeforces Round #354 (Div. 2) ABCD

    Codeforces Round #354 (Div. 2) Problems     # Name     A Nicholas and Permutation standard input/out ...

  4. Codeforces Round #368 (Div. 2)

    直达–>Codeforces Round #368 (Div. 2) A Brain’s Photos 给你一个NxM的矩阵,一个字母代表一种颜色,如果有”C”,”M”,”Y”三种中任意一种就输 ...

  5. cf之路,1,Codeforces Round #345 (Div. 2)

     cf之路,1,Codeforces Round #345 (Div. 2) ps:昨天第一次参加cf比赛,比赛之前为了熟悉下cf比赛题目的难度.所以做了round#345连试试水的深浅.....   ...

  6. Codeforces Round #279 (Div. 2) ABCDE

    Codeforces Round #279 (Div. 2) 做得我都变绿了! Problems     # Name     A Team Olympiad standard input/outpu ...

  7. Codeforces Round #262 (Div. 2) 1003

    Codeforces Round #262 (Div. 2) 1003 C. Present time limit per test 2 seconds memory limit per test 2 ...

  8. Codeforces Round #262 (Div. 2) 1004

    Codeforces Round #262 (Div. 2) 1004 D. Little Victor and Set time limit per test 1 second memory lim ...

  9. Codeforces Round #370 - #379 (Div. 2)

    题意: 思路: Codeforces Round #370(Solved: 4 out of 5) A - Memory and Crow 题意:有一个序列,然后对每一个进行ai = bi - bi  ...

随机推荐

  1. Solution -「AGC 034C」Tests

    \(\mathcal{Description}\)   Link.   给定非负整数序列 \(\{l_n\},\{r_n\},\{b_n\},X\),求最小的 \(s\),使得存在非负整数序列 \(\ ...

  2. .NET官方封装的Win32API类库

    大部分朋友在使用C#.NET调用Win32API时都不清楚API函数的声明,要么就是抄网上的代码,但是总会遇到各种各样奇奇怪怪难以解决的问题,打算自己封装又发现工作量实在太大. 其实完全没有必要自己动 ...

  3. suse 12 二进制部署 Kubernetets 1.19.7 - 第01章 - 创建CA证书和kubectl集群管理命令

    文章目录 1.kubernetes集群部署 1.0.创建CA证书和秘钥 1.0.0.安装cfssl工具 1.0.1.创建根证书 1.0.2.创建证书签名请求文件 1.0.3.生成CA证书和秘钥 1.0 ...

  4. LAMP环境下部署项目管理软件--禅道

    禅道与Jira的对比 禅道最大的特色是创造性的将产品.项目.测试这三者的概念明确分开,互相配合,又互相制约.通过需求.任务.bug来进行交相互动,最终通过项目拿到合格的产品. Jira设计以项目为主线 ...

  5. c++ 子类与父类之间的类型转换

    子类与父类之间的类型转换 先给一段代码 class Base { public: int a = 10; }; class pub_Derv : public Base { Base *getBase ...

  6. [镜像转换] ova文件转换成raw文件, 导入到openstack

    ova转raw 使用工具: qemu-img.libguestfs-tools.libguestfs-winsupport # 从vmware导出ova文件, tar 解压导出来的ova文件 $ ta ...

  7. IPv6 OSPFv3路由协议测试——信而泰网络测试仪实操

    关键词 IPv6; OSPFv3; OSPFv2. 前言:在网络部署中,动态路由协议是重要的组成部分.良好的路由协议,是保证网络可靠.稳定运行的基础.为了适应IPv6协议栈的变化,OSPFv3协议在保 ...

  8. 用python写九九乘法表

    用python来写九九乘法表,九九乘法表的结构是这样子的: 第一行是1 * 1 = 1,第二行是1 * 2 = 2 | 2 * 2 = 4...以此类推.注意到没,每一行的第一个乘的数字在从1到当行变 ...

  9. JZ-026-二叉搜索树与双向链表

    二叉搜索树与双向链表 题目描述 输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表.要求不能创建任何新的结点,只能调整树中结点指针的指向. 题目链接: 二叉搜索树与双向链表 代码 /** * ...

  10. LeetCode-024-两两交换链表中的节点

    两两交换链表中的节点 题目描述:给定一个链表,两两交换其中相邻的节点,并返回交换后的链表. 你不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换. 示例说明请见LeetCode官网. 来源:力 ...