构造一个数组$b_{i}$(初始为0),对于操作$[l_{i},r_{i}]$,令$b_{l_{i}}$和$b_{r_{i}+1}$值异或1,表示$i$和$i-1$的差值发生改变,最终即要求若干个$b_{i}$为1,其余为0

对于一组合法方案,通过重新排列操作的顺序,使得每一次操作都有至少一个修改是$b_{i}$由1变为0,然后进行以下构造:

1.仅有1个$b_{i}$由1变为0(即另一个$b_{i}$是由0变成1),我们将其称之为移动,即将这个$b_{i}$移动到了另一个位置,其中移动的距离必须为奇素数

2.两个都由1变为0,可以看作移动到了与另一个1相同的位置上,那么就可以消除,称之为匹配,显然每一个初始为1的位置出现且仅出现在一个匹配内

根据匹配的独立性,对于一组给定的匹配方式,即求每一对数$(x,y)$相同所需要的最小次数之和,同时由于加减互为逆运算,可以看作由$x$变为$y$(即不移动$y$)

求$x$变为$y$的最小次数,对$x$和$y$分类讨论:

1.$x=y$,答案为0

2.$|y-x|$为奇素数,答案为1

3.$|y-x|$为偶数,根据哥德巴赫猜想,答案为2(特别的,2=5-3)

4.$|y-x|$为其他情况,选择一个比其小的奇素数,剩下的偶数部分再分为两个奇素数,因此答案为3(特别的,1=3+3-5,3和5都为素数,是第2种情况)

对于第1种情况,直接贪心匹配,即对于一个连续的$a_{i}$,只需要记录左端点和右端点+1即可

假设初始为1的位置中,有$x$个奇数,$y$个偶数,然后选择了$k$对奇偶使得差为奇素数,然后对于剩下的显然尽量使用第3种,答案即$k+2(\lfloor\frac{x-k}{2}\rfloor+\lfloor\frac{y-k}{2}\rfloor)+3((x-k)mod\ 2)$

由于$2\lfloor\frac{x-k}{2}\rfloor=x-k-(x-k)mod\ 2$,同时$x-k\equiv y-k(mod\ 2)$(1总数为偶数),因此可以化简为$x+y-k+((x-k)mod\ 2)$

当$k$增大后上式一定单调不减,因此仅连奇素数的边,求最大匹配即可,时间复杂度为$o(n^{2})$

 1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 205
4 #define M 10000005
5 struct ji{
6 int nex,to;
7 }edge[N*N*2];
8 vector<int>v;
9 int E,n,x,ans,p[M],vis[M],a[N],head[N],match[N];
10 void add(int x,int y){
11 edge[E].nex=head[x];
12 edge[E].to=y;
13 head[x]=E++;
14 }
15 bool dfs(int k){
16 if (vis[k])return 0;
17 vis[k]=1;
18 for(int i=head[k];i!=-1;i=edge[i].nex)
19 if ((match[edge[i].to]==-1)||(dfs(match[edge[i].to]))){
20 match[k]=edge[i].to;
21 match[edge[i].to]=k;
22 return 1;
23 }
24 return 0;
25 }
26 int main(){
27 for(int i=2;i<M-4;i++){
28 if (!vis[i])p[++p[0]]=i;
29 for(int j=1;(j<=p[0])&&(i*p[j]<M-4);j++){
30 vis[i*p[j]]=1;
31 if (i%p[j]==0)break;
32 }
33 }
34 vis[0]=vis[1]=vis[2]=1;
35 scanf("%d",&n);
36 for(int i=1;i<=n;i++){
37 scanf("%d",&x);
38 a[2*i-1]=x;
39 a[2*i]=x+1;
40 }
41 n*=2;
42 sort(a+1,a+n+1);
43 for(int i=1;i<=n;i++)
44 if (((i==1)||(a[i]!=a[i-1]))&&((i==n)||(a[i]!=a[i+1])))v.push_back(a[i]);
45 n=v.size(),x=0;
46 for(int i=0;i<n;i++)
47 if (v[i]&1)x++;
48 memset(head,-1,sizeof(head));
49 for(int i=0;i<n;i++)
50 for(int j=0;j<n;j++)
51 if (!vis[abs(v[i]-v[j])])add(i,j);
52 memset(match,-1,sizeof(match));
53 for(int i=0;i<n;i++)
54 if (v[i]&1){
55 memset(vis,0,sizeof(vis));
56 ans+=dfs(i);
57 }
58 printf("%d",n-ans+((ans&1)!=(x&1)));
59 }

[atARC080F]Prime Flip的更多相关文章

  1. [Arc080F]Prime Flip

    [Arc080F]Prime Flip Description 你有无限多的"给给全",编号为1,2,3,....开始时,第x1,x2,...,xN个"给给全" ...

  2. Prime Flip AtCoder - 2689

    发现我们每次区间取反,相邻位置的正反关系只有两个位置发生改变 我们定义bi为ai和ai-1的正反关系,即ai=ai-1时bi=0,否则bi=1,每次取反l~r,b[l]和b[r+1]会发生改变 容易发 ...

  3. AT2689 [ARC080D] Prime Flip

    简要题解如下: 区间修改问题,使用差分转化为单点问题. 问题变成,一开始有 \(2n\) 个点为 \(1\),每次操作可以选择 \(r - l\) 为奇质数的两个点 \(l, r\) 使其 ^ \(1 ...

  4. 【arc080F】Prime Flip

    Portal --> arc080_f Solution ​  这题的话..差分套路题(算吗?反正就是想到差分就很好想了qwq) ​​  (但是问题就是我不会这种套路啊qwq题解原话是:&quo ...

  5. 【ARC080F】Prime Flip 差分+二分图匹配

    Description ​ 有无穷个硬币,初始有n个正面向上,其余均正面向下.  你每次可以选择一个奇质数p,并将连续p个硬币都翻转.  问最小操作次数使得所有硬币均正面向下. Input ​ 第一行 ...

  6. 【Atcoder】ARC 080 F - Prime Flip

    [算法]数论,二分图最大匹配 [题意]有无限张牌,给定n张面朝上的牌的坐标(N<=100),其它牌面朝下,每次操作可以选定一个>=3的素数p,并翻转连续p张牌,求最少操作次数使所有牌向下. ...

  7. AT2689 Prime Flip

    传送门 这个题是真的巧妙 首先一个很巧妙的思路,差分 考虑假如\(a_i!=a_{i-1}\),则\(b_i=1\),否则\(b_i=0\) 这样一来,一个区间的翻转就变成了对于两个数的取反了 然后我 ...

  8. Java基础之写文件——从多个缓冲区写(GatheringWrite)

    控制台程序,使用单个写操作将数据从多个缓冲区按顺序传输到文件,这称为集中写(GatheringWrite)操作.这个功能的优势是能够避免在将信息写入到文件中之前将信息复制到单个缓冲区中.从每个缓冲区写 ...

  9. Atcoder 乱做

    最近感觉自己思维僵化,啥都不会做了-- ARC103 F Distance Sums 题意 给定第 \(i\) 个点到所有点的距离和 \(D_i\) ,要求构造一棵合法的树.满足第 \(i\) 个点到 ...

随机推荐

  1. Docker-初见

    目录 Docker概述 Docker历史 Docker Docker的基本组成 Docker安装 使用流程 底层原理 Docker的常用命令 Portainer 可视化面板安装 镜像原理之联合文件系统 ...

  2. PTA实验7-2-3 求矩阵的局部极大值 (15分)

    实验7-2-3 求矩阵的局部极大值 (15分) 给定M行N列的整数矩阵A,如果A的非边界元素A[i][j]大于相邻的上下左右4个元素,那么就称元素A[i][j]是矩阵的局部极大值.本题要求给定矩阵的全 ...

  3. 讲讲java中线程池的实现

    今天跟一个同学谈到java中的线程池的实现,才发现有些知识点已经记不清了,所以特意把源码打开,对官方文档做了些说明. 其实这些理解了之后,读懂源码应该是没多大问题了,有感兴趣的小伙伴们可以看完说明后自 ...

  4. API代码实战

    API实例一: login.py文件 #!/usr/bin/env python #!coding:utf-8 from flask import Flask,jsonify from flask_r ...

  5. FastAPI 学习之路(三十七)元数据和文档 URL

    你可以在 FastAPI 应用中自定义几个元数据配置. 你可以设定: Title:在 OpenAPI 和自动 API 文档用户界面中作为 API 的标题/名称使用. Description:在 Ope ...

  6. ZK(ZooKeeper)分布式锁实现

    点赞再看,养成习惯,微信搜索[牧小农]关注我获取更多资讯,风里雨里,小农等你. 本文中案例都会在上传到git上,请放心浏览 git地址:https://github.com/muxiaonong/Zo ...

  7. 【UE4 C++】学习笔记汇总

    UE4 概念知识 基础概念--文件结构.类型.反射.编译.接口.垃圾回收.序列化[导图] GamePlay架构[导图] 类的继承层级关系[导图] 反射机制 垃圾回收机制/算法 序列化 Actor 的生 ...

  8. Java中的函数式编程(八)流Stream并行编程

    写在前面 在本系列文章的第一篇,我们提到了函数式编程的优点之一是"易于并发编程". Java作为一个多线程的语言,它通过 Stream 来提供了并发编程的便利性. 题外话: 严格来 ...

  9. Spring动态添加定时任务

    Spring动态添加定时任务 一.背景 二.需求和实现思路 1.能够动态的添加一个定时任务. 2.能够取消定时任务的执行. 3.动态的修改任务执行的时间. 4.获取定时任务执行的异常 三.代码实现 四 ...

  10. BF算法和KMP算法

    这两天复习数据结构(严蔚敏版),记录第四章串中的两个重要算法,BF算法和KMP算法,博主主要学习Java,所以分析采用Java语言,后面会补上C语言的实现过程. 1.Brute-Force算法(暴力法 ...