大厂秋招真题【BFS+DP】华为20230921秋招T3-PCB印刷电路板布线(留学生专场)
华为20230921秋招T3-PCB印刷电路板布线(留学生专场)
题目描述与示例
题目描述
在PCB印刷电路板设计中,器件之间的连线,要避免线路的阻抗值增大,而且器件之间还有别的器任和别的干扰源,在布线时我们希望受到的干扰尽量小。
现将电路板简化成一个M × N
的矩阵,每个位置(单元格)的值表示其源干扰度。
如果单元格的值为0
,表示此位置没有干扰源,如果单元格的值为非0
,则表示此位置是干扰源,其值为源干扰度。连线经过干扰源或干扰源附近会增加连线的总干扰度。
位置A[x,y]
的干扰源的源干扰广为d (d>0)
,则连线的干扰度计算如下:
1、若连线经过位置A[x,y]
,则其总干扰度会增加加
2、若连线经过离位置A[x,y]
距离小于d
的位置时,设其距离为k
,则总干扰度会增加(d-k)
3、若连线经过离位置A[x,y]
距离大于或等于d
的位置时,总干扰都不会增加;
注:位置[x1,y1]
和位置[x2,y2]
之间距离的定义为:|x1-x2|+|y1-y2|
。
如下3x3
矩阵,位置[1,1]
的源干扰度是2
,连线的位置序列为:[0,0]->[0,1]->[0,2]->[1,2]->[2,2]
。
其中[0,1]
和[1,0]
到干扰源的距离为1
,会叠加1
的干扰度;其他位置到[1,1]
的距离均大于等于2
,所以不会叠加干扰度。因此这条连线的总干扰度为2
。
现在我们需要将左上角的器件到右下角的器件进行连线,两个器件的位置分别是左上角的[0,0]
和右下角的[M-1,N-1]
。由于我们希望连线尽量地短,从位置[0,0]
到[M-1,N-1]
的连线途中,我们规定连线只能向下或向右。
请根据输入(M × N
的矩阵),计算出连线的最小干扰度。
输入描述
第一行是两个整数M
和N
(M
和N
最大值为1000
),表示行数和列数;
接着是M
行的数据,每一包含N
个整数,代表每个位置的源干扰度,每个源干扰度小于50
。
输出描述
左上角[0,0]
到右下角[M-1,N-1]
连线的最小总干扰度。
示例一
输入
3 3
0 0 0
0 2 0
0 0 0
输出
2
说明
其中一条可以使干扰度最小的路径为:[0,0]->[0,1]->[0,2]->[1,2]->[2,2]
,其干扰度为2
。
示例二
输入
5 5
0 0 0 0 0
0 0 2 0 0
0 2 0 2 0
0 0 0 0 0
0 0 0 0 0
输出
1
说明
先从[0,0]
往下走到最下面[4,0]
,再往石走到右下角[4,4]
,途径[2,0]
时叠加一个干扰度。
示例三
输入
5 5
0 0 0 0 0
0 0 2 0 0
0 2 0 2 0
0 0 2 0 0
0 0 0 0 0
输出
2
时空限制
时间限制: C/C++ 2000MS
,其他语言4000MS
内存限制: C/C++ 256MB
,其他语言512MB
解题思路
本题属于综合性较强的题目,结合了BFS和DP两个知识点。
首先我们需要根据原矩阵,构建出每一个位置干扰值叠加的结果,得到一个新的矩阵grid_new
。这里显然就是一个基于BFS计算层数的问题。
在得到新的矩阵grid_new
之后,问题就转变为,对grid_new
构建一条从左上到右下的路径,每次只能够向右或向下移动,路径经过的点的总和需要最小。这是一个经典的路径DP问题,和LeetCode64、最小路径和完全一致。
代码
Python
# 题目:【DP】华为2023秋招-PCB印刷电路板布线
# 作者:闭着眼睛学数理化
# 算法:DP/BFS
# 代码有看不懂的地方请直接在群上提问from collections import dequeDIRECTIONS = [(0, 1), (1, 0), (0, -1), (-1, 0)]# 对于grid中每一个干扰源,以干扰源作为起点进行BFS,更新grid_new
def BFS_update_grid_new(val, grid_new, i, j, m, n):check_list = [[0] * n for _ in range(m)]check_list[i][j] = 1grid_new[i][j] += valq = deque()q.append((i, j))# 注意这里退出循环的条件和val相关while val > 1:val -= 1qSize = len(q)for _ in range(qSize):cur_x, cur_y = q.popleft()for dx, dy in DIRECTIONS:nxt_x, nxt_y = cur_x+dx, cur_y+dyif 0 <= nxt_x < m and 0 <= nxt_y < n and check_list[nxt_x][nxt_y] == 0:q.append((nxt_x, nxt_y))grid_new[nxt_x][nxt_y] += valcheck_list[nxt_x][nxt_y] = 1# 用于解决最小路径和问题的函数
def find_min_sum_path(grid_new, m, n):# 构建大小为m*n的dp数组,dp[i][j]表示# 到达grid_new中的点(i,j),所需的最小路径和dp = [[0] * (n) for _ in range(m)]# 初始化(0,0)位置dp[0][0] = grid_new[0][0]# 初始化dp数组第0行,只能从左边向右转移得到for j in range(1, n):dp[0][j] += dp[0][j-1] + grid_new[0][j]# 初始化dp数组第0列,只能从上边向下转移得到for i in range(1, m):dp[i][0] += dp[i-1][0] + grid_new[i][0]# 遍历剩余所有点# 点(i,j)的状态,只能从点(i-1,j)向下或者从点(i,j-1)向右转移得到# 故动态转移方程为dp[i][j] = min(dp[i-1][j], dp[i][j-1]) + grid_new[i][j]for i in range(1, m):for j in range(1, n):dp[i][j] = min(dp[i-1][j], dp[i][j-1]) + grid_new[i][j]return dp[-1][-1]# 输入行数m,列数n
m, n = map(int, input().split())
# 构建原干扰值矩阵
grid = list()
for _ in range(m):grid.append(list(map(int, input().split())))# 初始化干扰值叠加后的新矩阵gird_new
grid_new = [[0] * n for _ in range(m)]for i in range(m):for j in range(n):# 对于每一个干扰源,使用BFS更新grid_newif grid[i][j] != 0:val = grid[i][j]BFS_update_grid_new(val, grid_new, i, j, m, n)# 调用函数find_min_sum_path,输出答案
print(find_min_sum_path(grid_new, m, n))
Java
import java.util.ArrayDeque;
import java.util.Deque;
import java.util.Scanner;public class Main {private static final int[][] DIRECTIONS = {{0, 1}, {1, 0}, {0, -1}, {-1, 0}};// Function to perform BFS and update gridNewprivate static void BFSUpdateGridNew(int val, int[][] gridNew, int i, int j, int m, int n) {int[][] checkList = new int[m][n];checkList[i][j] = 1;gridNew[i][j] += val;Deque<int[]> queue = new ArrayDeque<>();queue.add(new int[]{i, j});while (val > 1) {val--;int qSize = queue.size();for (int k = 0; k < qSize; k++) {int[] current = queue.poll();int curX = current[0];int curY = current[1];for (int[] dir : DIRECTIONS) {int nextX = curX + dir[0];int nextY = curY + dir[1];if (nextX >= 0 && nextX < m && nextY >= 0 && nextY < n && checkList[nextX][nextY] == 0) {queue.add(new int[]{nextX, nextY});gridNew[nextX][nextY] += val;checkList[nextX][nextY] = 1;}}}}}// Function to find the minimum sum pathprivate static int findMinSumPath(int[][] gridNew, int m, int n) {int[][] dp = new int[m][n];dp[0][0] = gridNew[0][0];for (int i = 1; i < m; i++) {dp[i][0] = dp[i - 1][0] + gridNew[i][0];}for (int j = 1; j < n; j++) {dp[0][j] = dp[0][j - 1] + gridNew[0][j];}for (int i = 1; i < m; i++) {for (int j = 1; j < n; j++) {dp[i][j] = Math.min(dp[i - 1][j], dp[i][j - 1]) + gridNew[i][j];}}return dp[m - 1][n - 1];}public static void main(String[] args) {Scanner sc = new Scanner(System.in);int m = sc.nextInt();int n = sc.nextInt();int[][] grid = new int[m][n];for (int i = 0; i < m; i++) {for (int j = 0; j < n; j++) {grid[i][j] = sc.nextInt();}}int[][] gridNew = new int[m][n];for (int i = 0; i < m; i++) {for (int j = 0; j < n; j++) {if (grid[i][j] != 0) {int val = grid[i][j];BFSUpdateGridNew(val, gridNew, i, j, m, n);}}}System.out.println(findMinSumPath(gridNew, m, n));}
}
C++
#include <iostream>
#include <vector>
#include <deque>
using namespace std;const vector<pair<int, int>> DIRECTIONS = {{0, 1}, {1, 0}, {0, -1}, {-1, 0}};// Function to perform BFS and update grid_new
void BFSUpdateGridNew(int val, vector<vector<int>>& gridNew, int i, int j, int m, int n) {vector<vector<int>> checkList(m, vector<int>(n, 0));checkList[i][j] = 1;gridNew[i][j] += val;deque<pair<int, int>> q;q.push_back({i, j});while (val > 1) {val--;int qSize = q.size();for (int k = 0; k < qSize; k++) {int curX = q.front().first;int curY = q.front().second;q.pop_front();for (const auto& dir : DIRECTIONS) {int nextX = curX + dir.first;int nextY = curY + dir.second;if (nextX >= 0 && nextX < m && nextY >= 0 && nextY < n && checkList[nextX][nextY] == 0) {q.push_back({nextX, nextY});gridNew[nextX][nextY] += val;checkList[nextX][nextY] = 1;}}}}
}// Function to find the minimum sum path
int FindMinSumPath(const vector<vector<int>>& gridNew, int m, int n) {vector<vector<int>> dp(m, vector<int>(n, 0));dp[0][0] = gridNew[0][0];for (int i = 1; i < m; i++) {dp[i][0] = dp[i - 1][0] + gridNew[i][0];}for (int j = 1; j < n; j++) {dp[0][j] = dp[0][j - 1] + gridNew[0][j];}for (int i = 1; i < m; i++) {for (int j = 1; j < n; j++) {dp[i][j] = min(dp[i - 1][j], dp[i][j - 1]) + gridNew[i][j];}}return dp[m - 1][n - 1];
}int main() {int m, n;cin >> m >> n;vector<vector<int>> grid(m, vector<int>(n));for (int i = 0; i < m; i++) {for (int j = 0; j < n; j++) {cin >> grid[i][j];}}vector<vector<int>> gridNew(m, vector<int>(n, 0));for (int i = 0; i < m; i++) {for (int j = 0; j < n; j++) {if (grid[i][j] != 0) {int val = grid[i][j];BFSUpdateGridNew(val, gridNew, i, j, m, n);}}}cout << FindMinSumPath(gridNew, m, n) << endl;return 0;
}
时空复杂度
时间复杂度:O(MNk)
。其中k
为干扰源的数目,一共需要进行k
次BFS,每次BFS的时间复杂度为O(MN)
。另外,DP过程的时间复杂度为O(MN)
。
空间复杂度:O(MN)
。grid_new
、check_list
、dp
等二维矩阵所占空间均为O(MN)
。
华为OD算法/大厂面试高频题算法练习冲刺训练
-
华为OD算法/大厂面试高频题算法冲刺训练目前开始常态化报名!目前已服务100+同学成功上岸!
-
课程讲师为全网50w+粉丝编程博主@吴师兄学算法 以及小红书头部编程博主@闭着眼睛学数理化
-
每期人数维持在20人内,保证能够最大限度地满足到每一个同学的需求,达到和1v1同样的学习效果!
-
60+天陪伴式学习,40+直播课时,300+动画图解视频,300+LeetCode经典题,200+华为OD真题/大厂真题,还有简历修改、模拟面试、专属HR对接将为你解锁
-
可上全网独家的欧弟OJ系统练习华子OD、大厂真题
-
可查看链接 大厂真题汇总 & OD真题汇总(持续更新)
-
绿色聊天软件戳
od1336
了解更多
相关文章:

大厂秋招真题【BFS+DP】华为20230921秋招T3-PCB印刷电路板布线(留学生专场)
华为20230921秋招T3-PCB印刷电路板布线(留学生专场) 题目描述与示例 题目描述 在PCB印刷电路板设计中,器件之间的连线,要避免线路的阻抗值增大,而且器件之间还有别的器任和别的干扰源,在布线时我们希望受…...

OpenCV Python – 使用SIFT算法实现两张图片的特征匹配
OpenCV Python – 使用SIFT算法实现两张图片的特征匹配 1.要实现在大图中找到任意旋转、缩放等情况下的小图位置,可以使用特征匹配算法,如 SIFT (尺度不变特征变换) 或 SURF (加速稳健特征)。这些算法可以在不同尺度和旋转情况下寻找匹配的特征点 impo…...

doc转html后添加style和导航
public static void main(String[] args) throws Exception {docxToHtml(); } public static void docxToHtml() throws Exception {//D:\zpdtolly\工作总结文档\zpd使用文档\v4\用户使用手册\客户端使用手册String sourceFileName "C:\\Users\\luoguoqing\\Desktop\\202…...

Python中跨越多个文件使用全局变量
嗨喽,大家好呀~这里是爱看美女的茜茜呐 这个琐碎的指南是关于在 Python 中跨多个文件使用全局变量。 但是在进入主题之前,让我们简单地看看全局变量和它们在多个文件中的用途。 👇 👇 👇 更多精彩机密、教程ÿ…...

设计模式 - 解释器模式
目录 一. 前言 二. 实现 三. 优缺点 一. 前言 解释器模式(Interpreter Pattern)指给定一门语言,定义它的文法的一种表示,并定义一个解释器,该解释器使用该表示来解释语言中的句子,属于行为型设计模式。是…...

javascript禁止鼠标右键和复制功能
要禁止鼠标右键和复制功能,可以编写如下的封装函数: function preventDefaultCopy(event) {// 禁止右键 菜单和复制event.preventDefault();event.stopPropagation();return false; }// 在需要禁止复制的元素上添加该事件监听器 element.addEventListen…...

WebDAV之π-Disk派盘 + 咕咚云图
咕咚云图是一款强大的图床传图软件,它能够让您高效地对手机中的各种图片进行github传输,多个平台快速编码上传,支持远程删除不需要的图片,传输过程安全稳定,让您可以很好的进行玩机或者其他操作。 可帮你上传手机图片到图床上,并生成 markdown 链接,支持七牛云、阿里云…...

C语言-数组
C 语言支持数组数据结构,数组是一个由若干相同类型变量组成的有序集合。 这里的有序是指数组元素在内存中的存放方式是有序的,即所有的数组都是由连续的内存位置组成。最低的地址对应第一个元素,最高的地址对应最后一个元素。 在 C 语言中&am…...

Linux UWB Stack实现——MCPS调度接口(API)
在上一篇文章中,介绍了MCPS调度接口涉及的相关数据结构实现MCPS调度接口(数据结构),本文继续介绍调度相关的方法的实现。 1. 域处理 1.1 域注册与注销 注册/注销一个mcps802154_region,分别在模块加载(mo…...

el-tree中插入图标并且带提示信息
<template><div class"left"><!-- default-expanded-keys 默认展开 --><!-- expand-on-click-node 只有点击箭头才会展开树 --><el-tree :data"list" :props"defaultProps" node-click"handleNodeClick" :…...

竞赛选题 深度学习 YOLO 实现车牌识别算法
文章目录 0 前言1 课题介绍2 算法简介2.1网络架构 3 数据准备4 模型训练5 实现效果5.1 图片识别效果5.2视频识别效果 6 部分关键代码7 最后 0 前言 🔥 优质竞赛项目系列,今天要分享的是 🚩 基于yolov5的深度学习车牌识别系统实现 该项目较…...

Direct3D网格(一)
创建网格 我们可以用D3DXCreateMeshFVF函数创建一个"空"网格对象 ,空网格对象是指我们指定了网格的面片总数和顶点总数,然后由该函数为顶点缓存、索引缓存和属性缓存分配大小合适的内存,之后即可手工填入网格数据。 HRESULT WINA…...

C语言打印菱形
一、运行结果图 二、源代码 # define _CRT_SECURE_NO_WARNINGS # include <stdio.h>int main() {//初始化变量值;int line 0;int i 0;int j 0;//获取变量值;scanf("%d", &line);//循环打印上半部分;for (i 0; i <…...

ElasticSearch搜索引擎:数据的写入流程
一、ElasticSearch 写数据的总体流程: (1)ES 客户端选择一个节点 node 发送请求过去,这个节点就是协调节点 coordinating node (2)协调节点对 document 进行路由,通过 hash 算法计算出数据应该…...

python3 调用 另外一个python脚本
3种python调用其他脚本脚本的方法_python 调用python脚本_linjingyg的博客-CSDN博客 Python之系统交互(调用系统命令)subprocess_subprocess.getoutput(cmd) 参数格式不正确-CSDN博客 subprocess.call()只能返回状态码。subprocess.getoutput(cmd)只能输出命令结果。 str(py…...

【13】c++设计模式——>简单工厂模式
工厂模式的定义 c中的工厂模式是一种创建型设计模式,它提供一种创建对象的接口,但具体创建的对象类型可以在运行时决定,这样可以将对象的创建与使用代码分离,提高代码的灵活性和可维护性。 在c中实现工厂模式,通常会定…...

系统架构设计:2 论软件设计方法及其应用
目录 一 软件设计方法 1结构化设计 2信息工程 3面向对象设计 4原型设计...

基于Winform的UDP通信
1、文件结构 2、UdpReceiver.cs using System; using System.Collections.Generic; using System.Linq; using System.Net; using System.Net.Sockets; using System.Text; using System.Threading.Tasks;namespace UDPTest.Udp {public class UdpStateEventArgs : EventArgs…...

掌握 BERT:自然语言处理 (NLP) 从初级到高级的综合指南(1)
简介 BERT(来自 Transformers 的双向编码器表示)是 Google 开发的革命性自然语言处理 (NLP) 模型。它改变了语言理解任务的格局,使机器能够理解语言的上下文和细微差别。在本文[1]中,我们将带您踏上从 BERT 基础知识到高级概念的旅…...

Linux Ftrace介绍
文章目录 一、简介二、内核函数调用跟踪参考链接: 一、简介 Ftrace 是 Linux 官方提供的跟踪工具,在 Linux 2.6.27 版本中引入。Ftrace 可在不引入任何前端工具的情况下使用,让其可以适合在任何系统环境中使用。 Ftrace 可用来快速排查以下相…...

Go语言进阶------>init()函数
Init()包初始化 执行优先级 Init()函数的执行优先级比main()函数的执行优先级要高,也就是说程序会优先执行Init()函数之后再执行main()函数. 代码如下 package mainimport "fmt"func init() {fmt.Println("执行了Init()函数") }func main() {fmt.Println…...

云计算:常用微服务框架
目录 一、理论 1.Java微服务框架 2.Go微服务框架 3.Python微服务框架 4.Node.js微服务框架 5..Net微服务框架 一、理论 1.Java微服务框架 Spring Cloud:最早最成熟,Java开源微服务框架方案 SpringBoot:全新框架,设计目的是…...

jmeter添加断言(详细图解)
先创建一个线程组,再创建一个http请求。 为了方便观察,我们添加两个监听器,察看结果树和断言结果。 添加断言:响应断言,响应断言也是比较常用的一个断言 设置响应断言:正常情况下响应代码是200。选择响应代…...

few shot object detection via feature reweight笔记
摘要部分 few shot很多用的都是faster R-CNN为基础,本文用的是one-stage 结构。 用了一个meta feature learner和reweighting模块。 和其他的few shot一样,先学习base数据集,再推广到novel数据集。 feature learner会从base数据集中提取meta…...

工会排队模式:电商新营销模式吸引消费者,提升销售!
随着电商行业的繁荣发展,私域流量已经成为了电商平台争夺消费者和促进销售的重要手段。工会排队模式正是在这种背景下应运而生的一种创新性的电商营销模式。这种模式通过奖金池的资金来为消费者和商家提供返现和排队奖励,构建了一个实现消费者和商家共赢…...

定档通知2024中国(北京)国际红外技术及设备展览会
时间:2024年7月14-16日 地点:北京国家会议中心 ◆展会背景background: 各有关红外企业厂商:2024年7月14~16日,2024中国国际红外技术及设…...

自助建站系统,一建建站系统api版,自动建站
安装推荐php7.2或7.2以下都行 可使用虚拟主机或者服务器进行搭建。 分站进入网站后台 域名/admin 初始账号123456qq.com密码123456 找到后台的网站设置 将主站域名及你在主站的通信secretId和通信secretKey填进去。 即可正常使用 通信secretId和通信secretKey在主站的【账号…...

算法框架-LLM-1-Prompt设计(一)
原文:算法框架-LLM-1-Prompt设计(一) - 知乎 目录 收起 1 prompt-engineering-for-developers 1.1 Prompt Engineering 1.1.1 提示原则 1. openai的环境 2. 两个基本原则 3. 示例 eg.1 eg.2 结构化输出 eg.3 模型检验 eg.4 提供示…...

一个rar压缩包如何分成三个?
一个rar压缩包体积太大了,想要将压缩包分为三个,该如何做到?其实很简单,方法就在我们经常使用的WinRAR当中。 我们先将压缩包内的文件解压出来,然后查看一下,然后打开WinRAR软件,找到文件&…...

批量获取拼多多商品详情数据,拼多多商品详情API接口
批量获取拼多多商品详情数据可以采用以下方式: 使用拼多多开放平台API接口。 拼多多开放平台提供了API接口,可以通过API接口获取拼多多平台上的商品信息,使用API接口需要进行权限申请和认证,操作较为复杂。 使用第三方工具。 市面…...