2023NOIP A层联测25 T4 滈葕

2023NOIP A层联测25 T4 滈葕

配血实验与2-SAT。

思路

z=1 表示配血实验发生凝集反应,设 ai,bi 分别表示第 i 个人有无凝集原 A,B。(无凝集原 A,肯定有抗 A 凝集素,B同理)那么发生反应的必要条件是 ax\and¬aybx\and¬by,所以 z=(ax\and¬ay)\or(bx\and¬by)

z=0 则有 ¬(ax\and¬ay)\and¬(bx\and¬by)=(¬ax\oray)\and(¬bx\andby)

z=1 则有 z=(ax\and¬ay)\or(bx\and¬by)=(ax\orbx)\and(ax\or¬by)\and(¬ay\orbx)\and(¬ay\or¬by)

这两个都可以使用 2-SAT 解决。

2-SAT

CODE

// ubsan: undefined
// accoders
#include<bits/stdc++.h>
using namespace std;

const int maxn=5e5+5;

struct node
{
    int to,nxt;
}edge[maxn*8];

int n,m,tot,dfntot,num;
int dfn[maxn],low[maxn],id[maxn],A[maxn][2],B[maxn][2],head[maxn];

bool vis[maxn];

stack<int>stk;

void add(int x,int y)
{
    tot++;
    edge[tot].to=y;
    edge[tot].nxt=head[x];
    head[x]=tot;
}

void dfs(int u)
{
    stk.push(u),dfn[u]=low[u]=++dfntot,vis[u]=1;
    for(int i=head[u];i;i=edge[i].nxt)
    {
        int v=edge[i].to;
        if(!dfn[v])
        {
            dfs(v);
            low[u]=min(low[u],low[v]);
        }
        else if(vis[v]) low[u]=min(low[u],dfn[v]);
    }
    if(low[u]==dfn[u])
    {
        num++;
        while(stk.top()!=u) vis[stk.top()]=0,id[stk.top()]=num,stk.pop();
        vis[stk.top()]=0,id[stk.top()]=num;
        stk.pop();
    }
}

int main()
{
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++) A[i][0]=i,A[i][1]=i+n,B[i][0]=i+2*n,B[i][1]=i+3*n;
    for(int i=1;i<=m;i++)
    {
        int x,y,z;
        scanf("%d%d%d",&x,&y,&z);
        if(z==0)
        {
            add(A[x][1],A[y][1]),add(A[y][0],A[x][0]);//!a a
            add(B[x][1],B[y][1]),add(B[y][0],B[x][0]);//!b b
        }
        else
        {
            add(A[x][0],B[x][1]),add(B[x][0],A[x][1]);
            add(A[x][0],B[y][0]),add(B[y][1],A[x][1]);
            add(A[y][1],B[x][1]),add(B[x][0],A[y][0]);
            add(A[y][1],B[y][0]),add(B[y][1],A[y][0]);
        }
    }

    for(int i=1;i<=n*4;i++) if(!dfn[i]) dfs(i);

    for(int i=1;i<=n;i++) if(id[A[i][0]]==id[A[i][1]]||id[B[i][0]]==id[B[i][1]]) printf("NO"),exit(0);
    printf("YES\n");
    for(int i=1;i<=n;i++)
    {
        if(id[A[i][0]]<id[A[i][1]]&&id[B[i][0]]<id[B[i][1]]) printf("D");
        if(id[A[i][0]]<id[A[i][1]]&&id[B[i][0]]>id[B[i][1]]) printf("A");
        if(id[A[i][0]]>id[A[i][1]]&&id[B[i][0]]<id[B[i][1]]) printf("B");
        if(id[A[i][0]]>id[A[i][1]]&&id[B[i][0]]>id[B[i][1]]) printf("C");
    }
}
posted @   彬彬冰激凌  阅读(12)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现
点击右上角即可分享
微信分享提示