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

《算法竞赛·快冲300题》每日一题:“连接草坪(II)”

算法竞赛·快冲300题》将于2024年出版,是《算法竞赛》的辅助练习册。
所有题目放在自建的OJ New Online Judge。
用C/C++、Java、Python三种语言给出代码,以中低档题为主,适合入门、进阶。

文章目录

  • 题目描述
  • 题解
  • C++代码
  • Java代码
  • Python代码

连接草坪(II)” ,链接: http://oj.ecustacm.cn/problem.php?id=1868

题目描述

【题目描述】 在N×M的地图上,X表示草,.表示土地。
  一个X与上下左右的X相连形成一片草坪。
  现在已知地图上有三片草坪,最少需要将多少个单位上的土地变成草,才能把两块草坪连接成一块草坪。
【输入格式】 输入第一行为正整数N和M,不超过50。
  接下来N行,每行M个字符。
**【输出格式】**输出一个数字表示答案。
【输入样例】

6 16
................
..XXXX....XXX...
...XXXX....XX...
.XXXX......XXX..
........XXXXX...
..XXX....XXX....

【输出样例】

4

题解

   题目给出了3块互不连通的草地,问最少把多少块土地变成草,可以把3块草地连成一块草地。考虑两种情况:
  (1)从土地的角度考虑。对任意一块土地坐标(x,y),分别计算它到3块草地的3个最少土地(最短路径),然后相加,得到土地(x,y)到3块草地的总最短路径count(x,y)。在所有count(x,y)中取最小值,是否就是答案?不一定,因为这些路径可能穿过其它的草地,导致重复计算。例如样例中左上角(0,0),它到3块草地的最短路径分别是3、11、7,但是它到3块草地的总最短路径实际上是3+4+4。
  (2)从草地的角度考虑。计算任意两块草地之间的最少土地(最短路径),记为Min[],其中Min[1]是土地(1-2)之间的最短路、Min[2]是土地(2-3)之间的最短路、Min[3]是土地(1-3)之间的最短路。那么是否min(Min[1]+Min[2], Min[2]+Min[3], Min[1]+Min[3]就是答案?不一定,它可能还不如情况(1)算出的最短路。
  例如下图,根据(2)算总最短路,3块草地之间的最短路是1、3、3,总短短路min(1+3, 3+3, 1+3)=4。但是根据情况(1)算最短路,箭头指向的土地k到3个草地的距离是1、3、1,总最短路是1+3+1-2=3,这里减2,是因为k被算了3次,其实只需要算1次。
在这里插入图片描述
  根据情况(1)和(2)算出的结果,取它们的最小值,就是答案。。
【重点】 DFS的应用 。

C++代码

   代码分4步:
  1、标记每个点属于哪个连通块,用DFS编码。
  2、枚举每块土地,计算它到3个草地的最小距离,即情况(1)。
  3、计算3个草地之间的最短距离,即情况(2)。
  4、在情况(1)和情况(2)中找最小值,就是答案。
  代码的复杂度约为O(NM)。

#include<bits/stdc++.h>
using namespace std;
int n, m;
char Map[55][55];           //存图
int id[55][55],id_cnt=0;    //id[x][y]=id_cnt: 点(x,y)属于第id_cnt个草地,id_cnt=1,2,3
vector<pair<int,int>>A[4];  //A[i]: 第i个草地中有哪些点
int dir[4][2] = {1,0,0,1,-1,0,0,-1};   //上下左右四个方向
void dfs(int x, int y, int c){  //从(x,y)开始搜它的邻居草地,并标记属于c个草地id[x][y] = c;               //点(x,y)属于第c个草地A[c].push_back(make_pair(x, y));    //这样写更好: A[c].emplace_back(x, y);for(int i = 0; i < 4; i++){         //上下左右4个邻居int nx = x + dir[i][0], ny = y + dir[i][1];   //邻居坐标if(nx < 0 || nx >= n || ny < 0 || ny >= m)  continue;if(Map[nx][ny] == '.')  continue;  //土地不在草地中if(id[nx][ny])          continue;  //这个点已经遍历过dfs(nx, ny, c);                    //继续}
}
int Count(int x, int y, int i){         //计算(x,y)到第i个草地的最短距离int ans = 100;for(auto a : A[i])ans = min(ans, abs(a.first - x) + abs(a.second - y));return ans;
}
int main(){cin >> n >> m;for(int i = 0; i < n; i++)  cin >> Map[i];//1、标记每个点属于哪个连通块for(int i = 0; i < n; i++)for(int j = 0; j < m; j++)if(Map[i][j] == 'X' && !id[i][j])dfs(i, j, ++id_cnt);int ans = 100;                //答案//2、枚举每块土地,计算它到3个草地的最小距离,即情况(1)for(int i = 0; i < n; i++)     //任意1个土地到其它草地的最小距离for(int j = 0; j < m; j++)if(Map[i][j] == '.')  //如果(i,j)是土地,计算它到3个草地的最小距离ans = min(ans, Count(i,j,1) + Count(i,j,2) + Count(i,j,3) - 2);  
//为什么要 -2 ?因为把自己算了3次,其实只需要算1次//3、计算3个草地之间的最短距离:1-2 2-3 1-3。int Min[4] = {0, 100, 100, 100};  //例如Min[1]是草地1-2的最短距离for(int i = 1; i <= 3; i++){      //第i个草地和第j个草地的最短距离int j = i+1 <= 3 ? i+1 : 1;for(auto &a : A[i])Min[i] = min(Min[i], Count(a.first, a.second, j));}//4、计算连通3个草地的最短距离,找最小值,即情况(2)。并与情况(1)的结果比较。for(int i = 1; i <= 3; i++)ans = min(ans, Min[i] + Min[i+1 <= 3 ? i+1 : 1] - 2);cout<<ans<<endl;return 0;
}

Java代码

import java.util.*;
public class Main {static int n, m, id_cnt = 0;static char[][] Map = new char[55][55];static int[][] id = new int[55][55];static List<List<Pair<Integer, Integer>>> A = new ArrayList<>(4);static int[][] dir = {{1, 0}, {0, 1}, {-1, 0}, {0, -1}};static void dfs(int x, int y, int c) {id[x][y] = c;A.get(c).add(new Pair<>(x, y));for (int i = 0; i < 4; i++) {int nx = x + dir[i][0], ny = y + dir[i][1];if (nx < 0 || nx >= n || ny < 0 || ny >= m) continue;if (Map[nx][ny] == '.') continue;if (id[nx][ny] != 0) continue;dfs(nx, ny, c);}}static int Count(int x, int y, int i) {int ans = 100;for (Pair<Integer, Integer> a : A.get(i)) ans = Math.min(ans, Math.abs(a.getKey() - x) + Math.abs(a.getValue() - y));return ans;}public static void main(String[] args) {Scanner cin = new Scanner(System.in);n = cin.nextInt();m = cin.nextInt();for (int i = 0; i < n; i++) Map[i] = cin.next().toCharArray();for (int i = 0; i < 4; i++) A.add(new ArrayList<>());for (int i = 0; i < n; i++) for (int j = 0; j < m; j++)if (Map[i][j] == 'X' && id[i][j] == 0) dfs(i, j, ++id_cnt);int ans = 100;for (int i = 0; i < n; i++) for (int j = 0; j < m; j++) if (Map[i][j] == '.') ans = Math.min(ans, Count(i, j, 1) + Count(i, j, 2) + Count(i, j, 3) - 2);int[] Min = {0, 100, 100, 100};for (int i = 1; i <= 3; i++) {int j = i + 1 <= 3 ? i + 1 : 1;for (Pair<Integer, Integer> a : A.get(i)) Min[i] = Math.min(Min[i], Count(a.getKey(), a.getValue(), j));}for (int i = 1; i <= 3; i++) ans = Math.min(ans, Min[i] + Min[i + 1 <= 3 ? i + 1 : 1] - 2);System.out.println(ans);cin.close();}static class Pair<K, V> {public K key;public V value;public Pair(K key, V value) {this.key = key;this.value = value;}public K getKey() {  return key;    }public V getValue() {return value;  }}
}

Python代码

  

n, m = map(int, input().split())
Map = [input() for _ in range(n)]
id, id_cnt = [[0] * m for _ in range(n)], 0
A = [[] for _ in range(4)]
dir = [(1, 0), (0, 1), (-1, 0), (0, -1)]
def dfs(x, y, c):id[x][y] = cA[c].append((x, y))for i in range(4):nx, ny = x + dir[i][0], y + dir[i][1]if nx < 0 or nx >= n or ny < 0 or ny >= m: continueif Map[nx][ny] == '.': continueif id[nx][ny] != 0:    continuedfs(nx, ny, c)
def Count(x, y, i):ans = 100for a in A[i]:ans = min(ans, abs(a[0] - x) + abs(a[1] - y))return ans
for i in range(n):for j in range(m):if Map[i][j] == 'X' and id[i][j] == 0:dfs(i, j, id_cnt+1)id_cnt += 1
ans = 100
for i in range(n):for j in range(m):if Map[i][j] == '.':ans = min(ans, Count(i, j, 1) + Count(i, j, 2) + Count(i, j, 3) - 2)Min = [0, 100, 100, 100]
for i in range(1, 4):j = i+1 if i+1 <= 3 else 1for a in A[i]:Min[i] = min(Min[i], Count(a[0], a[1], j))
for i in range(1, 4):ans = min(ans, Min[i] + Min[i+1 if i+1 <= 3 else 1] - 2)
print(ans)

相关文章:

《算法竞赛·快冲300题》每日一题:“连接草坪(II)”

《算法竞赛快冲300题》将于2024年出版&#xff0c;是《算法竞赛》的辅助练习册。 所有题目放在自建的OJ New Online Judge。 用C/C、Java、Python三种语言给出代码&#xff0c;以中低档题为主&#xff0c;适合入门、进阶。 文章目录 题目描述题解C代码Java代码Python代码 “ 连…...

LNMP及论坛搭建(第一个访问,单节点)

LNMP&#xff1a;目前成熟的一个企业网站的应用模式之一&#xff0c;指的是一套协同工作的系统和相关软件 能够提供静态页面服务&#xff0c;也可以提供动态web服务&#xff0c;LNMP是缩写 L&#xff1a;指的是Linux操作系统。 N&#xff1a;指的是nginx&#xff0c;nginx提…...

EXCEL,多条件查询数字/文本内容的4种方法

目录 1 问题&#xff1a;如何根据多条件查询到想要的内容 2 方法总结 2.1 方法1&#xff1a; sumif() 和sumifs() 适合查找符合条件的多个数值之和 2.2 方法2&#xff1a;使用lookup(1,0/((区域1条件1)*(区域2条件2)*....),结果查询区域) 2.3 方法3&#xff1a;使用 ind…...

全志D1-H (MQ-Pro)驱动 OV5640 摄像头

内核配置 运行 m kernel_menuconfig 勾选下列驱动 Device Drivers ---><*> Multimedia support --->[*] V4L platform devices ---><*> Video Multiplexer[*] SUNXI platform devices ---><*> sunxi video input (camera csi/mipi…...

2023下半年软考初级网络管理员报名入口-报名流程-备考方法

软考初级网络管理员2023下半年考试时间&#xff1a; 2023年下半年软考初级网络管理员的考试时间为11月4日、5日。考试时间在全国各地一致&#xff0c;建议考生提前备考。共分两科&#xff0c;第一科基础知识考试具体时间为9:00到11:30&#xff1b;第二科应用技术考试具体时间为…...

QEMU源码全解析29 —— QOM介绍(18)

接前一篇文章&#xff1a;QEMU源码全解析28 —— QOM介绍&#xff08;17&#xff09; 本文内容参考&#xff1a; 《趣谈Linux操作系统》 —— 刘超&#xff0c;极客时间 《QEMU/KVM》源码解析与应用 —— 李强&#xff0c;机械工业出版社 特此致谢&#xff01; 前文讲解了类…...

从入门到精通——【初识网络】

文章目录 前言1.网络发展背景2.计算机网络分类3.通信协议4.协议分层5. TCP/IP协议6.网络协议支持7. 封装&分用8. 客户端&服务端 前言 计算机网络是指将地理位置不同的具有独立功能的多台计算机及其外部设备&#xff0c;通过通信线路连接起来&#xff0c;在网络操作系统…...

MySQL alter命令修改表详解

目录 ALTER TABLE 语法 ALTER TABLE 实例 添加一列 添加多列 重命名列 修改列定义 修改列名和定义 添加主键 删除列 重命名表 修改表的存储引擎 结论 在使用表的过程中&#xff0c;如果您需要对表进行修改&#xff0c;您可以使用 ALTER TABLE 语句。通过 ALTER TAB…...

Vulnhub: ColddWorld: Immersion靶机

kali&#xff1a;192.168.111.111 靶机&#xff1a;192.168.111.183 信息收集 端口扫描 nmap -A -sC -v -sV -T5 -p- --scripthttp-enum 192.168.111.183 查看login的源码发现提示&#xff1a;page和文件/var/carls.txt 漏洞利用 wfuzz探测account.php页面发现文件包含&am…...

Redis 总结【6.0版本的】

还差什么&#xff1f;【按照这个为基础&#xff0c;对照他的Redis路线图&#xff0c;冲冲冲】 Redis的常见操作和命令&#xff1a;Redis基本操作命令&#xff08;图文详解&#xff09;_redis 命令_进击小高的博客-CSDN博客 Redis的持久化&#xff0c;一致性&#xff1a;AOF&…...

状态模式(C++)

定义 允许一个对象在其内部状态改变时改变它的行为。从而使对象看起来似乎修改了其行为。 应用场景 在软件构建过程中&#xff0c;某些对象的状态如果改变&#xff0c;其行为也会随之&#xff0c;而发生变化&#xff0c;比如文档处于只读状态&#xff0c;其支持的行为和读写…...

承泰科技Q3再获30多个智驾项目,新增订单0.86亿!累计近11亿!

中国毫米波雷达市场正处于高速发展期&#xff0c;以承泰科技为代表的本土供应商在前装量产赛道上展示出加速度。 高工智能汽车研究院预测&#xff0c;随着L2及L2持续处于市场增长的高速期&#xff0c;对应毫米波雷达上车量将在2023年实现30-50%的同比增速。 根据高工智能汽车…...

以太网Ethernet通信协议

一、以太网简介 计算机网络可分为局域网(LAN)、 城域网(MAN)、广域网(WAN)、互联网(Initernet)。局域网按传输介质所使用的访问控制方法可分为&#xff1a;以太网(Ethernet)、光纤分布式数据接口(FDDI)、异步传输模式(ATM)、令牌环网(Token Ring)、交换网(Switching) 等&#x…...

内网横向移动—资源约束委派

内网横向移动—资源约束委派 1. 资源约束委派1.1. 基于资源的约束委派的优势1.2. 约束性委派和基于资源的约束性委派配置的差别1.3. 利用条件1.3.1. 什么用户能够修改msDS-AllowedToActOnBehalfOfOtherIdentity属性1.3.2. 将机器加入域的域用户 2. 案例操作2.1. 获取目标信息2.…...

Spring Boot Logback日志格式改为JSON

在阿里云、或者日志分析时使用JSON格式输出日志更加方便。 依赖 增加Logbak JSON解析依赖。 另外需要注意的是JSON格式输出依赖Jackson&#xff0c;根据工程情况按需添加Jackson依赖。 <!--日志--><dependency><groupId>ch.qos.logback.contrib</grou…...

Linux 块设备操作函数

和字符设备的fil_operations一样&#xff0c;块设备也有操作集&#xff0c;为结构体block_device_operations&#xff0c;此结构体定义在include/linux/blkdev.h中&#xff0c;结构体内容如下&#xff1a; struct block_device_operations {int (*open) (struct block_device …...

linux c++网络编程基础:服务端与客户端的实现

在Linux环境下,我们可以使用socket编程来实现网络通信。下面是一个简单的C++版本的客户端和服务端的示例代码。 服务端代码: #include <sys/socket.h> #include <netinet/in.h> #include <unistd.h> #include <string.h> #...

坐标转换-使用geotools读取和转换地理空间表的坐标系(sqlserver、postgresql)

前言&#xff1a; 业务上通过GIS软件将空间数据导入到数据库时&#xff0c;因为不同的数据来源和软件设置&#xff0c;可能导入到数据库的空间表坐标系是各种各样的。 如果要把数据库空间表发布到geoserver并且统一坐标系&#xff0c;只是在geoserver单纯的设置坐标系只是改了…...

JavaScript的主要应用场景有哪些?请描述一下JavaScript的基本数据类型和引用数据类型分别是哪些?

1、JavaScript的主要应用场景有哪些&#xff1f; JavaScript是一种广泛使用的编程语言&#xff0c;它主要用于Web开发、移动应用开发、游戏开发、物联网设备开发等场景。以下是JavaScript的主要应用场景&#xff1a; Web开发&#xff1a;JavaScript是Web开发中最常用的编程语…...

webpack性能优化

文章目录 1. 性能优化-分包2. 动态导入3. 自定义分包4. Prefetch和Preload5. CDN加载配置6. CSS的提取7. terser压缩7.1 Terser在webpack中配置7.2 css压缩 8. Tree Shaking 消除未使用的代码8.1 usedExports 配置8.2 sideEffects配置8.3 CSS实现Tree Shaking 9. Scope Hoistin…...

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇&#xff0c;在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下&#xff1a; 【Note】&#xff1a;如果你已经完成安装等操作&#xff0c;可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作&#xff0c;重…...

C++:std::is_convertible

C++标志库中提供is_convertible,可以测试一种类型是否可以转换为另一只类型: template <class From, class To> struct is_convertible; 使用举例: #include <iostream> #include <string>using namespace std;struct A { }; struct B : A { };int main…...

黑马Mybatis

Mybatis 表现层&#xff1a;页面展示 业务层&#xff1a;逻辑处理 持久层&#xff1a;持久数据化保存 在这里插入图片描述 Mybatis快速入门 ![在这里插入图片描述](https://i-blog.csdnimg.cn/direct/6501c2109c4442118ceb6014725e48e4.png //logback.xml <?xml ver…...

【JVM】- 内存结构

引言 JVM&#xff1a;Java Virtual Machine 定义&#xff1a;Java虚拟机&#xff0c;Java二进制字节码的运行环境好处&#xff1a; 一次编写&#xff0c;到处运行自动内存管理&#xff0c;垃圾回收的功能数组下标越界检查&#xff08;会抛异常&#xff0c;不会覆盖到其他代码…...

MMaDA: Multimodal Large Diffusion Language Models

CODE &#xff1a; https://github.com/Gen-Verse/MMaDA Abstract 我们介绍了一种新型的多模态扩散基础模型MMaDA&#xff0c;它被设计用于在文本推理、多模态理解和文本到图像生成等不同领域实现卓越的性能。该方法的特点是三个关键创新:(i) MMaDA采用统一的扩散架构&#xf…...

Nuxt.js 中的路由配置详解

Nuxt.js 通过其内置的路由系统简化了应用的路由配置&#xff0c;使得开发者可以轻松地管理页面导航和 URL 结构。路由配置主要涉及页面组件的组织、动态路由的设置以及路由元信息的配置。 自动路由生成 Nuxt.js 会根据 pages 目录下的文件结构自动生成路由配置。每个文件都会对…...

根据万维钢·精英日课6的内容,使用AI(2025)可以参考以下方法:

根据万维钢精英日课6的内容&#xff0c;使用AI&#xff08;2025&#xff09;可以参考以下方法&#xff1a; 四个洞见 模型已经比人聪明&#xff1a;以ChatGPT o3为代表的AI非常强大&#xff0c;能运用高级理论解释道理、引用最新学术论文&#xff0c;生成对顶尖科学家都有用的…...

JVM 内存结构 详解

内存结构 运行时数据区&#xff1a; Java虚拟机在运行Java程序过程中管理的内存区域。 程序计数器&#xff1a; ​ 线程私有&#xff0c;程序控制流的指示器&#xff0c;分支、循环、跳转、异常处理、线程恢复等基础功能都依赖这个计数器完成。 ​ 每个线程都有一个程序计数…...

使用Spring AI和MCP协议构建图片搜索服务

目录 使用Spring AI和MCP协议构建图片搜索服务 引言 技术栈概览 项目架构设计 架构图 服务端开发 1. 创建Spring Boot项目 2. 实现图片搜索工具 3. 配置传输模式 Stdio模式&#xff08;本地调用&#xff09; SSE模式&#xff08;远程调用&#xff09; 4. 注册工具提…...

十九、【用户管理与权限 - 篇一】后端基础:用户列表与角色模型的初步构建

【用户管理与权限 - 篇一】后端基础:用户列表与角色模型的初步构建 前言准备工作第一部分:回顾 Django 内置的 `User` 模型第二部分:设计并创建 `Role` 和 `UserProfile` 模型第三部分:创建 Serializers第四部分:创建 ViewSets第五部分:注册 API 路由第六部分:后端初步测…...