zoukankan      html  css  js  c++  java
  • #Tarjan#洛谷 5676 [GZOI2017]小z玩游戏

    题目


    分析

    可能玩两次也就是形成环即是Tarjan缩点后在同一个强连通分量
    如果按照游戏连边数量将达到(O(n^2)),当中其实有很多边可以共用,
    考虑(i)连向(i)的倍数,以及有趣程度连接兴奋程度,其实连接倍数可以优化一下,
    比如说(i)连向(j)(j)连向(k),那么(i)连向(k)的边完全可以被省掉,
    那么对于每个数连接其与一个质数的乘积,根据埃氏筛建的边应该为(O(nloglogn))
    实际上仍然达不到上界,总边数在(4n)以内


    代码

    #include <cstdio>
    #include <cctype>
    #include <cstring>
    #define rr register
    using namespace std;
    const int M=100000,N=M|15; struct node{int y,next;}e[N<<2];
    int dfn[N],stac[N],low[N],et,eT,v[N],St[N],Ed[N],ans;
    int Cnt,col[N],Top,as[N],prime[N],bs[N],n,tot,cnt;
    inline signed iut(){
    	rr int ans=0; rr char c=getchar();
    	while (!isdigit(c)) c=getchar();
    	while (isdigit(c)) ans=(ans<<3)+(ans<<1)+(c^48),c=getchar();
    	return ans;
    }
    inline void print(int ans){
    	if (ans>9) print(ans/10);
    	putchar(ans%10+48);
    }
    inline signed min(int a,int b){return a<b?a:b;}
    inline void add(int x,int y){e[++et]=(node){y,as[x]},as[x]=et;}
    inline void tarjan(int x){
    	dfn[x]=low[x]=++tot,stac[++Top]=x,v[x]=1;
    	for (rr int i=as[x];i;i=e[i].next)
    	if (!dfn[e[i].y]){
    		tarjan(e[i].y);
    		low[x]=min(low[x],low[e[i].y]);
    	}else if (v[e[i].y])
    	    low[x]=min(low[x],dfn[e[i].y]);
    	if (dfn[x]==low[x]){
    		rr int y; ++cnt;
    		do{
    			y=stac[Top--];
    			col[y]=cnt,v[y]=0;
    		}while (y!=x);
    	}
    }
    signed main(){
    	for (rr int i=2;i<=M;++i){
    		if (!v[i]) prime[++Cnt]=i;
    		for (rr int j=1;j<=Cnt&&prime[j]<=M/i;++j){
    			v[i*prime[j]]=1;
    			if (i%prime[j]==0) break;
    		}
    	}
    	for (rr int i=1;i<=M;++i)
    	for (rr int j=1;j<=Cnt&&prime[j]<=M/i;++j)
    	    add(i,i*prime[j]);
    	memcpy(bs,as,sizeof(as)),eT=et;
    	for (rr int T=iut();T;--T){
    		memcpy(as,bs,sizeof(bs)),et=eT;
    		memset(dfn,0,sizeof(dfn));
    		memset(v,0,sizeof(v));
    		memset(col,0,sizeof(col));
    		memset(low,0,sizeof(low));
    		n=iut(),ans=tot=0;
    		for (rr int i=1;i<=n;++i) St[i]=iut();
    		for (rr int i=1;i<=n;++i) Ed[i]=iut();
    		for (rr int i=1;i<=n;++i) add(St[i],Ed[i]);
    		tarjan(1);
    		for (rr int i=1;i<=n;++i)
    		if (col[St[i]]==col[Ed[i]]) ++ans;
    		print(ans),putchar(10);
    	}
    	return 0;
    } 
    
  • 相关阅读:
    如何借助BM算法轻松理解KMP算法
    如何实现文本编辑器中的查找替换功能?——BF算法
    C++中求数组长度与memset的用法
    什么是素数/质数/合数
    深度和广度优先搜索:如何找出社交网络中的三度好友关系?
    如何存储微博、微信等社交网络中的好友关系?
    为什么说堆排序没有快速排序快?
    HTML5中标记与特殊属性
    margin外边距问题
    html热点区域
  • 原文地址:https://www.cnblogs.com/Spare-No-Effort/p/14488619.html
Copyright © 2011-2022 走看看