(蓝桥真题)剪格子(搜索+剪枝)

news2024/11/16 21:00:49

 样例1输入:

3 3
10 1 52
20 30 1
1 2 3

样例1输出:

3

样例2输入:

4 3
1 1 1 1
1 30 80 2
1 1 1 100

样例2输出:

10

分析:这道题目我们直接从(1,1)点开始进行dfs搜索即可,但是需要注意一点的是我们搜索的时候并不是沿着一条路径进行搜索,而是从当前已经走过的所有点中选出一个点然后沿着不同方向去搜索,这样我们就可以搜出所有的连通块,每次搜出一个连通块时还需要检测剩余的部分是否是一个连通块,那么检测剩余部分是否是一个连通块我们可以用并查集来实现。这样大体的步骤就实现了,接下来就是剪枝了,为了防止同样的状态被多次搜索,我们可以用哈希优化一下,随意设置一个哈希函数,然后求出每个连通块对应的哈希值然后进行去重即可,还有可以优化的一点就是我们每次尽可能选取值大的点进行搜索,这样得到目标值的连通块内的点就会尽可能小

需要说明的一点就是:由于蓝桥原题是没有明确说明两部分都必须连通的,所以也就没必要加上判断连通的那部分,而且他数据中都是一笔画形成的连通块,没有考虑周全,所以本代码在这两方面进行了优化,但会在洛谷上提交时会有一个点超时,那是因为本代码充分考虑到其余部分是否连通以及连通块形状任意这两个问题。

细节见代码:

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<map>
#include<queue>
#include<vector>
#include<cmath>
#include<unordered_set>
using namespace std;
typedef pair<int,int> PII;
const int N=12;
const int P=13331;//P用于哈希
unordered_set<unsigned long long>st;
PII p[N*N];//存放当前已选的点
int a[N][N];
bool vis[N][N];
int fu[N*N],sum,ans,n,m;
int dx[4]={0,0,1,-1},dy[4]={1,-1,0,0};
bool cmp(PII x,PII y)
{
	return a[x.first][x.second]>a[y.first][y.second];
}
int find(int x)
{
	if(fu[x]!=x) return fu[x]=find(fu[x]);
	return x;
}
bool check_connect(int cnt)//检查剩余的n*m-cnt个点是否连通 
{
	for(int i=1;i<=n*m;i++) fu[i]=i;
	int t=n*m-cnt;
	for(int i=1;i<=n;i++)
	for(int j=1;j<=m;j++)
	if(!vis[i][j])
	{
		for(int k=0;k<4;k++)
		{
			int nx=i+dx[k],ny=j+dy[k];
			if(nx<1||nx>n||ny<1||ny>m) continue;
			if(vis[nx][ny]) continue;
			int fx=find((i-1)*m+j),fy=find((nx-1)*m+ny);
			if(fx==fy) continue;
			t--;
			fu[fx]=fy;
		}
	}
	return t==1;
} 
bool check_exit()//检查当前连通块是否已经被搜索过,是的话返回true,否则返回false 
{
	unsigned long long t=0; 
	for(int i=1;i<=n;i++)
	for(int j=1;j<=m;j++)
		if(vis[i][j])
			t=t*P+i*(P-3)+j*(P+102);
	if(st.count(t)) return true;//哈希去重
	st.insert(t);
	return false;
}
void dfs(int s,int cnt)//s和cnt分别代表当前已经选出来的数的和及个数 
{
	if(s==sum/2)
	{
		if(check_connect(cnt)) ans=min(ans,cnt);//检查其他格子是否为一个连通块 
		return ;
	}
	if(s>sum/2||cnt>=ans) return ;//剪枝 
	PII next[N*N];//存放下一次搜索的点 
	int t=0;
	for(int i=1;i<=cnt;i++)
	{
		int nowx=p[i].first,nowy=p[i].second;
		for(int j=0;j<4;j++)
		{
			int nx=nowx+dx[j],ny=nowy+dy[j];
			if(nx<1||nx>n||ny<1||ny>m) continue;
			if(vis[nx][ny]) continue;
			next[++t]={nx,ny};
		}
	}
	sort(next+1,next+t+1,cmp);
	for(int i=1;i<=t;i++)
	{
		if(next[i]==next[i-1]) continue;
		p[cnt+1]=next[i];
		vis[next[i].first][next[i].second]=true;
		if(!check_exit())//检查当前连通块是否已经被搜索过
			dfs(s+a[next[i].first][next[i].second],cnt+1);
		vis[next[i].first][next[i].second]=false;
	}
}

int main()
{
	cin>>m>>n;
	for(int i=1;i<=n;i++)
	for(int j=1;j<=m;j++)
	{
		scanf("%d",&a[i][j]);
		sum+=a[i][j];
	}
	if(sum&1)
	{
		printf("0");
		return 0;
	}
	ans=0x3f3f3f3f;
	vis[1][1]=true;
	p[1]={1,1};
	dfs(a[1][1],1);
	if(ans==0x3f3f3f3f) ans=0;
	printf("%d",ans);
	return 0;
}

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

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

相关文章

FPGA和IC设计怎么选?哪个发展更好?

很多人纠结FPGA和IC设计怎么选&#xff0c;其实往小了说&#xff0c;要看你选择的具体是哪个方向岗位。往大了说&#xff0c;将来你要是走更远&#xff0c;要成为大佬&#xff0c;那基本各个方向的都要有涉及的。 不同方向就有不同的发展&#xff0c;目前在薪资上IC设计要比FP…

Vue3返回顶部组件及返回顶部js封装

介绍 vue3中,封装监听页面滚动的js, 及页面滚动到一定像素时,显示返回顶部的按钮,点击按钮会有放大的动画,并逐渐滚动到顶部的组件。效果如下: 代码 封装js,监听屏幕滚动事件,以及是否显示返回顶部的按钮; 在项目目录下新建 utils文件夹,并在该文件夹下创建index.…

国外SEO优化的重要性及应对策略

SEO是指搜索引擎优化&#xff0c;是一种通过优化网站的结构和内容&#xff0c;提高网站在搜索引擎中的排名&#xff0c;从而吸引更多的流量和潜在客户的过程。 国外SEO优化尤为重要&#xff0c;因为搜索引擎在全球范围内广泛使用&#xff0c;而谷歌是全球最受欢迎的搜索引擎之…

java Math类 和 System类 详解(通俗易懂)

Math类介绍Math类常用方法及演示System类简介System类常用方法及演示一、前言本节内容是我们《API-常用类》专题的第四小节了。本节内容主要讲Math类和System类&#xff0c; 内容包括Math类介绍、Math类常用方法、System类介绍&#xff0c;System类常用方法。该小节内容基本不涉…

【教程】你现在还不知道微软的New Bing?你out了,快点进来看

哈喽啊&#xff0c;大家好&#xff0c;好久不见&#xff0c;我是木易巷&#xff01; 不禁感叹&#xff0c;AI人工智能时代真的已经来临&#xff01; 目前&#xff0c;谷歌和微软就各自面向大众的产品发布了重大公告。谷歌推出了一款名为Bard实验性对话式 AI 服务&#xff0c;而…

Python开发入门之了解Python高阶函数

上段时间有小伙伴询问&#xff1a;高阶函数的问题&#xff0c;今天小编就带大家一起来看一看&#xff1a; 一、什么是高阶函数? 高阶函数是在Python中一个非常有用的功能函数&#xff0c;所谓高阶函数就是一个函数可以用来接收另一个函数作为参数&#xff0c;这样的函数叫做…

扬帆优配|数字经济刮起“东风”,龙头晋级7连板

今日两市共40只涨停股&#xff0c;主要集中于数字经济、6G板块&#xff0c;上一个交易日涨停股为29股&#xff1b;除掉18只ST股及3只一字板新股&#xff0c;共19股涨停。另外&#xff0c;4股封板未遂&#xff0c;整体封板率为83%。 6股封单金额超亿元 从收盘涨停板封单量来看&…

第54章 图片URL的后端获取

1 注意&#xff1a; 在.NetCore WebApi框架中&#xff0c;在默认情况下由于没有集成“UseStaticFiles”内置管道中间件方法&#xff0c;如果想要通过图片URL显示图片&#xff0c;由会显示“404”错误&#xff0c;必须先把“UseStaticFiles”内置管道中间件方法集成到.NetCore W…

安装MySQL数据库8.0服务实例

前言 之前尝试去安装了MySQL5.7的社区版本&#xff0c;今天来安装MySQL8.0的版本&#xff0c;并且以两种方式进行安装&#xff0c;一个是通过RPM包的安装&#xff0c;另一个则是编译的方式。 一. 前期准备 查看服务器IP [rootlocalhost ~]# hostname -I 192.168.161.166 19…

js中window自带的四舍五入toFixed方法中的坑以及解决办法

Hello&#xff0c;各位&#xff0c;我胡汉三~啊呸&#xff0c;我又回来啦&#xff0c;还改了名&#xff0c;换了头像&#xff0c;哈哈哈&#xff01;时隔这么长时间不更新了&#xff0c;太忙了&#xff0c;平时笔记都记在了自己的电脑上&#xff0c;从今天起&#xff0c;继续更…

vxe-grid 全局自定义filter过滤器,支持字典过滤

一、vxe-table的全局筛选器filters的实现 官网例子&#xff1a;https://vxetable.cn/#/table/renderer/filter 进入之后&#xff1a;我们可以参照例子自行实现&#xff0c;也可以下载它的源码&#xff0c;进行调整 下载好后并解压&#xff0c;用vscode将解压后的文件打开。全局…

CRM系统中的营销自动化能解决什么问题

CRM客户管理系统营销自动化的范围远远超出了人们的认知。许多人认为它只是自动化完成重复和乏味的任务来减少营销人员的工作量。虽然这确实占了很大一部分&#xff0c;但它真正的价值在于提高潜客转化&#xff0c;增加业务收入。那么&#xff0c;什么是CRM系统营销自动化&#…

195、【动态规划】AcWing —— 91. 最短Hamilton路径(C++版本)

题目描述 原题链接&#xff1a;91. 最短Hamilton路径 解题思路 动态规划五步曲&#xff1a; &#xff08;1&#xff09;dp[i][j]含义&#xff1a; 到达点j并且状态为i时&#xff0c;具有的最短路径长度&#xff0c;其中状态j用状态压缩二进制的方式表示。j中从0-n-1位分别对…

【玩转c++】List讲解和模拟底层实现

本期主题&#xff1a;list的讲解和模拟实现博客主页&#xff1a;小峰同学分享小编的在Linux中学习到的知识和遇到的问题小编的能力有限&#xff0c;出现错误希望大家不吝赐1.list的介绍和使用1.1.list的介绍1.list是可以在常数范围内在任意位置进行插入和删除的序列式容器&…

Good Idea, 利用MySQL JSON特性优化千万级文库表

&#x1f473;我亲爱的各位大佬们好&#x1f618;&#x1f618;&#x1f618; ♨️本篇文章记录的为 利用MySQL JSON特性优化千万级文库表 相关内容&#xff0c;适合在学Java的小白,帮助新手快速上手,也适合复习中&#xff0c;面试中的大佬&#x1f649;&#x1f649;&#x1f…

常见问题整理1

目录 偏差和方差 欠拟合underfitting 过拟合overfitting 梯度消失和梯度爆炸 归一化 偏差和方差 偏差&#xff1a;算法期望预测和真实预测之间的偏差程度。反应的是模型本身的拟合能力。 方差&#xff1a;度量了同等大小的训练集的变动导致学习性能的变化&#xff0c;刻画…

万字长文带你走进MySql优化(系统层面优化、软件层面优化、SQL层面优化)

文章目录系统层面优化采用分布式架构使用缓存使用搜索引擎软件层面优化调整 MySQL 参数配置定期清理无用数据创建索引创建索引普通索引唯一索引全文索引组合索引空间索引主键索引外键索引索引前缀适合创建索引的场景不适合创建索引的场景优化表结构分库分表SQL优化explain执行计…

Python3-File(文件) 方法

Python3 File(文件) 方法 open() 方法 Python open() 方法用于打开一个文件&#xff0c;并返回文件对象。 在对文件进行处理过程都需要使用到这个函数&#xff0c;如果该文件无法被打开&#xff0c;会抛出 OSError。 注意&#xff1a;使用 open() 方法一定要保证关闭文件对…

Nowcoder .链表分割

文章目录哨兵位节点哨兵位节点 链表分割 小于X 尾插到一个新链表 大于等于X 尾插到另一个链表 最后将两个链表链接起来 需要注意的细节&#xff1a;将第一个链表的尾与第二个链表的头相连接&#xff0c;再返回连接后的整个链表的头&#xff08;哨兵位头节点的下一个&#xff0…

ECharts 环形图组件封装

一、ECharts引入1.安装echarts包npm install echarts --save2.引入echarts这里就演示全局引入了&#xff0c;挂载到vue全局&#xff0c;后面使用时&#xff0c;直接使用 $echartsimport * as echarts from echarts Vue.prototype.$echarts echarts二、写echarts组件这里演示环…