题目描述

输入格式

输入数据第一行是图顶点的数量,一个正整数N。 接下来N行,每行N个字符。第i行第j列的1表示顶点i到j有边,0则表示无边。

输出格式

输出一行一个整数,表示该图的连通数。

样例

样例输入

3
010
001
100

样例输出

9

数据范围与提示

对于100%的数据,N不超过2000。

solution:

这道题给出三种算法:

DFS:

这道题搜索可以过

用vecter建边,若有一条由i指向j的边,那么把j压到i的vector中(这种建边方法好像比前向星快)

建立bool数组vis,vis[i][j]=1表示已经访问过由i指向j的边

我们对于每一个点进行dfs,其中dfs(i,j)表示以i为起点开始搜索,当前搜到了j点,依次向下dfs并统计答案

由于这里的vis存的是边的信息,所以不用考虑环。

 #include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<queue>
#include<vector>
#define MAXN 2005
using namespace std;
int n,ans[MAXN],res=;
char ch[MAXN][MAXN];
vector<int>mapa[MAXN];
bool vis[MAXN][MAXN];//vis[i][j]表示是否访问过由i向j的边
void dfs(int st,int now){//st:起点,now:当前节点
ans[st]++;
vis[st][now]=;
int m=mapa[now].size();
for(int i=;i<m;i++){
if(!vis[st][mapa[now][i]])
dfs(st,mapa[now][i]);
}
}
int main(){
scanf("%d",&n);
for(int i=;i<=n;i++){
scanf("%s",ch[i]+);
for(int j=;j<=n;j++){
if(ch[i][j]=='')
mapa[i].push_back(j);
//cout<<ch[i][j];
}
//cout<<endl;
}
for(int i=;i<=n;i++)
dfs(i,i),res+=ans[i];
printf("%d\n",res);
return ;
}

代码在这里!

TARJAN:

缩点再建图,这是最主流的方法,不再赘述:

#include<bits/stdc++.h>
using namespace std;
bitset<> t[];
int n,ans=,e[],in[];char s[];
int dfn[],low[],st[],num=,top=,cnt=;
bool ins[];
int tot=,first[],len[];
vector<int> edge[],scc[];
struct node{int v,next;}eg[];
inline void add(int x,int y)
{
eg[++tot].v=y;
eg[tot].next=first[x];
first[x]=tot;
}
inline void tarjan(int x)
{
dfn[x]=low[x]=++num;
st[++top]=x;ins[x]=;
for(register int i=;i<edge[x].size();i++)
{
int to=edge[x][i];
if(!dfn[to])
{
tarjan(to);
low[x]=min(low[x],low[to]);
}
else if(ins[to])
low[x]=min(low[x],dfn[to]);
}
if(dfn[x]==low[x])
{
cnt++;int y;
do{
y=st[top--],ins[y]=;
t[cnt][y]=,++len[cnt];
e[y]=cnt,scc[cnt].push_back(y);
}while(x!=y);
}
}
int main()
{
scanf("%d",&n);
for(register int i=;i<=n;i++)
{
scanf("%s",s+);
for(register int j=;j<=n;j++)
if(s[j]=='')edge[i].push_back(j);
}
for(register int i=;i<=n;i++)
if(!dfn[i])tarjan(i);
for(register int i=;i<=n;i++)
for(register int j=;j<edge[i].size();j++)
{
int to=edge[i][j];
if(e[i]==e[to])continue;
add(e[i],e[to]);
in[e[to]]++;
}
queue<int> q;
for(register int i=;i<=cnt;i++)
if(!in[i])q.push(i);
while(!q.empty())
{
int x=q.front();q.pop();
for(register int i=first[x];i;i=eg[i].next)
{
int to=eg[i].v;
t[to]|=t[x];
in[to]--;
if(!in[to])q.push(to);
}
}
for(register int i=;i<=n;i++)
ans+=t[i].count()*len[i];
printf("%d",ans);
}

BITSET:

我们借助c++STL解决问题,对于每个点建一个bitset;

bitset可以理解为一个加长的二进制数,普通二进制数的位运算bitset都能做,(&|~^>><<)

C++的 bitset 在 bitset 头文件中,它是一种类似数组的结构,它的每一个元素只能是0或1,每个元素仅用1bit空间。

定义:bitset<n>b//长度为n,名称为b的一个bitset

相关函数:

b.size() 返回大小(位数)

b.count() 返回1的个数

b.any() 返回是否有1

b.none() 返回是否没有1

b.set() 全都变成1

b.set(p) 将第p + 1位变成1

b.set(p, x) 将第p + 1位变成x

b.reset() 全都变成0

b.reset(p) 将第p + 1位变成0

b.flip() 全都取反

b.flip(p) 将第p + 1位取反

b.to_ulong() 返回它转换为unsigned long的结果,如果超出范围则报错

b.to_ullong() 返回它转换为unsigned long long的结果,如果超出范围则报错

b.to_string() 返回它转换为string的结果

至于它的时间复杂度,和电脑本身有关,一般来说是使复杂度/32

一段关于bitset的代码:

#include <cstdio>
#include <bitset>
#include <cstring>
#include <iostream>
#include <algorithm>
const int N=10;
std::bitset<N> a,b;//如果定义数组写成bitset<N> a[M];
int main(){
puts("stage 0");
std::cout<<a<<std::endl;
std::cout<<b<<std::endl;
a[1]=1;
b[0]=1;
puts("stage 1");
std::cout<<a[1]<<std::endl;
std::cout<<b[0]<<std::endl;
std::cout<<a<<std::endl;
std::cout<<b<<std::endl;
a=a|b;//等效于a|=b
puts("stage 2");
std::cout<<a<<std::endl;
a=a<<3;//等效于a<<=3
puts("stage 3");
std::cout<<a<<std::endl;
a=a>>3;//等效于a>>=3
puts("stage 4");
std::cout<<a<<std::endl;
a=a^b;//等效于a^=b
puts("stage 5");
std::cout<<a<<std::endl;
a=a&b;//等效于a&=b;
puts("stage 6");
std::cout<<a<<std::endl;
a.set();
puts("stage 7");
std::cout<<a<<std::endl;
a.reset();
puts("stage 8");
std::cout<<a<<std::endl;
a=~b;
puts("stage 9");
std::cout<<a<<std::endl;
std::cout<<b<<std::endl;
a[2]=0,a[5]=0;
puts("stage 10");
std::cout<<a<<std::endl;
std::cout<<a.count()<<" "<<a.size()<<std::endl;
b=a;
puts("stage 11");
std::cout<<b<<std::endl;
a=5;
puts("stage 12");
std::cout<<a<<std::endl;
return 0;
}
//时间复杂度,整体操作都是长度/32(64位机器除64),单个操作(操作单个位)是O(1)的
//空间复杂度,8位1字节,具体计算规则为在32位机器上Size = 4 * ((N + 31) / 32)在64位机器上Size = 8* ((N + 63) / 64)
    bitset<> foo ("");

    string s = foo.to_string();  //将bitset转换成string类型
unsigned long a = foo.to_ulong();  //将bitset转换成unsigned long类型
unsigned long long b = foo.to_ullong();  //将bitset转换成unsigned long long类型 cout << s << endl;  //
cout << a << endl;  //
cout << b << endl;  //
     bitset<> foo ("");

     cout << foo.flip() << endl;  //10011111  (flip函数传参数时,用于将参数位取反,本行代码将foo下标2处"反转",即0变1,1变0
cout << foo.flip() << endl;   //01100000  (flip函数不指定参数时,将bitset每一位全部取反 cout << foo.set() << endl;    //11111111  (set函数不指定参数时,将bitset的每一位全部置为1
cout << foo.set(,) << endl;  //11110111  (set函数指定两位参数时,将第一参数位的元素置为第二参数的值,本行对foo的操作相当于foo[3]=0
cout << foo.set() << endl;   //11111111  (set函数只有一个参数时,将参数下标处置为1 cout << foo.reset() << endl;  //11101111  (reset函数传一个参数时将参数下标处置为0
cout << foo.reset() << endl;   //00000000  (reset函数不传参数时将bitset的每一位全部置为0

以上三段代码均来自大佬博客和题解(%%)

了解了bitset,我们要用它来解题了

b[i]表示一种状态,若b[i][j]=1,则表示有一条由i向j连的边(i也要和自己连边)

若i能到j,那么i也能到j所能到的点,那么直接b[i]|b[j]即能把状态转移:

for(int j=1;j<=n;j++)
for(int i=1;i<=n;i++)
if(bit[i][j]) bit[i]|=bit[j];

这里一定是要j在外层,i在内层,原因博主还没有想通,有理解的欢迎在评论区留言

最后我们用bitset函数中的count()统计每个bitset中1的个数,然后输出答案

代码:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<queue>
#include<vector>
#include<bitset>
#define MAXN 2005
using namespace std;
int n,ans=0;
bitset<MAXN>bit[MAXN];
char st[MAXN];
int main(){
scanf("%d",&n);
for(int i=1;i<=n;i++){
scanf("%s",st+1);
for(int j=1;j<=n;j++)
if(st[j]=='1')
bit[i][j]=1;
bit[i][i]=1;
}
for(int j=1;j<=n;j++)
for(int i=1;i<=n;i++)
if(bit[i][j]) bit[i]|=bit[j];
for(int i=1;i<=n;i++)
ans+=bit[i].count();
printf("%d\n",ans);
return 0;
}

[JSOI2010]连通数 (dfs或tarjan或bitset)+bitset学习的更多相关文章

  1. 【BZOJ2208】[Jsoi2010]连通数 DFS

    [BZOJ2208][Jsoi2010]连通数 Description Input 输入数据第一行是图顶点的数量,一个正整数N. 接下来N行,每行N个字符.第i行第j列的1表示顶点i到j有边,0则表示 ...

  2. BZOJ 2208: [Jsoi2010]连通数( DFS )

    n只有2000,直接DFS就可以过了... -------------------------------------------------------------------------- #in ...

  3. BZOJ2208:[JSOI2010]连通数(DFS)

    Description Input 输入数据第一行是图顶点的数量,一个正整数N. 接下来N行,每行N个字符.第i行第j列的1表示顶点i到j有边,0则表示无边. Output 输出一行一个整数,表示该图 ...

  4. BZOJ 2208: [Jsoi2010]连通数 tarjan bitset

    2208: [Jsoi2010]连通数 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...

  5. bzoj2208 [Jsoi2010]连通数(scc+bitset)

    2208: [Jsoi2010]连通数 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1879  Solved: 778[Submit][Status ...

  6. BZOJ_2208_[Jsoi2010]连通数_强连通分量+拓扑排序+手写bitset

    BZOJ_2208_[Jsoi2010]连通数_强连通分量+拓扑排序+手写bitset Description Input 输入数据第一行是图顶点的数量,一个正整数N. 接下来N行,每行N个字符.第i ...

  7. 【BZOJ2208】[JSOI2010]连通数(Tarjan)

    [BZOJ2208][JSOI2010]连通数(Tarjan) 题面 BZOJ 洛谷 题解 先吐槽辣鸡洛谷数据,我写了个\(O(nm)\)的都过了. #include<iostream> ...

  8. bzoj2208:[Jsoi2010]连通数

    http://blog.csdn.net/u013598409/article/details/47037499 里面似乎有生成数据的... //我本来的想法是tarjan缩点之后然后将图遍历一遍就可 ...

  9. 2208: [Jsoi2010]连通数

    2208: [Jsoi2010]连通数 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1371  Solved: 557[Submit][Status ...

随机推荐

  1. 阿里云在云栖大会发布RPA最新3.4版本,将与达摩院联合探索人工智能领域

    9月26日,在2019年杭州云栖大会上,阿里云发布了RPA最新V3.4版本,全新升级了增加诸如录屏审计.JAVA应用录制能力.达摩院OCR内置组件.语法检查与智能提示能力增强等功能. RPA全名称Ro ...

  2. Delphi让所有的窗口的标题和图标显示在任务栏上

    Delphi:让所有的窗口的标题和图标显示在任务栏上在Delphi中,除了主窗口之外,当其它的窗口显示或切换到焦点时.默认情况下,窗口标题和图标并不会显示在任务栏中,为了实现像主窗口一样,每当窗口显示 ...

  3. py测试一个Socket实例

    本实例旨在了解py和socket的一些相关知识. 1.服务器端搭建py监听程序. 在客户端搭建python,linux默认自带了python2.7,先不管安装了. 接着编写socket程序,可以在本地 ...

  4. 通过apiservice反向代理访问service

    第一种:NodePort类型 type: NodePort ports: - port: 80 targetPort: 80 nodePort: 30008 ​ 第二种:ClusterIP类型 typ ...

  5. dashboard服务

    1.上传镜像,并导入,打标签 2.创建dashboard的deployment和service apiVersion: extensions/v1beta1 kind: Deployment meta ...

  6. USACO 2007 February Silver The Cow Lexicon /// DP oj24258

    题目大意: 输入w,l: w是接下来的字典内的单词个数,l为目标字符串长度 输入目标字符串 接下来w行,输入字典内的各个单词 输出目标字符串最少删除多少个字母就能变成只由字典内的单词组成的字符串 Sa ...

  7. Cefsharp实现快捷键功能

    原文:Cefsharp实现快捷键功能 1 . 实现IKeyboardHandler接口 public class KeyBoardHander : IKeyboardHandler { public ...

  8. 自己新机器安装CM时候 server服务启动DB配置

    com.cloudera.cmf.db.type=mysqlcom.cloudera.cmf.db.host=localhost:3306com.cloudera.cmf.db.name=cmfcom ...

  9. C++开发系列-纯虚函数和抽象类

    概念 纯虚函数和抽象类 纯虚函数是一个在基类中说明的虚函数,在基类中没有定义,要求任何派生类都实现该函数. 纯虚函数为各派生类提供了一个公共界面(接口的封装和设计.软件的模块功能的划分) 纯虚函数说明 ...

  10. php中heredoc使用方法

    Heredoc技术,在正规的PHP文档中和技术书籍中一般没有详细讲述,只是提到了这是一种Perl风格的字符串输出技术.但是现在的一些论坛程序,和部分文章系统,都巧妙的使用heredoc技术,来部分的实 ...