tarjan有向图缩点的基础应用。把原图中某点的连通数转化为反向图中”能够到达某点的个数“。缩点后,每个新点的贡献等于

  原dcc大小 * f[i]

  其中f[i]表示(包括该点自身)通向该点的点的个数。设u点为v的入度,满足转移方程:

    

  所以我们按照拓扑序dp求解即可。f[i]的初值设为该分量的节点数。

  这个题引出一个很重要的想法:如何避免两个强连通分量缩点时连有重边?对于2000的数据范围,一个二维布尔数组完全可以承受,但显然有更普适的优秀做法,这就是Hash。去重边实际上是二元组的判重问题,我们只需要一个合适的“进位”技术,就可以保证任意两个二元组所映射的键值是绝不相同的。如果key值太大,就要套用Hash表解决了。

  1. #include <iostream>
  2. #include <cstdio>
  3. #include <cstring>
  4. #include <queue>
  5. #include <cctype>
  6. #define maxn 2010
  7. using namespace std;
  8. template <typename T>
  9. void read(T &x) {
  10. x = 0;
  11. char ch = getchar();
  12. while (!isdigit(ch))
  13. ch = getchar();
  14. while (isdigit(ch))
  15. x = x * 10 + (ch ^ 48),
  16. ch = getchar();
  17. return;
  18. }
  19. struct E {
  20. int to, nxt;
  21. } edge[maxn * maxn], edge2[maxn * maxn];
  22. int n, head[maxn], top, head2[maxn], top2;
  23. inline void insert(int u, int v) {
  24. edge[++top] = (E) {v, head[u]};
  25. head[u] = top;
  26. }
  27. inline void insert2(int u, int v) {
  28. edge2[++top2] = (E) {v, head2[u]};
  29. head2[u] = top2;
  30. }
  31. int dfn[maxn], low[maxn], timer,
  32. sta[maxn], stp,
  33. c[maxn], cnt,
  34. w[maxn];
  35. bool ins[maxn];
  36. void tarjan_dfs(int u) {
  37. dfn[u] = low[u] = ++timer;
  38. sta[++stp] = u, ins[u] = true;
  39. for (int i = head[u]; i; i = edge[i].nxt) {
  40. int v = edge[i].to;
  41. if (!dfn[v])
  42. tarjan_dfs(v), low[u] = min(low[u], low[v]);
  43. else if (ins[v])
  44. low[u] = min(low[u], dfn[v]);
  45. }
  46. if (dfn[u] == low[u]) {
  47. ++cnt;
  48. int x;
  49. do {
  50. x = sta[stp--];
  51. ins[x] = false;
  52. c[x] = cnt;
  53. ++w[cnt];
  54. } while (x != u);
  55. }
  56. }
  57. void tarjan() {
  58. for (int i = 1; i <= n; ++i)
  59. if (!dfn[i]) tarjan_dfs(i);
  60. }
  61. namespace Hash_table {
  62. //  const int Size(23333309), step = 7;//空间足够,不用取模
  63. bool tb[4004001];
  64. inline int H (int u, int v) {
  65. return u * 2001 + v;
  66. }
  67. bool Hash(int u, int v) {
  68. int key = H(u, v);
  69. if (tb[key]) return false;
  70. tb[key] = true;
  71. return true;
  72. }
  73. } using namespace Hash_table;
  74. int ind[maxn];
  75. //bool done[maxn][maxn];//Hash更为优秀
  76. void build() {
  77. for (int u = 1; u <= n; ++u)
  78. for (int i = head[u]; i; i = edge[i].nxt) {
  79. int v = edge[i].to;
  80. if (c[u] != c[v] && Hash(c[u], c[v])) {
  81. insert2(c[u], c[v]);
  82. ++ind[c[v]];
  83. }
  84. }
  85. }
  86. long long sum = 0;
  87. long long f[maxn];
  88. queue<int> que;
  89. void dp() {
  90. for (int i = 1; i <= cnt; ++i) {
  91. f[i] = w[i];
  92. if (!ind[i]) {
  93. sum += w[i] * w[i];
  94. que.push(i);
  95. }
  96. }
  97. while (!que.empty()) {
  98. int u = que.front(); que.pop();
  99. for (int i = head2[u]; i; i = edge2[i].nxt) {
  100. int v = edge2[i].to;
  101. f[v] += f[u];
  102. --ind[v];
  103. if (!ind[v]) {
  104. sum += w[v] * f[v];
  105. que.push(v);
  106. }
  107. }
  108. }
  109. }
  110. int main() {
  111. read(n);
  112. for (int u = 1; u <= n; ++u)
  113. for (int v = 1; v <= n; ++v) {
  114. char ch = getchar();
  115. while (!isdigit(ch)) ch = getchar();
  116. if (ch == '1') insert(v, u);//反向存图
  117. }
  118. tarjan();
  119. build();
  120. dp();
  121. printf("%lld", sum);
  122. return 0;
  123. }

Luogu P4306 JSOI2010 连通数的更多相关文章

  1. Luogu P4306 [JSOI2010]连通数 传递闭包

    正解其实是\(Tarjan\) + \(拓扑拓扑\),但是却可以被\(O(N^3 / 32)\)复杂度的传递闭包水过去.心疼一下写拓扑的小可爱们. 学到一个\(bitset\)优化布尔图的骚操作,直接 ...

  2. P4306 [JSOI2010]连通数

    思路 要求求每个点能到达的点数就是传递闭包 然后n^3Floyd可做,但是n=2000,然后bitset压位 复杂度\(O(\frac{n^3}{32})\),能过 代码 #include <c ...

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

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

  4. bzoj2208:[Jsoi2010]连通数

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

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

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

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

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

  7. 2208: [Jsoi2010]连通数

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

  8. bzoj 2208 [Jsoi2010]连通数

    2208: [Jsoi2010]连通数 Time Limit: 20 Sec  Memory Limit: 512 MB Description Input 输入数据第一行是图顶点的数量,一个正整数N ...

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

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

随机推荐

  1. 1. HttpRunner介绍及环境准备

    介绍 HttpRunner 是一款面向 HTTP(S) 协议的通用测试框架 只需编写维护一份 YAML/JSON脚本,即可实现自动化测试.性能测试.线上监控.持续集成等多种测试需求 官方文档: htt ...

  2. python中,*args和**kwargs这两个参数的作用是什么?

    *args和**kwargs这两个都是不定长参数,可以解决函数中参数不固定的问题,*args可以把位置参数转化成元组,**kwagrs可以把关键字参数转化成字段

  3. GPRS DTU工作的原理与应用场景

    GPRS DTU是属于物联网无线数据终端设备的中一种,它主要是利用公用运营商的GPRS网络(又称G网)来为用户提供无线长距离数据传输的功能.一般都是采用的高性能工业级8/16/32位通信处理器和工业级 ...

  4. 程序员小抄——GitHub 热点速览 Vol.44

    作者:HelloGitHub-小鱼干 这周热点是什么?youtube-dl!就是那个超过 72+ star,又因为版权问题被 GitHub 关闭的项目,GitHub Trending 上一篇" ...

  5. .NetCore中简单使用EasyNetQ

    前言 我们在.Net中使用RabbitMQ,最原始的就是基于RabbitMQ.Client进行编码,在这个过程中我们需要通过代码约定和维护队列,Exchange等.如果是自行编码封装通用型的Rabbi ...

  6. Kubernetes-17:Kubernets包管理工具—>Helm介绍与使用

    Kubernets包管理工具->Helm 什么是Helm? 我们都知道,Linux系统各发行版都有自己的包管理工具,比如Centos的YUM,再如Ubuntu的APT. Kubernetes也有 ...

  7. DM的SQL优化入门笔记

    1.查看执行计划EXPLAIN SELECT A.C1+1,B.D2 FROM T1 A, T2 B WHERE A.C1 = B.D1; 2.执行计划: 1 #NSET2: [0, 16, 9] 2 ...

  8. nginx vhost配置

    server { listen 80; server_name crsdemo.my; index index.html index.htm index.php default.html defaul ...

  9. 服务器断电导致的ORACLE异常 : ORA-00214 ORA-01033 ORA-01034 ORA-00172 ORA-27101

    工作环境中的集群迁移之后,oracle出了挺多问题,最开始一直没找到原因,后来发现做迁移的人是冷迁移的,且数据库节点是硬关机的,惊了( 表现症状有不能登陆,登录了不能操作等 第一个报的是 ORA-00 ...

  10. Entity Fremework以及Fluentapi学习

    一.Entity Framework的入门  我这里采用的方式是数据库自己建立  然后模型类自己建立 数据库上下文类自己建立的方式 目的在于弄懂原理 其他的数据库优先等方式这里就不写了  教程有很多. ...