zoukankan      html  css  js  c++  java
  • 20180429模拟赛T1——添边问题

    【问题描述】

    没有环的有向图称为有向无环图,这是一个多么美好的结构吖。

    如果有一张有 N 个点的有向图,我们可能需要删掉一些边使它变成一张有向无环图。假设初始时我们只有 N 个互不相连的点,当然它也是一张有向无环图。依次给出 T 条边和每条边的方向。 每给出一条边就要立即决定是否要加入这一条边,使得这张图始终是一张有向无环图(意思是:按顺序处理每条边,能加就加,让你模拟这个过程,自环不能加入)。计算在满足要求的情况下一共有多少条边没有被加入。如果所有边都可以加入这张图则输出 0。

    【输入格式】

    第一行为两个整数:N(1<=N<=250),T(0<=T<=100,000)。接下来 T 行,每行两个整数 x,y(1 <=x,y<= N),表示一条从 x 到 y 的单向边。

    【输出格式】

    一个整数,表示没有被加入的边数。

    【样例输入】

    3 6
    1 2
    1 3
    3 1
    2 1
    1 2
    2 3

    【样例输出】

    2

    【样例说明】

    1-->2,之前 2-->1 没有路径,不会造成环,加入

    1-->3,之前 3-->1 没有路径,不会造成环,加入

    3-->1,之前 1-->3 有路径,使得图有环,不加入

    2-->1,之前 1-->2 有路径,不加入

    1-->2 , 之前 2-->1 没有路径,加入

    2-->3,之前 3-->2 没有路径,加入

    因此答案是 2

    【数据规模】

    对于40%的数据,n<=50,T<=1000

    对于90%的数据,n<=150,T<=100000

    对于100%的数据,n<=250,T<=100000

    题解

    我们用(f[i][j])表示(i)是否能到(j)(1:能 0:不能)。首先忽略重边。判断一条边能否加入显然是(O(1))的,只要看看(f[y][x])是不是(1)即可。

    若加入,就要维护连通性。

    如图,对于两个点(A,B),圈表示点集,箭头表示边,连边(A o B)后我们又要连蓝边与绿边。

    不难发现,我们可以把(A)与蓝色集合、(B)与绿色集合合并,于是问题就变成了:

    蓝色集合向绿色集合连边。

    蓝色集合为能到达(A)的点集;绿色集合表示能到达(B)的点集。

    (n^2)大力连边即可。

    我的代码

    #include <cstring>
    #include <fstream>
    
    using namespace std;
    
    ifstream fin("stock.in");
    ofstream fout("stock.out");
    
    const int maxm=100000;
    const int maxn=256;
    
    int a[maxn][maxn];
    int b[maxm],c[maxm];
    
    int main()
    {
    	int n, m;
    	fin >> n >> m;
    	int ans=0;
    	for(int i=0; i<m; ++i)
    	{
    		int aa, bb;
    		fin >> aa >> bb;
    		if(aa==bb || a[bb][aa]) ans++;
    		else
    		{
    			int k1=0;
    			int k2=0;
    			b[++k1]=aa;
    			c[++k2]=bb;
    			for(int j=1; j<=n; ++j)
    				if(a[j][aa] && !a[j][bb])
    					b[++k1]=j;
    			for(int j=1; j<=n; ++j)
    				if(a[bb][j] && !a[aa][j])
    					c[++k2]=j;
    			for(int j=1; j<=k1; ++j)
    				for(int k=1; k<=k2; ++k)
    					a[b[j]][c[k]]=1;
    		}
    	}
    	fout << ans << '
    ';
    	return 0;
    }
    

    bitset优化

    以上代码是可过的,但还可以用bitset优化一下,上一发zd大佬的代码。

    #include <iostream>
    #include <bitset>
    #include <cstdio>
    
    using namespace std;
    
    typedef int ll;
    
    inline char gc()
    {
    	static char buf[1<<14],*p1=buf,*p2=buf;
    	return p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<14,stdin),p1==p2)?EOF:*p1++;
    }
    
    #define dd c=gc()
    inline ll read()
    {
    	ll x=0,f=1;
    	char dd;
    	for(; !isdigit(c); dd)if(c=='-')f=-1;
    	for(; isdigit(c); dd)x=(x<<1)+(x<<3)+(c^48);
    	return x*f;
    }
    #undef dd
    
    inline void write(ll x)
    {
    	if(x<0)putchar('-'),x=-x;
    	if(x>9)write(x/10);
    	putchar(x%10|48);
    }
    
    bitset<252>a[252];
    
    int main()
    {
    	freopen("stock.in","r",stdin);
    	freopen("stock.out","w",stdout);
    	register ll n=read(),ans=0;
    	for (register ll i=0; i<n; ++i) a[i][i]=1;
    	for (register ll m=read(),x,y; m; --m)
    	{
    		x=read()-1,y=read()-1;
    		if (a[y][x]) ++ans;
    		else if (!a[x][y])
    			for (register ll i=0; i<n; ++i)
    			{
    				if (a[i][x]) a[i]=a[i]|a[y];
    			}
    	}
    	return write(ans),0;
    }
    
  • 相关阅读:
    mysql 错误 1067: 进程意外终止
    VPS主机MSQL意外中断重启就好但10来个小时又中断的了如些反复
    使用hibernate连接mysql自动中断的问题
    40个国外联盟
    从服务里删除mysql
    外国广告联盟[16个]
    stm32学习笔记:GPIO外部中断的使用
    NO.2 设计包含min 函数的栈
    GPS数据,实测
    LATEX使用总结
  • 原文地址:https://www.cnblogs.com/pfypfy/p/8973393.html
Copyright © 2011-2022 走看看