【Link】:https://csacademy.com/contest/round-37/task/reconstruct-graph/statement/

【Description】



给你一张图;

包含n个点;m条边;

你可以在这张图的基础上添加边(不能有重边和自环);

使得节点1到节点i的最短距离为d[i]

问你可能的图的个数;

【Solution】



对于某一个节点x;

它只能和d的值为d[x]-1或者d[x]的节点直接相连

因为,d的值相同的x和y,如果它们俩之间连一条边的话,它们到节点1的距离不会变的;

而每个节点x;

它必然要和一个d值为d[x]-1的点相连;

不然,就不能得到x的d值为d[x]了;

x节点是不能和d值小于d[x]-1的点相连的,因为那样x的d值就会小于d[x];

这样;

对于x,考虑每一个d值为d[x]-1的点,(设总数为cnt),要么和它连,要么不连;

如果原图上,x和d值为d[x]-1的点,没有一个点相连,那么就要去除全都不连的情况,不然就错了,所以乘上2cnt−1

如果原图上,x和d值为d[x]-1的点,有temp个已经相连了;(temp>0)

则乘上2cnt−temp

然后,d的值相同的点之间,可以任意相连;

设d值为i的节点个数为bo[i],然后他们之间在原图上已经有temp条边;

则答案乘上2(bo[i]∗(bo[i]−1)2−temp)

每条边也都能选择连或不连

之后,考虑无解的情况;

如果d[i]出现了,而d[i]-1却没出现;

那么就无解;

如果d的值为0的点的个数大于1个,则无解;

如果原图上的一条边x->y

1<|d[x]−d[y]|

也无解



【NumberOf WA】



1



【Reviw】



构造题,一般有明显的结论的;

没多想,就跳过了;



【Code】

#include <bits/stdc++.h>
using namespace std;
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define LL long long
#define rep1(i,a,b) for (int i = a;i <= b;i++)
#define rep2(i,a,b) for (int i = a;i >= b;i--)
#define mp make_pair
#define pb push_back
#define fi first
#define se second
#define ms(x,y) memset(x,y,sizeof x)
#define Open() freopen("F:\\rush.txt","r",stdin)
#define Close() ios::sync_with_stdio(0) typedef pair<int,int> pii;
typedef pair<LL,LL> pll; const int dx[9] = {0,1,-1,0,0,-1,-1,1,1};
const int dy[9] = {0,0,0,-1,1,-1,1,-1,1};
const double pi = acos(-1.0);
const int N = 1e5;
const LL MOD = 1e9+7; int n,m,d[N+10],bo[N+10],num_edge[N+10],ma = 0;
vector <int> g[N+10]; bool wujie(){
rep1(i,1,n)
{
int len = g[i].size();
rep1(j,0,len-1){
int y = g[i][j];
if ( abs(d[y]-d[i])>1) return true;
}
} rep1(i,1,n) {
bo[d[i]]++;
ma = max(ma,d[i]);
} rep1(i,0,ma)
if (!bo[i]) return true; if (bo[0]>1) return true; return false;
} LL Pow(LL x,LL y){
LL temp = 1;
while (y){
if (y&1){
temp = (temp*x)%MOD;
}
x = (x*x)%MOD;
y>>=1;
}
return temp;
} int main(){
//Open();
//Close();
scanf("%d%d",&n,&m);
rep1(i,1,n)
scanf("%d",&d[i]);
rep1(i,1,m){
int x,y;
scanf("%d%d",&x,&y);
g[x].pb(y);
g[y].pb(x);
} if (wujie()) {
printf("0\n");
return 0;
} //计算同层内的边有多少条,顺便计算可能方案;
rep1(i,1,n){
int len = g[i].size();
rep1(j,0,len-1){
int y = g[i][j];
if (i < y && d[i]==d[y]){
num_edge[d[i]]++;
}
}
} LL ans = 1;
//计算和d[i]-1层的点有多少条边,顺便计算可能方案;
rep1(i,1,n)
if (i!=1){
int len = g[i].size(),temp = 0;
rep1(j,0,len-1){
int y = g[i][j];
if (d[y] == d[i]-1){
temp++;
}
}
int remain = bo[d[i]-1]-temp;
if (temp==0){
ans = ans*(Pow(2,remain)-1)%MOD;
}else{
ans = ans*Pow(2,remain)%MOD;
}
} //计算每一层可以连的边
rep1(i,0,ma){
ans = ans*Pow(2,1LL*bo[i]*(bo[i]-1)/2 - num_edge[i])%MOD;
}
printf("%lld\n",ans);
return 0;
} /*
写完之后,明确每一步的作用
*/

【CS Round #37 (Div. 2 only) D】Reconstruct Graph的更多相关文章

  1. 【CS Round #37 (Div. 2 only) B】Group Split

    [Link]:https://csacademy.com/contest/round-37/task/group-split/ [Description] 让你把一个数分成两个数a.b的和; (a,b ...

  2. 【CS Round #37 (Div. 2 only) A】Boring Number

    [Link]:https://csacademy.com/contest/round-37/task/boring-number/ [Description] 让你找离平均数最近的一个数的下标; [S ...

  3. 【CS Round #39 (Div. 2 only) C】Reconstruct Sum

    [Link]:https://csacademy.com/contest/round-39/task/reconstruct-sum/ [Description] 给你一个数字S; 让你找有多少对A, ...

  4. 【CS Round #36 (Div. 2 only) A】Bicycle Rental

    [题目链接]:https://csacademy.com/contest/round-36/task/bicycle-rental/ [题意] 让你从n辆车中选一辆车; 每一辆车有3个属性 1.到达车 ...

  5. 【CS Round #39 (Div. 2 only) D】Seven-segment Display

    [Link]:https://csacademy.com/contest/round-39/task/seven-segment-display/ [Description] 0..9各自有一个数字, ...

  6. 【CS Round #39 (Div. 2 only) B】Circle Elimination

    [Link]:https://csacademy.com/contest/round-39/task/circle-elimination/ [Description] [Solution] 把n个点 ...

  7. 【CS Round #39 (Div. 2 only) A】Removed Pages

    [Link]: [Description] [Solution] 每读入一个x; 把a[(x-1)/2]置为1即可; 统计1的个数 [NumberOf WA] [Reviw] [Code] /* */ ...

  8. 【CS Round #46 (Div. 1.5) E】Ultimate Orbs

    [链接]链接 [题意] n个人从左到右站在一条直线上.每个人都有一个能力值g[i],然后每个人可以将相邻的一个人打败. 然后它的能力值能够增加相应的能力值(就是打败了的那个人的能力值). A能够打败B ...

  9. 【CS Round #46 (Div. 1.5) C】Set Subtraction

    [链接]h在这里写链接 [题意] 一开始有n个数字,然后有一个数字X,把每个数字都减去X,又生成N个新的数字. 然后把这2*N个数字混在一起. 告诉你这2*N个数字是什么.让你复原出原来的N个数字,以 ...

随机推荐

  1. [Perl系列—] 2. Perl 中的引用使用方法

    Perl 中的引用,为什么要使用引用? 对于熟悉C语言的开发人员来说, 指针这个概念一定不陌生. Perl 的引用就是指针,能够指向变量.数组.哈希表甚至子程序. Perl5中的两种Perl引用类型为 ...

  2. nyoj--42--一笔画问题(并查集)

    一笔画问题 时间限制:3000 ms  |  内存限制:65535 KB 难度:4 描述 zyc从小就比较喜欢玩一些小游戏,其中就包括画一笔画,他想请你帮他写一个程序,判断一个图是否能够用一笔画下来. ...

  3. nyoj--108--士兵杀敌(一)(区间求和&&树状数组)

    士兵杀敌(一) 时间限制:1000 ms  |  内存限制:65535 KB 难度:3 描述 南将军手下有N个士兵,分别编号1到N,这些士兵的杀敌数都是已知的. 小工是南将军手下的军师,南将军现在想知 ...

  4. POJ1284 Primitive Roots (原根)

    题目链接:http://poj.org/problem?id=1284 题目描述: 题目大意: 一个质数原根的个数 题解: 结论题 一个数n的原根的个数等于$\varphi(\varphi(n))$ ...

  5. tf.nn.top_k(input, k, name=None)和tf.nn.in_top_k(predictions, targets, k, name=None)

    tf.nn.top_k(input, k, name=None) 这个函数的作用是返回 input 中每行最大的 k 个数,并且返回它们所在位置的索引. input: 一个张量,数据类型必须是以下之一 ...

  6. java编程思想--学习心得

    学习Java编程思想,需要了解语言特性,对于各种名词,能够借助项目代码,解释其含义,不借助搜索工具,明白其在什么样场景下使用,会带来什么样的问题,能否避免这类问题. 学习的过程,与软件开发相同,一样是 ...

  7. Java 异常的捕获与处理详解 (一)

    一,异常的产生(Exception) 异常是程序之中导致程序中断的一种指令流,异常一旦出现并且没有进行合理处理的话,那么程序就会中断执行. An exception is a flow of inst ...

  8. 二 JDK + mysql + yum + rpm

    如果系统环境崩溃.   调用/usr/bin/vim /etc/profile   1  网络搭建 2  host配置 3  SSH无密码登录   4  rpm 安装     yum install ...

  9. 编程算法 - 篱笆修理(Fence Repair) 代码(C)

    篱笆修理(Fence Repair) 代码(C) 本文地址: http://blog.csdn.net/caroline_wendy 题目: 把一块木板切成N块, 每次切两块, 分割的开销是木板长度, ...

  10. 【图像配准】基于互信息的图像配准算法:MI、EMI、ECC算法

    简单介绍: 基于互信息的图像配准算法以其较高的配准精度和广泛的适用性而成为图像配准领域研究的热点之中的一个.而基于互信息的医学图像配准方法被觉得是最好的配准方法之中的一个.基于此.本文将介绍简单的基于 ...