UVA-818 切断圆环链 题解答案代码 算法竞赛入门经典第二版

news2024/9/22 5:31:19

GitHub - jzplp/aoapc-UVA-Answer: 算法竞赛入门经典 例题和习题答案 刘汝佳 第二版

题目要求切断部分圆环后再链接起来成为一个链,要求切断的圆环最少。注意链不需要按照序号1-n排列,可以乱序。

我一开始的方法好像可以解决问题,但是大幅超时。方法是对每个环断开再对其它圆环每个尝试连接,试图遍历所有场景找出可以形成一条链的组合。但是这种方法产生了不少重复遍历场景(断开的先后顺序不重要),且实际上不需要尝试链接,就能判断是否组成一条链,因此这里时间复杂度太高,大幅超时了。(超时代码放到最后了)

后来参考网上的方法:

1. 遍历所有断开的圆环的组合。设有n个圆环,这里使用的是对 0 ~ (2^n - 1)进行遍历。其中给每个数字了包含了圆环是否断开的二进制数据。getBreakPoint函数是将二进制数字转换为断开的点数组。

2. breakMap函数,将断开的圆环和其它的圆环的链接删掉,组成一个新的图。因为圆环是断开后就不和所有圆环有关系了。

3. judge函数,判断 断开后的圆环组合是否符合规范。这里有几个点:

3.1. 首先看每个节点的度,如果度超过2,说明有一个点连接了超过两个点,那肯定需要再断开其他圆环才能形成链,说明这个断开组合不符合要求。

3.2. 看这个图中是否还有多个圆环组成的“环”。有的话说明再断开其他圆环才能形成链,不符合要求。判断方法是dfs遍历,遍历一条边就删掉一条边,然后看有没有点访问次数超过2。

3.3. 统计没有联通的子图的数量,设为m。设断开的圆环数量为a。如果m - 2 * a - 1 <= 0,则说明这个图符合要求,可以组成一条链。

其中3.3即是“实际上不需要尝试链接,就能判断是否组成一条链”的公式。这里来解释一下:

首先断开的圆环,是可以链接其他已有的每个连通子图的。一个断开的圆环可以链接两个子图。但是,链接两个子图后,就形成了一个新的子图。所以事实上a个断开的圆环可以链接a+1个子图。再加上断开的圆环本身也是一个子图。所以,2*a+1是可以连接的最大子图数量。如果子图数量比它要大,那就说明无法链接成一条完整的链了。

AC代码

#include <stdio.h>
#include <string.h>
#define MAXN 17

// 序号从1到n
int map[MAXN][MAXN];
int n;

// 保存当前断开场景下的图
int mapTemp[MAXN][MAXN];
// 当前断开的点
int breakPoint[MAXN];
// judge中点是否访问过
int findPoint[MAXN];

void init()
{
  memset(map, 0, sizeof(map));
}

// 复制一份图
void copyMap()
{
  int i, j;
  for (i = 1; i <= n; ++i)
    for (j = 1; j <= n; ++j)
      mapTemp[i][j] = map[i][j];
}

// 由二进制数字转换为断开的点数组
int getBreakPoint(int cnt)
{
  int i, j, k = 0;
  for (i = 0; i < n; ++i)
  {
    j = (cnt / (1 << i)) % 2;
    if (j)
      ++k;
    breakPoint[i + 1] = j;
  }
  return k;
}

// 根据断开点数组来断开图
void breakMap()
{
  int i, j;
  for (i = 1; i <= n; ++i)
  {
    if (!breakPoint[i])
      continue;
    for (j = 1; j <= n; ++j)
    {
      mapTemp[i][j] = 0;
      mapTemp[j][i] = 0;
    }
  }
}

void dfs(int point)
{
  ++findPoint[point];
  for (int j = 1; j <= n; ++j)
  {
    if (!mapTemp[point][j] || j == point)
      continue;
    // 访问过的边就删掉,避免重复访问
    // 对应的,已访问过的节点可以重复访问
    mapTemp[point][j] = 0;
    mapTemp[j][point] = 0;
    dfs(j);
  }
}

// 判断有几个链,以及是否有环
int judge()
{
  // 链数 单个节点也是链
  int linkNum = 0;
  int i, j, k;

  // 首先看每个节点的度
  for (i = 1; i <= n; ++i)
  {
    k = 0;
    for (j = 1; j <= n; ++j)
      if (mapTemp[i][j])
        ++k;
    if (k > 2)
      return -1;
  }
  // 然后看每个节点是否有环
  memset(findPoint, 0, sizeof(findPoint));
  for (i = 1; i <= n; ++i)
  {
    if (findPoint[i])
      continue;
    ++linkNum;
    dfs(i);
    for (j = 1; j <= n; ++j)
    {
      if (findPoint[j] >= 2)
        return -1;
    }
  }
  return linkNum;
}

int computed()
{
  int i, j, k;
  int breakMin = n, breakNum;
  int linkNum;
  for (i = 0; i < (1 << n); ++i)
  {
    copyMap();
    breakNum = getBreakPoint(i);
    if (breakNum >= breakMin)
      continue;
    breakMap();
    linkNum = judge();
    if (linkNum < 0)
      continue;
    if (linkNum - breakNum * 2 - 1 > 0)
      continue;
    breakMin = breakNum;
  }
  return breakMin;
}

int main()
{
  int cnt = 0;
  int i, j, sum = 0;
  while (scanf("%d", &n) == 1 && n > 0)
  {
    ++cnt;
    init();
    while (scanf("%d %d", &i, &j) == 2 && i > 0)
    {
      map[i][j] = 1;
      map[j][i] = 1;
    }
    if (n > 1)
      sum = computed();
    printf("Set %d: Minimum links to open is %d\n", cnt, sum);
  }
  return 0;
}

超时代码

#include <stdio.h>
#include <string.h>
#define MAXN 17

// 序号从1到n
int map[MAXN][MAXN];
// 每个点是否已经重新设置
int setFlag[MAXN];
// 暂存每一层中被清空绑定关系的数据
int setFlagMap[MAXN][MAXN];
int n;
int sum;
// 判断是否联通图需要的记录
int mapFlag[MAXN];

void init()
{
  memset(map, 0, sizeof(map));
  memset(setFlag, 0, sizeof(setFlag));
  memset(setFlagMap, 0, sizeof(setFlagMap));
  sum = n;
}

void dfs(int i)
{
  mapFlag[i] = 1;
  for (int j = 1; j <= n; ++j)
  {
    if (mapFlag[j] || !map[i][j])
      continue;
    dfs(j);
  }
}

// 两个端点的度为1,其余的所有点度为2
bool judge()
{
  int i, j;
  int num1 = 0, num2 = 0, count;
  for (i = 1; i <= n; ++i)
  {
    count = 0;
    for (j = 1; j <= n; ++j)
      if (map[i][j])
        ++count;
    if (count == 1)
      ++num1;
    else if (count == 2)
      ++num2;
    else
      return false;
  }
  if (num1 != 2 && num2 != n - 2)
    return false;
  // 判断是否是联通图
  memset(mapFlag, 0, sizeof(mapFlag));
  dfs(1);
  for (i = 1; i <= n; ++i)
    if (!mapFlag[i])
      return false;
  return true;
}

void computed(int cnt)
{
  if (cnt >= sum)
    return;
  if (judge())
  {
    sum = cnt;
    return;
  }
  int i, j, k;
  // 选中i作为open
  for (i = 1; i <= n; ++i)
  {
    if (setFlag[i])
      continue;
    setFlag[i] = 1;
    // 清空i的绑定关系
    memset(setFlagMap[cnt + 1], 0, MAXN * sizeof(int));
    for (j = 1; j <= n; ++j)
    {
      setFlagMap[cnt + 1][j] = map[i][j];
      map[i][j] = 0;
      map[j][i] = 0;
    }

    // 绑定一个的情况
    for (j = 1; j <= n; ++j)
    {
      if (j == i)
        continue;
      map[i][j] = 1;
      map[j][i] = 1;
      computed(cnt + 1);
      map[i][j] = 0;
      map[j][i] = 0;
    }
    // 绑定两个的情况
    for (j = 1; j <= n; ++j)
    {
      if (j == i)
        continue;
      for (k = 1; k <= n; ++k)
      {
        if (k == i || k == j)
          continue;
        map[i][j] = 1;
        map[j][i] = 1;
        map[i][k] = 1;
        map[k][i] = 1;
        computed(cnt + 1);
        map[i][j] = 0;
        map[j][i] = 0;
        map[i][k] = 0;
        map[k][i] = 0;
      }
    }

    // 恢复原有的绑定关系
    for (j = 1; j <= n; ++j)
    {
      map[i][j] = setFlagMap[cnt + 1][j];
      map[j][i] = setFlagMap[cnt + 1][j];
    }
    setFlag[i] = 0;
  }
}

void printMap()
{
  int i, j;
  for (i = 1; i <= n; ++i)
  {
    for (j = 1; j <= n; ++j)
      printf("%d ", map[i][j]);
    putchar('\n');
  }
  putchar('\n');
}

int main()
{
  int cnt = 0;
  int i, j;
  while (scanf("%d", &n) == 1 && n > 0)
  {
    ++cnt;
    init();
    while (scanf("%d %d", &i, &j) == 2 && i > 0)
    {
      map[i][j] = 1;
      map[j][i] = 1;
    }
    if (n > 1)
      computed(0);
    printf("Set %d: Minimum links to open is %d\n", cnt, sum);
  }
  return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2154273.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

AI健身之俯卧撑计数和姿态矫正-角度估计

在本项目中&#xff0c;实现了Yolov7-Pose用于人体姿态估计。以下是如何在Windows 11操作系统上设置和运行该项目的详细步骤。 环境准备 首先&#xff0c;确保您的计算机已经安装了Anaconda。Anaconda是一个开源的Python发行版本&#xff0c;它包含了conda、Python以及众多科…

静态链表:实现、操作与性能优势【算法 16】

静态链表&#xff1a;实现、操作与性能优势 在算法和数据结构的探索中&#xff0c;链表作为一种基础且灵活的数据结构&#xff0c;广泛应用于各种场景。然而&#xff0c;在算法竞赛或需要高效内存管理的环境中&#xff0c;传统的动态链表可能会因为内存分配和释放的开销而影响性…

完整版:NacosDocker 安装

第一步&#xff1a;先直接通过命令安装 Nacos docker run --name nacos2.2.3 -d -p 8848:8848 -e MODEstandalone f151dab7a111 第二步&#xff1a;创建 Docker 挂载目录 # 创建 log 目录 mkdir -p /root/nacos 第三步&#xff1a;将 Docker 容器的文件复制到挂载目录中 …

C++之Person类

首先设置头文件&#xff0c;将题目中的要求完成。 #include <iostream>using namespace std;class Person { public:Person();Person(string name, int id, string address);~Person();void setPerson(string name, int id, string address);void setName(string name);…

python新手的五个练习题

代码 # 1. 定义一个变量my_Number,将其设置为你的学号&#xff0c;然后输出到终端。 my_Number "20240001" # 假设你的学号是20240001 print("学号:", my_Number) # 2. 计算并输出到终端:两个数(例如3和5)的和、差、乘积和商。 num1 3 num2 5 print(&…

计算机的错误计算(一百)

摘要 探讨 与 的计算精度问题。 从计算机的错误计算&#xff08;九十九&#xff09;知&#xff0c;运算 与 均被列在IEEE754-2019中。然而&#xff0c;似乎并没有哪种语言实现内置了第二个运算。 例1. 计算 与 不妨在Python 3.12.5 下计算&#xff0c;则有 然而&#…

华为HarmonyOS地图服务 5 - 利用UI控件和手势进行地图交互

场景介绍 本章节将向您介绍如何使用地图的手势。 Map Kit提供了多种手势供用户与地图之间进行交互,如缩放、滚动、旋转和倾斜。这些手势默认开启,如果想要关闭某些手势,可以通过MapComponentController类提供的接口来控制手势的开关。 接口说明 以下是地图的控件和手势相…

设计模式之代理

一、代理设计模式概念 代理模式 (Proxy) 是一种结构型设计模式&#xff0c; 为其他对象提供一种代理以控制对这个对象的访问。 代理模式介绍了一种访问对象的间接等级。一个远程代理可以隐藏一个对象在不同地址空间的细节。一个虚拟代理可以根据需要最优化创建对象的开销。而安…

黄酮类化合物及其衍生物生物合成的进展:构建酵母细胞工厂的系统策略-

Advances in Flavonoid and Derivative Biosynthesis: Systematic Strategies for the Construction of Yeast Cell FactoriesCli 黄酮类化合物及其衍生物生物合成的进展&#xff1a;构建酵母细胞工厂的系统策略 摘要 黄酮类化合物是一类重要的天然多酚化合物&#xff0c;具有…

ESP32-WROOM-32 [创建AP站点-客户端-TCP透传]

简介 基于ESP32-WROOM-32 开篇(刚买)&#xff0c; 本篇讲的是基于固件 ESP32-WROOM-32-AT-V3.4.0.0&#xff08;内含用户指南, 有AT指令说明&#xff09;的TCP透传设置与使用 设备连接 TTL转USB线, 接ESP32 板 的 GND&#xff0c;RX2&#xff0c; TX2 指令介绍 注意,下面指…

【iOS】KVC的学习

【iOS】KVC的学习 文章目录 【iOS】KVC的学习前言KVC定义KVC设值KVC取值KVC使用keyPathKVC处理异常处理nil异常 KVC的一些应用修改动态的设置值实现高阶的消息传递 小结 前言 笔者简单学习了有关与KVC的相关内容&#xff0c;这里写一篇博客简单介绍一下相关内容。 KVC 定义 KV…

从零到一,监控网关上网设置教程

要让监控网关成功连接互联网&#xff0c;需要正确配置网络设置。监控网关通常位于本地局域网&#xff08;LAN&#xff09;或广域网&#xff08;WAN&#xff09;中&#xff0c;用于连接摄像头、传感器等监控设备&#xff0c;并通过网络上传数据到远程服务器或云平台。以下是监控…

计算机毕业设计 社区医疗服务系统的设计与实现 Java实战项目 附源码+文档+视频讲解

博主介绍&#xff1a;✌从事软件开发10年之余&#xff0c;专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精…

【GMNER】Grounded Multimodal Named Entity Recognition on Social Media

Grounded Multimodal Named Entity Recognition on Social Media 动机解决方法特征抽取多模态索引设计索引生成框架EncoderDecoder 实体定位、实体-类型-区域三元组重建 出处&#xff1a;ACL2023 论文链接&#xff1a;https://aclanthology.org/2023.acl-long.508.pdf code链接…

windows安装docker 本地打包代码

参考文章1&#xff1a;https://gitcode.csdn.net/65ea814b1a836825ed792f4a.html 参考文章2&#xff1a; Windows 安装docker&#xff08;详细图解&#xff09;-CSDN博客 一 下载 Docker Desktop 在官网上下载 Docker Desktop&#xff0c;可以从以下链接下载最新版本&#x…

重生之我们在ES顶端相遇第15 章 - ES 的心脏-倒排索引

文章目录 前言为什么叫倒排索引数据结构如何生成如何查询TF、IDF参考文档 前言 上一章&#xff0c;简单介绍了 ES 的节点类型。 本章&#xff0c;我们要介绍 ES 中非常重要的一个概念&#xff1a;倒排索引。 ES 的全文索引就是基于倒排索引实现的。 本章内容建议重点学习&…

基于python的api扫描器系统的设计与实现

&#x1f497;博主介绍&#x1f497;&#xff1a;✌在职Java研发工程师、专注于程序设计、源码分享、技术交流、专注于Java技术领域和毕业设计✌ 温馨提示&#xff1a;文末有 CSDN 平台官方提供的老师 Wechat / QQ 名片 :) Java精品实战案例《700套》 2025最新毕业设计选题推荐…

『功能项目』QFrameWork拾取道具UGUI【69】

本章项目成果展示 我们打开上一篇68QFrameWork扔到地上UGUI的项目&#xff0c; 本章要做的事情是实现当物品在地上时&#xff0c;点击物品将对应物品转移到道具栏中 制作一个提示UI界面 添加Button组件设置为点击即将父物体隐藏 拖拽到文件夹中在场景中删除 创建脚本&#xf…

Postman cURL命令导入导出

导入cURL命令 cURL是一种用于发出HTTP请求的流行命令行工具。在测试Web应用程序或API时&#xff0c;cURL使您能够直接从命令行进行交互&#xff0c;使用API开发人员社区中常见的完善语法。如果在不同的地方有多个cURL命令&#xff0c;可以将它们导入Postman。 ​ 将cURL命令导入…

医院伤员消费点餐限制———未来之窗行业应用跨平台架构

一、点餐上限 医院点餐上限具有以下几方面的意义&#xff1a; 1. 控制成本 - 有助于医院合理规划餐饮预算&#xff0c;避免食物的过度供应造成浪费&#xff0c;从而降低餐饮成本。 2. 保障饮食均衡 - 防止患者或陪护人员过度点餐某一类食物&#xff0c;有利于引导合…