zoukankan      html  css  js  c++  java
  • BZOJ1064 [Noi2008]假面舞会 【dfs】

    题目

    一年一度的假面舞会又开始了,栋栋也兴致勃勃的参加了今年的舞会。今年的面具都是主办方特别定制的。每个参加舞会的人都可以在入场时选择一 个自己喜欢的面具。每个面具都有一个编号,主办方会把此编号告诉拿该面具的人。为了使舞会更有神秘感,主办方把面具分为k (k≥3)类,并使用特殊的技术将每个面具的编号标在了面具上,只有戴第i 类面具的人才能看到戴第i+1 类面具的人的编号,戴第k 类面具的人能看到戴第1 类面具的人的编号。 参加舞会的人并不知道有多少类面具,但是栋栋对此却特别好奇,他想自己算出有多少类面具,于是他开始在人群中收集信息。 栋栋收集的信息都是戴第几号面具的人看到了第几号面具的编号。如戴第2号面具的人看到了第5 号面具的编号。栋栋自己也会看到一些编号,他也会根据自己的面具编号把信息补充进去。由于并不是每个人都能记住自己所看到的全部编号,因此,栋栋收集的信 息不能保证其完整性。现在请你计算,按照栋栋目前得到的信息,至多和至少有多少类面具。由于主办方已经声明了k≥3,所以你必须将这条信息也考虑进去。

    输入格式

    第一行包含两个整数n, m,用一个空格分隔,n 表示主办方总共准备了多少个面具,m 表示栋栋收集了多少条信息。接下来m 行,每行为两个用空格分开的整数a, b,表示戴第a 号面具的人看到了第b 号面具的编号。相同的数对a, b 在输入文件中可能出现多次。

    输出格式

    包含两个数,第一个数为最大可能的面具类数,第二个数为最小可能的面具类数。如果无法将所有的面具分为至少3 类,使得这些信息都满足,则认为栋栋收集的信息有错误,输出两个-1。

    输入样例

    6 5

    1 2

    2 3

    3 4

    4 1

    3 5

    输出样例

    4 4

    提示

    100%的数据,满足n ≤ 100000, m ≤ 1000000。

    题解

    根据题目的描述,所有人的关系形成一种环状关系
    假若给出的关系中存在环,那么环长一定是k的倍数
    设所有环长的gcd为x,此时答案为[x大于3的因子,x]

    假若没有环,结果就是所有最长链之和

    现在问题是如何求环以及最长链
    有一种dfs的方法很厉害
    我们将原边赋值为1,建一个反边赋值为-1,这样就构造出了一个类似无向图的东西,我们就可以从一个点出发访问整个联通块
    由于-1的存在,走反边会导致负值,走正边会形成正值,这样两点间长度就可以用差来求出
    跑dfs时,遇到了返祖边,则形成环,统计答案,同时统计这个联通块的最小权值和最大权值,只差 + 1即为最长链长度

    #include<iostream>
    #include<cmath>
    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    #define LL long long int
    #define REP(i,n) for (int i = 1; i <= (n); i++)
    #define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
    #define BUG(s,n) for (int i = 1; i <= (n); i++) cout<<s[i]<<' '; puts("");
    using namespace std;
    const int maxn = 100005,maxm = 2000005,INF = 1000000000;
    inline int read(){
    	int out = 0,flag = 1; char c = getchar();
    	while (c < 48 || c > 57) {if (c == '-') flag = -1; c = getchar();}
    	while (c >= 48 && c <= 57) {out = (out << 3) + (out << 1) + c - '0'; c = getchar();}
    	return out * flag;
    }
    int h[maxn],ne = 2;
    struct EDGE{int to,nxt,w;}ed[maxm];
    void build(int u,int v){
        ed[ne] = (EDGE){v,h[u],1}; h[u] = ne++;
        ed[ne] = (EDGE){u,h[v],-1}; h[v] = ne++;
    }
    int n,m,f[maxn],vis[maxn],pre[maxn],gmax[maxn],gmin[maxn],now;
    int ansl,ansr;
    int gcd(int a,int b){return b ? gcd(b,a % b) : a;}
    int find(int u){return u == pre[u] ? u : pre[u] = find(pre[u]);}
    void dfs(int u,int last){
    	gmax[now] = max(gmax[now],f[u]);
    	gmin[now] = min(gmin[now],f[u]);
    	vis[u] = true;
    	Redge(u) if ((k ^ 1) != last){
    		if (!vis[to = ed[k].to]) f[to] = f[u] + ed[k].w,dfs(to,k);
    		else {
    			ansr = gcd(abs(f[u] + ed[k].w - f[to]),ansr);
    			//printf("%d to %d
    ",u,to);
    		}
    	}
    }
    int main(){
    	n = read(); m = read();
    	int a,b,fa,fb;
    	for (int i = 1; i <= n; i++) pre[i] = i;
    	while (m--){
    		a = read(); b = read();
    		build(a,b);
    		fa = find(a); fb = find(b);
    		if (fa != fb) pre[fb] = fa;
    	}
    	for (int i = 1; i <= n; i++) if (!vis[i]) now = find(i),dfs(i,0);
    	if (ansr){
    		for (int i = 3; i <= ansr; i++) if (ansr % i == 0){ansl = i; break;}
    		if (ansl < 3) puts("-1 -1");
    		else printf("%d %d
    ",ansr,ansl);
    	}else {
    		for (int i = 1; i <= n; i++) if (find(i) == i) ansr += (gmax[i] - gmin[i] + 1);
    		if (ansr < 3) puts("-1 -1");
    		else printf("%d 3
    ",ansr);
    	}
    	return 0;
    }
    
    
  • 相关阅读:
    AngularJS 国际化——Angular-translate
    Elasticsearch Span Query跨度查询
    Elasticsearch 连接查询
    Elasticsearch DSL中Query与Filter的不同
    Lucene查询语法详解
    Elasticsearch 文件目录解释
    Elasticsearch 安装与启动
    《大屏可视化数据》该怎么设计?
    两个大屏可视化案例的布局与实现
    maven 安装后变成 mvn 不是内部命令解决方法
  • 原文地址:https://www.cnblogs.com/Mychael/p/8496468.html
Copyright © 2011-2022 走看看