首页 > 其他分享 >[刷题笔记] [JSOI2010] 连通数

[刷题笔记] [JSOI2010] 连通数

时间:2023-08-10 23:24:10浏览次数:46  
标签:JSOI2010 连通 ch int pos 连通数 topsort include 刷题

Description

Problem

由于题目太短我直接上图罢
image

Analysis

题目描述非常简单,但是直接爆搜肯定会TLE,毕竟 \(n\leq 2000\)并且time limit=300ms。

我们发现如果题目保证无环直接topsort即可,问题就在环上,如何处理环呢?

我们可以缩点,缩点笔记,显然我们只需要统计答案数,缩完点后就变成了一个DAG,在DAG上跑topsort,统计答案统计上两个强连通分量祖先的儿子数乘积即可,显然若一个强连通分量祖先可以到达另一个强连通分量祖先,那么它们的儿子一定互相可达。

至于如何判断一个强连通分量可以到达另一个强连通分量呢?上文提到显然可以topsort,在topsort的时候可以开一个bitset优化一下,显然两个强连通分量祖先之间的关系要么为0要么为1(可达or不可达),直接bitset做运算继承关系即可。如果看起来很抽象见代码。

本题思路就这么简单,实现的时候需要注意很多细节。A了这道题,我对缩点有了更深刻的认识呢qwq

Code

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <vector>
#include <queue>
#include <bitset>
#include <string>
#include <stack>
using namespace std;
const int N=20500,M=4000500;
int n;
vector <int> Edge[N];
int dfn[N],low[N];
int cnt = 0;
int in_stack[N];
stack <int> s;
int vis[N];
int fa[N];
int fa_cnt = 0;
int fa_num[N];
bitset <N> new_map[N]; //bitset继承两个强连通分量祖先的关系
vector <int> new_Edge[N];
int ans = 0;
int in[N];
int v[N];
inline int read(){
    register int x = 0, t = 1;
    register char ch=getchar(); 
    while(ch<'0'||ch>'9'){
        if(ch=='-')
            t=-1;
        ch=getchar();
    }
    while(ch>='0'&&ch<='9'){
        x=(x<<1)+(x<<3)+(ch^48);  
        ch=getchar();
    }
    return x*t;
}
inline void write(int x)
{
    if(x<0){
    	putchar('-');
		x=-x;
	}
    if(x>9) 
		write(x/10);
    putchar(x%10+'0');
}
int gc(){
    int x=getchar();
    if(x!='0'&&x!='1'){
        return gc();
    }
    else{
        return x;
    }
}
void Tarjan(int pos)
{
    dfn[pos] = low[pos] = ++cnt;
    v[pos] = 1;
    in_stack[pos] = 1;
    s.push(pos);
    for(int i=0;i<Edge[pos].size();i++)
    {
        if(!dfn[Edge[pos][i]])
        {
            Tarjan(Edge[pos][i]);
            low[pos] = min(low[pos],low[Edge[pos][i]]);
        }
        else if(in_stack[Edge[pos][i]])
        {
            low[pos] = min(low[pos],dfn[Edge[pos][i]]);
        }
    }
    if(dfn[pos] == low[pos])
    {
        fa_cnt ++;
        while(s.top() != pos)
        {
            fa_num[fa_cnt] ++;
            v[s.top()] = 0;
            fa[s.top()] = fa_cnt;
            in_stack[s.top()] = 0;
            s.pop();
        }
        s.pop();
        fa_num[fa_cnt] ++;
        fa[pos] = fa_cnt;
        in_stack[pos] = 0;
        new_map[fa_cnt][fa_cnt] = 1; //显然每个祖先自己一定可以到达自己
    }
}
void topsort()
{
    queue <int> q; //使用队列进行topsort
    for(int i=1;i<=fa_cnt;i++)
    {
        if(!in[i]) q.push(i);
    }
    while(!q.empty())
    {
        int noww = q.front();
        q.pop();
        for(int i=0;i<new_Edge[noww].size();i++)
        {
            new_map[new_Edge[noww][i]] |= new_map[noww]; //继承关系,显然只要有一个点可以到达另一个点则它们都可以到达那个点
            in[new_Edge[noww][i]] --; //每个节点的入度
            if(!in[new_Edge[noww][i]]) q.push(new_Edge[noww][i]);
        }
    }
}
int main()
{
    n = read();
    for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=n;j++) 
        {
            char flg=gc(); //奇怪的输入被卡了....
            if(flg=='1') Edge[i].push_back(j);
        }
    }
    for(int i=1;i<=n;i++) 
    {
        if(!dfn[i]) Tarjan(i); //我们不保证原图一定联通,需要以每个点为root进行Tarjan,这个前提显然是该节点没有被访问,因为如果被访问过则已经操作了
    }
    for(int i=1;i<=n;i++)
    {
       for(int j=0;j<Edge[i].size();j++)
       {
            if(fa[i] != fa[Edge[i][j]])  //建反图
            {
                bool can_push = true;
                for(int k=0;k<new_Edge[fa[Edge[i][j]]].size();k++)
                {
                    if(new_Edge[fa[Edge[i][j]]][k] == fa[i]) 
                    {
                        can_push = false;
                        break;
                    }
                }
                if(can_push)
                {
                    new_Edge[fa[Edge[i][j]]].push_back(fa[i]);
                    in[fa[i]] ++;
                }
            }
       }
    }
    topsort();
    for(int i=1;i<=fa_cnt;i++)
    {
        for(int j=1;j<=fa_cnt;j++)
        {
            if(new_map[i][j]) ans += fa_num[i]*fa_num[j];//统计答案
        }
    }
    write(ans);
    return 0;
}

标签:JSOI2010,连通,ch,int,pos,连通数,topsort,include,刷题
From: https://www.cnblogs.com/SXqwq/p/17621865.html

相关文章

  • 2023/8刷题记录
    2023/8刷题记录luogu-P6885向黑板上写数字,左右写,求所有序列的最长\(LIS\)并统计\(LIS\)的个数。向左边放和向右边放相当于把一个序列拆成两个子序列,然后把第一个翻转接在第二个前面。整个序列的\(LIS\)就相当于第一个序列的\(LDS\)和第二个子序列的\(LIS\)接起来,但......
  • 刷题准备
    算法名称算法链接刷题范围(leetcode)排序算法排序56、147、220、252堆最大堆,最小堆问题 215、253、347、624、703分治法分治1 分治24、23、53、215、240、327回溯法回溯1 回溯210、17、22、39、46(经典回溯)、1239贪心算法贪心1 贪心2253、406、......
  • LRU机制:哈希表+双向链表 [labuladong-刷题打卡 day9]
    今天的知识点LRU缓存机制的实现。学过计组都知道LRU算法(leastrecentlyused最近最少使用算法)是资源管理中的常用算法。那么他是如何实现的呢?LRU原理和Redis实现146.LRU缓存此题算是对LRU机制的一个简化。为了使查找删除在O(1)中实现,我们结合哈希表和双向链表各自在查找和......
  • [刷题笔记] Luogu P1280 尼克的任务
    ProblemAnalysis首先,如果一个时间只有一个任务开始,则她必须做。如果一个时间有多个任务开始,她可以选一个去做。我们发现这样的决策是取决于后面的空暇时间,而不是前面。所以在dp的时候需要从后往前搜时间(当然如果从前往后可以跑记搜)考虑转移,如果一个时间有多个任务开始,则选一个......
  • Leetcode刷题记录本
    Leetcode刷题记录本ID:1点击查看代码暴力破解法classSolution(object):deftwoSum(self,nums,target):""":typenums:List[int]:typetarget:int:rtype:List[int]""" #暴力破解法fori......
  • 递归反转链表局部[labuladong-刷题打卡 day8]
    写在前面前两天刷题打卡,感觉东哥的代码模板没有题解中的简洁,或者那些极限优化的代码中有很多优化技巧,但今天去感受递归的含义的时候,觉得毕竟我现在是在学习算法,理解算法含义才是学习的意义。至于优化,那是之后的事,所以刷题的时候不必过于追求简洁,就像追求简洁而降低可读性一样属......
  • 【刷题笔记】9. Palindrome Number
    题目Determinewhetheranintegerisapalindrome.Aninteger is a palindromewhenit readsthesamebackwardasforward.Example1:Input:121Output:trueExample2:Input:-121Output:falseExplanation:Fromlefttoright,itreads-121.Fromrightto......
  • 刷题记录(二)
    catcat-new点击主页的一个链接http://61.147.171.105:55571/info?file=Persiancat.txt可能存在任意文件读取漏洞,读取/etc/passwd文件读取当前进程的命令行参数?file=../../proc/self/cmdline,发现有一个通过python启动app.py的命令,得知网站使用的是python的flask框架。读取a......
  • 【刷题笔记】8. String to Integer (atoi)
    题目Implementthe myAtoi(strings) function,whichconvertsastringtoa32-bitsignedinteger(similartoC/C++'s atoi function).Thealgorithmfor myAtoi(strings) isasfollows:Readinandignoreanyleadingwhitespace.Checkifthenextcharact......
  • 【刷题笔记】7. Reverse Integer
    题目Givena32-bitsignedinteger,reversedigitsofaninteger.Example1:Input:123Output:321Example2:Input:-123Output:-321Example3:Input:120Output:21**Note:**Assumewearedealingwithanenvironmentwhichcouldonlystoreintegerswi......