BZOJ 1194 HNOI2006 潘多拉的盒子 BFS+Tarjan+拓扑序DP

题目大意:给定一些自动机,如果某个自动机A能产生的所有串都能在自动机B中产生,则称B是A的一个升级,求最长链

这题TM有毒

数据范围50,暴力枚举每一对点之间的关系,然后Tarjan缩点求最长链就行了

现在对于一对自动机A和B,我想知道A能产生的所有串是否都能在B中产生,那么BFS就可以了

我们用一个二元组(x,y)表示走了某个串后A走到了节点x,B走到了节点y,那么如果x是输出节点而y不是,则不满足条件,否则继续广搜(x,y)的两个后继

由于这样的二元组(x,y)只有O(n2)个,因此标记一下,单次判断的复杂度就是O(n2)的了

#include <vector>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#define M 55
using namespace std;
struct abcd{
    int to,next;
}table[M*M];
int head[M],tot;
int n,cnt;
bool v[M];
int belong[M];
vector<int> member[M];
int f[M],ans;
void Add(int x,int y)
{
    table[++tot].to=y;
    table[tot].next=head[x];
    head[x]=tot;
}
struct Automaton{
    int n,m;
    int trans[M][2];
    bool outlet[M];
    void Read()
    {
        int i,x;
        cin>>n>>m;
        for(i=1;i<=m;i++)
        {
            scanf("%d",&x);
            outlet[x]=true;
        }
        for(i=0;i<n;i++)
            scanf("%d%d",&trans[i][0],&trans[i][1]);
    }
}a[M];
bool BFS(const Automaton &x,const Automaton &y)
{
    static pair<int,int> q[65540];
    static bool v[M][M];
    int i,r=0,h=0;
    memset(v,0,sizeof v);
    q[++r]=make_pair(0,0);v[0][0]=true;
    while(r!=h)
    {
        pair<int,int> sta=q[++h];
        if( x.outlet[sta.first] && !y.outlet[sta.second] )
            return false;
        for(i=0;i<2;i++)
        {
            int xx=x.trans[sta.first][i];
            int yy=y.trans[sta.second][i];
            if(v[xx][yy]) continue;
            v[xx][yy]=true;
            q[++r]=make_pair(xx,yy);
        }
    }
    return true;
}
void Tarjan(int x)
{
    static int stack[M],top;
    static int dpt[M],low[M],T;
    int i;
    stack[++top]=x;
    dpt[x]=low[x]=++T;
    for(i=head[x];i;i=table[i].next)
    {
        if(v[table[i].to])
            continue;
        if(dpt[table[i].to])
            low[x]=min(low[x],dpt[table[i].to]);
        else
            Tarjan(table[i].to),low[x]=min(low[x],low[table[i].to]);
    }
    if(dpt[x]==low[x])
    {
        int t;++cnt;
        do{
            t=stack[top--];
            belong[t]=cnt;
            member[cnt].push_back(t);
            v[t]=true;
        }while(t!=x);
    }
}
void Topology_Sort()
{
    static int q[M],degree[M];
    int i,j,r=0,h=0;
    for(j=1;j<=n;j++)
        for(i=head[j];i;i=table[i].next)
            if(belong[table[i].to]!=belong[j])
                degree[belong[table[i].to]]++;
    for(i=1;i<=cnt;i++)
        if(!degree[i])
            q[++r]=i;
    while(r!=h)
    {
        j=q[++h];
        vector<int>::iterator it;
        f[j]+=member[j].size();
        for(it=member[j].begin();it!=member[j].end();it++)
        {
            int x=*it;
            for(i=head[x];i;i=table[i].next)
            {
                if(belong[table[i].to]==j)
                    continue;
                f[belong[table[i].to]]=max(f[belong[table[i].to]],f[j]);
                if(!--degree[belong[table[i].to]])
                    q[++r]=belong[table[i].to];
            }
        }
        ans=max(ans,f[j]);
    }
}
int main()
{
    int i,j;
    cin>>n;
    for(i=1;i<=n;i++)
        a[i].Read();
    for(i=1;i<=n;i++)
        for(j=1;j<=n;j++)
            if( i!=j && BFS(a[i],a[j]) )
                Add(i,j);
    for(i=1;i<=n;i++)
        if(!v[i])
            Tarjan(i);
    Topology_Sort();
    cout<<ans<<endl;
    return 0;
}
时间: 2024-08-05 07:08:33

BZOJ 1194 HNOI2006 潘多拉的盒子 BFS+Tarjan+拓扑序DP的相关文章

【BZOJ】1093: [ZJOI2007]最大半连通子图(tarjan+拓扑序)

http://www.lydsy.com/JudgeOnline/problem.php?id=1093 两个条件综合起来加上求最大的节点数,那么很明显如果是环一定要缩点. 然后再仔细思考下就是求dag的最长路的数目啦... 然后wa了... 看了题解...噗!第一次注意到缩点后会有重边QAQ...于是.. orz orz 然后思考了下怎么处理重边...很简单,每个点bfs时记录一下就行了.. #include <cstdio> #include <cstring> #includ

第16章 潘多拉的盒子

在分班后的高二上半学期的期末,孙斌新忽然不想上学了,想去当兵.我们不明白是什么导致他会有这样的想法.是因为高中的压力吗,还是适应不了这炼狱般的生活呢?我们不知道,知道的只是孙斌新在期末考完回家的路上,忽然对我们说,过年时他不会再去学校了. 无论我们怎么问都问不出一点消息.他就爱这样,有什么事情都不会告我们说的.其实不只我们,他如果有事甚至连他爸妈都不会说.他喜欢一个人琢磨,他说他的路他要自己走,这样将来才不会后悔.可是他骨子里又藏着一份怯弱,害怕会出错.所以他会时不时的小翼翼的泄漏些自己的秘密.

bzoj-1194 潘多拉的盒子

题意: 给出S个自动机,每个自动机有n个结点和m个输出结点: 每个结点有两个后继'0'和'1',将当前走过的串末尾加那个字符,然后走到下一个结点: 每次从0号点,以一个空串出发,到了输出结点时可以选择输出当前串: 如果一个自动机x可以输出的所有串另一个自动机y也都可以输出,那么y是x的升级: 求最大升级序列: S,n,m<=50: 题解: 这道题稍微考虑一下之后,发现难以处理的地方是自动机: 那先讨论一下别的地方: begin 如果已知了所有自动机之间的升级关系之后,怎么求出答案呢? 可以将x升

BZOJ 4011 HNOI2015 落忆枫音 拓扑序DP

题目大意:给定一张有向无环图,现在要求加入一条边,求加入后以1为根的树形图个数 首先不考虑加入的这条边,那么这个图是一个DAG 由朱刘算法的推论可知,如果除根节点外每个点都选择一条入边,由于没有环,因此一定会形成一个树形图 因此答案就是∏ni=2degreei 其中degreei表示第i个点的入度 现在加入这条边之后,我们仍然可以套用这个公式,但是这样就会有一些不合法的方案被统计进来,我们需要把这些不合法的方案减掉 一个方案如果不合法,那么一定会形成一个环,而环一定包含新加入的那条边 因此我们减

HG255D刷潘多拉共享惠普1020打印机的问题汇总

借鉴原帖地址:http://www.right.com.cn/forum/forum.php?mod=viewthread&tid=212810&page=1 实现环境:HG255D(16MB闪存+64MB内存+USB),潘多拉固件版本号PandoraBox 14.09 r1147 / LuCI 0.12 (0.12+git-4d0a20b)HP1020打印机 硬件ID:03f0:2b17(此解决方案适用于HP 1000 1005 1018 1020打印机,其他潘多拉固件请参照此文活学活用

潘多拉卡密邀请码领取及使用方法

最近大家都在问潘多拉邀请码是什么?潘多拉邀请码多少? 潘多拉卡密是什么?潘多拉卡密分享怎么获取? 潘多拉邀请码:121111   建议官方取卡:http://t.cn/EZXOG03# 一键进入,开启浪漫邂逅 超多花样百出的节目等着你,更有小伙伴陪你玩国王直播宝盒功能功能1.最帖心,强大的消息推送,无论你在何时何地,都能把她捧在手中.2.最浪漫,炫酷的礼物特效,可以表达浓浓爱意.一见倾心.3.最约会,开启约会模式,浪漫邂逅你的爱情.4.最有型,强大的"VIP.守护"体系彰显你的尊贵,秀

潘多拉卡密邀请码获取和使用方法

最近大家都在问潘多拉邀请码是什么?潘多拉邀请码多少? 潘多拉卡密是什么?潘多拉卡密分享怎么获取? 潘多拉邀请码:110211 → 潘多拉卡密官网自助渠道入口 建议官方取卡:http://t.cn/RkzHidT 一键进入,开启浪漫邂逅 超多花样百出的节目等着你,更有小伙伴陪你玩 随时随地放大你的所见所闻,一不下心,你就成为大明星 秀出自己的精彩,分享你的喜怒哀乐,够胆你就来 ------------------------------------------------------ 无论回信的内

潘多拉卡密领取及邀请码获取方法

最近很多朋友都在问潘多拉邀请码是什么?潘多拉邀请码多少? 潘多拉卡密是什么?潘多拉卡密分享怎么获取? 潘多拉邀请码 121111 潘多拉官方卡密自助入口 建议取卡地址 http://t.cn/EApJIVB 潘多拉魂无忧,魂无忧的诗,魂无忧的简介, 修斯盗火人间获,远古神王也怒生. 狡诈多奸欺世道,纯实质朴乱心声. 男儿不懂真情事,佳丽奇思伪善名. 慵懒的身体翻来覆去住惯了平板床不觉席梦思的美妙舞动的灵魂无法安睡 七魂六魄随风荡漾 无意中触碰了潘多拉魔盒 原文地址:https://www.cnb

BZOJ 1924 所驼门王的宝藏(强连通分量+树形DP)

题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1924 题意: 思路:首先建立所有可达点之间的有向图.之后求强连通分量SCC,缩点重新构图.然后就是一个树,树形DP一下即可. int n,r,c; map<i64,int> mp; map<int,int> mp1,mp2; struct node { int x,y,op; }; node a[N]; int visit[N]; vector<int> V1