[arc080d]Prime Flip

探讨无限数组中通过质数翻转实现所有元素为0的最少操作次数。利用哥德巴赫猜想,通过匈牙利算法匹配奇偶位置,实现最优解。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

题目大意

有一个长度为无限的数组,其中有若干个位置x[1~n]为1,其他为0。现在你每次操作,可以选择一个大于3的质数p,然后把某个长度为p的区间01翻转,问至少操作多少次?
n<=100,x[i]<=1e7

解题思路

先考虑一下为什么一定操作成功。发现1可以由7-3-3凑出来,2可以5-3,4可以17-13,6可以23-17…更大的数和到底要多少次操作,我们考虑分类讨论。
1,奇质数,1次操作;
2,偶数,可以2次操作。哥德巴赫猜想。
3,其他奇数,3次操作。拆成一个偶数和一个奇质数。
原本哥德巴赫猜想是能够有1的,但小的情况手算发现没问题。
那怎么用这个结论解决本题呢?
考虑差分。把区间操作变成两点操作。这里我们使用xor差分。即delta[i]=pd[i]^pd[i-1],其中pd[X[i]]=1。
然后就十分好做了,我们考虑先把delta[i]=1的i奇偶分组,然后尽量多地用奇质数消去一组(i,j)(情况1),那么我们匈牙利匹配一下,然后消不去的组内匹配(情况2),最后如果每组剩下一个用3。
可以证明点的个数一定是偶数。

代码

#include<cstdio> 
#include<algorithm>
#include<cstring>
#include<cmath>
#include<set>
#include<map>
using namespace std;
#define fo(i,j,k) for(i=j;i<=k;i++)
#define fd(i,j,k) for(i=j;i>=k;i--)
#define cmax(a,b) (a=(a>b)?a:b)
#define cmin(a,b) (a=(a<b)?a:b)
typedef long long ll;
const int N=1e6+5,M=1e7+50,mo=1e9+7;
int vis[N],pdd[N],pd[M],pri[N],Ref[N],d[N],e[N],c[M],x,ttt,n,i,j,ans,y;
int tt,b[N],nxt[N],fst[N];
void cr(int x,int y)
{
    tt++;
    b[tt]=y;
    nxt[tt]=fst[x];
    fst[x]=tt;
}
int dfs(int x)
{
    if (vis[x]==ttt) return 0;
    vis[x]=ttt;
    for (int p=fst[x];p;p=nxt[p])
    {
        if (!Ref[b[p]]||dfs(Ref[b[p]]))
        {
            Ref[b[p]]=x;
            pdd[x]=b[p];
            return 1;
        }
    }
    return 0;
}
void match()
{
    fo(i,1,n)
        if (!pdd[i])
        {
            ttt++;
            if (dfs(i)) ans++;
        }
}
void predo(int n)
{
    fo(i,2,n)
    {
        if (!pd[i]) pri[++pri[0]]=i;
        fo(j,1,pri[0])
        {
            if (1ll*i*pri[j]>n) break;
            pd[i*pri[j]]=1;
        }
    }
}
int main()
{
    freopen("t13.in","r",stdin);
    //freopen("t13.out","w",stdout);
    scanf("%d",&n);
    fo(i,1,n)
    {
        scanf("%d",&x);
        c[x]^=1;
        c[x+1]^=1;
    }
    predo(1e7+5);
    fo(i,1,1e7+5)
        if (c[i])
        {
            if (i%2)
                d[++d[0]]=i;
            else e[++e[0]]=i;
        }
    pd[1]=pd[2]=1;
    fo(i,1,d[0])
        fo(j,1,e[0])
        if (!pd[abs(e[j]-d[i])])
            cr(i,j);
    ans=0;
    match();
    x=y=0;
    fo(i,1,d[0]) if (!pdd[i]) x++;
    fo(i,1,e[0]) if (!Ref[i]) y++;
    ans+=x-x%2+y-y%2;
    if (x%2) ans+=3;
    printf("%d",ans);
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值