当前位置: 首页 > article >正文

见面礼——图论

给定一个 n 个点 n 条边的无向图,你需要求有多少种选择图上的一个点 p 和一条边 (x,y) 的方案,使得删去 (x,y) 后图变成一棵树,且这棵树以 p 为根时每个节点的儿子个数均不超过 3。保证至少存在一种这样的方案。

Input
输入的第一行一个整数 n(2≤n≤105) 表示节点数,接下来 n 行每行两个整数 x,y(1≤x,y≤n) 描述图上的一条边。保证图中没有重边自环。

Output
输出一行一个正整数表示答案。

Input
6
1 2
1 3
1 4
1 5
1 6
2 3

Output
10

解析:

n个点n条边,所以该图就成一个环。只有将环中的一条边删去,该图才能变为一棵树。

#include <bits/stdc++.h>
using namespace std;
#define int long long 
#define ios ios::sync_with_stdio(false),cin.tie(0),cout.tie(0);
typedef pair<int,int> PII;
const int N=2e6+10;
vector <int> g[N];
map <int,int> k;
int p[N];
int d[N];
bool vis[N];
vector <PII> q;  //储存成环的边
int n;
void dfs(int u,int pa)
{
    p[u]=pa;
    vis[u]=1;
    for (auto v:g[u])
    {
        if (v==pa) continue;
        if (vis[v]==1&&q.size()==0)    //当点 v 被再次遍历时,现在已经建成一个环了,就可以回溯将环中的每条边放入队列 q 中
        {
            q.push_back({u,v});
            while (p[u]!=v)
            {
                q.push_back({u,p[u]});
                u=p[u];
            }
            q.push_back({u,p[u]});
        }
        if (vis[v]==1) continue;  //走过的点,不用继续操作了,否则会死循环
        dfs(v,u);
    }
}
signed main()
{
    ios;
    cin>>n;
    for (int i=1;i<=n;i++)
    {
        int u,v;
        cin>>u>>v;
        g[u].push_back(v);
        g[v].push_back(u);
        d[u]++;
        d[v]++;
    }
    for (int i=1;i<=n;i++)
    {
        k[d[i]]++;  //记录度数相同的点的数量
    }
    //for (auto x:k) cout<<x.first<<" "<<x.second<<endl;
    dfs(1,0);
    //for (auto x:q) cout<<x.first<<" "<<x.second<<endl;
    int ans=0;
    for (auto x:q)   //遍历每条要删掉的边
    {
        int du=d[x.first];
        int dv=d[x.second];
        k[du]--;
        k[dv]--;
        k[du-1]++;
        k[dv-1]++;
        int res=0;
        bool flag=0;
        for (auto y:k)       //删除边后,再遍历每个点,判断能否成为根节点
        {
            int cnt=y.first;
            int s=y.second;
            if (cnt<=3) res +=s;
            if (cnt>=5&&s>0) flag=1;    //既当不了根节点,也当不了儿子节点
        }
        if (flag==0) ans +=res;
        k[du]++;  //还原
        k[dv]++;
        k[du-1]--;
        k[dv-1]--;
    }
    cout<<ans;
    return 0;
}


http://www.kler.cn/a/135347.html

相关文章:

  • 执行flink sql连接clickhouse库
  • sol机器人pump机器人如何实现盈利的?什么是Pump 扫链机器人?
  • Axure设计之文本编辑器制作教程
  • 【插件】多断言 插件pytest-assume
  • 结构体是否包含特定类型的成员变量
  • three.js 杂记
  • 01_SHELL编程之变量定义(一)
  • 【Python】12 GPflow安装
  • 大批量合并识别成一个表或文档的方法
  • P8611 [蓝桥杯 2014 省 AB] 蚂蚁感冒(模拟)
  • UEFI实战——键盘操作
  • reset函数的使用
  • java桌面程序
  • leetcode刷题之哈希表的应用(1)
  • Java(三)(static,代码块,单例设计模式,继承)
  • Swagger-----knife4j框架
  • 迎接“全全闪”时代 星辰天合发布星海架构和星飞产品
  • 集群路径规划学习(一)之EGO-swarm仿真
  • Kettle 简介
  • 读《Segment Anything in Defect Detection》
  • Spring Boot 中使用 ResourceLoader 加载资源的完整示例
  • 使用 Amazon VPS 探索存储选项:实用指南
  • Flutter 3.16 中带来的更新
  • selenium长截图,截图整个页面
  • (C)一些题2
  • 成为电车销量的“中坚力量”,微小型车不能只有“低价”?