ZJNU 1269 - 灯塔——高级

Posted stelayuri

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了ZJNU 1269 - 灯塔——高级相关的知识,希望对你有一定的参考价值。

根据题目输入可以得到一个有向图

信号可以根据有向图的传递性传递,因此可以说是找到这个有向图的所有父亲即可

但又要考虑可能会出现环这类情况

所以跑一遍强连通分量模板,再根据分块后的图找到入度为0的块,把这些块当作信号发出源,就可以使全图都能够收到信号

所以答案就是入度为0的块的数量

(因为跑完程序刚好卡了时间限制,所以使用了缓冲区读入优化,最终程序耗时78ms)

 1 /*
 2 Written By StelaYuri
 3 */
 4 #pragma GCC optimize(3)
 5 #include<bits/stdc++.h>
 6 using namespace std;
 7 const int N=100005;
 8 vector<int> G[N];
 9 stack<int> S;
10 int pre[N],lowlink[N],sccno[N],dfs_clock,scc_cnt,ind[N];
11 const int bsz=1<<16;
12 char bf[bsz],*head,*tail;
13 inline char gc(){
14     if(head==tail){
15         int l=fread(bf,1,bsz,stdin);
16         tail=(head=bf)+l;
17     }
18     return *head++;
19 }
20 inline int read(){
21     int x=0;char c=gc();
22     while(!isdigit(c))c=gc();
23     for(;isdigit(c);c=gc())x=x*10+c-0;
24     return x;
25 }
26 inline void write(int x){
27     if(x>=10)write(x/10);
28     putchar(x%10+0);
29 }
30 void dfs(int in){
31     pre[in]=lowlink[in]=++dfs_clock;
32     S.push(in);
33     int num=G[in].size();
34     for(int i=0;i<num;i++){
35         int v=G[in][i];
36         if(!pre[v]){
37             dfs(v);
38             lowlink[in]=min(lowlink[in],lowlink[v]);
39         }
40         else if(!sccno[v])
41             lowlink[in]=min(lowlink[in],pre[v]);
42     }
43     if(lowlink[in]==pre[in]){
44         scc_cnt++;
45         while(1){
46             int x=S.top();
47             S.pop();
48             sccno[x]=scc_cnt;
49             if(x==in)
50                 break;
51         }
52     }
53 }
54 int main(){
55     int N=read(),M=read(),i,j,a,b;
56     for(i=0;i<M;i++){
57         a=read();
58         b=read();
59         G[a].push_back(b);
60     }
61     dfs_clock=scc_cnt=0;
62     memset(sccno,0,sizeof sccno);
63     memset(pre,0,sizeof pre);
64     for(i=1;i<=N;i++)
65         if(!pre[i])
66             dfs(i);
67     memset(ind,0,sizeof ind);
68     for(i=1;i<=N;i++){
69         a=G[i].size();
70         for(j=0;j<a;j++)
71             if(sccno[i]!=sccno[G[i][j]])
72                 ind[sccno[G[i][j]]]++;
73     }
74     for(a=0,i=1;i<=scc_cnt;i++)
75         if(!ind[i])
76             a++;
77     write(a);
78     
79     return 0;
80 }

 

以上是关于ZJNU 1269 - 灯塔——高级的主要内容,如果未能解决你的问题,请参考以下文章

ZJNU 1528 - War--高级

ZJNU 1244 - 森哥数——高级

ZJNU 1205 - 侦探推理——高级

ZJNU 1213 - 取水——高级

ZJNU 1310 - 排队——中高级

ZJNU 1223 - 素数距离——高级