题意
有一个无限长的01序列,其中n个位置上是1,其余都是0。每次可以选择一个长度为奇素数的区间,然后将这个区间内的元素取反。问最少多少步后可以将整个序列变为0。
n<=100,ai<=10^7
分析
首先差分一下,不难发现1的个数一定是偶数,那么操作就变成了每次可以把两个位置[l-1,r]同时取反。
现在问题就变成了让这偶数个1两两配对,使得总操作数最小。
对于两个数x和y
若|x-y|是奇素数,则需要一步操作。
若|x-y|是偶数,根据哥德巴赫猜想可知需要两步操作。
若|x-y|是奇合数,则可以先减掉一个3,然后就变成了偶数的情况,总共需要三步。
贪心地想肯定是步数越少的配对越多越好。
可以把位置按照下标奇偶性建二分图,然后跑匈牙利,最后不行的话再强行补后两种情况即可。
代码
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
using