【算法每日一练]-dfs bfs(保姆级教程 篇8 )#01迷宫 #血色先锋队 #求先序排列 #取数游戏 #数的划分

目录

今日知识点:

使用并查集映射点,构造迷宫的连通块

vis计时数组要同步当回合的处理

递归求先序排列

基于不相邻的取数问题:dfs+回溯

n个相同球放入k个相同盒子:dfs的优化分支暴力

01迷宫

血色先锋队

求先序排列

取数游戏 

数的划分 


        

        

01迷宫

思路1:

其实啊:在写这个题的时候,总觉得这个题没法做,感觉时间复杂度非常高。因为一共(n*m)^4,一共这么格子,然后每个格子有4种选择…………

然后我犯了一个错误,其实dfs的复杂度不是这么看的,因为其实很多分支根本就没走。远远达不到那种耗时。就比如说正常的走迷宫问题,设置一个vis数组,然后dfs的复杂度最大也就整张地图罢了,根本不会达到每个点都做那么多个分支的情况数。所以哈。dfs的实际复杂度还得看实际的效果!扯远了………………

        

设置dfs(int x,int y,int z,int t)进行四个方向的暴搜就行,因为最多也就是搜1000次,然后本次搜索过的之后就不需要再搜索了,完全不会超时。   

#include <bits/stdc++.h>
using namespace std;
char s[1005][1005];
int n,m,ans[100005],f[1005][1005];
void dfs(int x,int y,int z,int t){if(x<0||y<0||x>=n||y>=n||f[x][y]!=-1||s[x][y]-'0'!=z)return ;ans[t]++;f[x][y]=t;dfs(x-1,y,!z,t);dfs(x,y-1,!z,t);dfs(x+1,y,!z,t);dfs(x,y+1,!z,t);
}
int main(){cin>>n>>m;int x,y;memset(f,-1,sizeof(f));for(int i=0;i<n;i++)cin>>s[i];for(int i=0;i<m;i++){scanf("%d%d",&x,&y);x--;y--;if(f[x][y]==-1){dfs(x,y,s[x][y]-'0',i);	}else{ans[i]=ans[f[x][y]];	}	}for(int i=0;i<m;i++)printf("%d\n",ans[i]);
}

其实你也看到了, 就是一道连通块的题,扫一下每个连通块内有多少个块罢了。

思路2:

既然都是连通块的题了,并查集做才最合理!

操作就是不断的去合并两个点,合并点的话需要进行映射(说白了就是找一个东西来唯一的表示这个点)最直接做法的就是映射成一维。最后合并的时候一边维护fa一边维护cnt就行了。

注意:千万不能从(0,0)开始,不然会映射失败。

#include <bits/stdc++.h>
using namespace std;
string s[1005];
int n,m;
int dx[]={0,-1,0,1},dy[]={1,0,-1,0};
int fa[1005*1005],cnt[1005*1005];
int real (int i,int j){return (i-1)*n+j;}int find(int x){if(x!=fa[x])fa[x]=find(fa[x]);return fa[x];
}
int main(){cin>>n>>m;for(int i=1;i<=n;i++)cin>>s[i],s[i]="0"+s[i];int x,y,f1,f2;for(int i=1;i<=n*n;i++)fa[i]=i,cnt[i]=1;for(int i=1;i<=n;i++)for(int j=1;j<=n;j++)for(int k=0;k<4;k++){x=i+dx[k];y=j+dy[k];if(x>=1&&y>=1&&x<=n&&y<=n&&s[x][y]!=s[i][j]){f1=find(real(i,j));f2=find(real(x,y));if(f1!=f2){fa[f2]=f1;cnt[f1]+=cnt[f2];}}}while(m--){scanf("%d%d",&x,&y);printf("%d\n",cnt[find(real(x,y))]);}
}

        

        

血色先锋队

  思路:

 有一种特别直接的做法:其实想找领主的最早感染时候,不过就是直接曼哈顿距离就能求出来了。那么求一个领主需要O(a)时间,找b个领主需要O(a*b)时间,不好意思直接会超时的。

所以这不是最聪明的做法,而是做不好的做法。

虽然说感染源很多,但是我们可以把这个感染源一起进行扩散,然后扩散整个图的时候,所有领主的感染时间也就确定了。仅需要一张图的时间罢了。

这里有个bug改了好久才发现,vis标记一定要同步于当回合处理。

(果然,迷迷糊糊的时候尽量不要敲代码啊)

#include <bits/stdc++.h>
using namespace std;
int a,b,n,m;
int vis[550][550];
struct node{int x;int y;};
int dx[]={1,0,-1,0},dy[]={0,1,0,-1};
int main(){cin>>n>>m>>a>>b;int x,y,t=0;memset(vis,-1,sizeof(vis));queue<node>q;for(int i=1;i<=a;i++){scanf("%d%d",&x,&y);q.push(node{x,y});vis[x][y]=t;//入队就标记:一定要当回就标记,避免下回合开始之后上回合还没有标记}while(!q.empty()){int sz=q.size();t++;while(sz--){node cur=q.front();q.pop();for(int i=0;i<4;i++){int tx=cur.x+dx[i],ty=cur.y+dy[i];if(tx<1||ty<1||tx>n||ty>m||vis[tx][ty]!=-1)continue;q.push(node{tx,ty});vis[tx][ty]=t;//在利用队列中点进行下一层处理时候,当前队列的所有点均已经安全标记}	}}for(int i=1;i<=b;i++){scanf("%d%d",&x,&y);printf("%d\n",vis[x][y]);}
}

        

         

求先序排列

思路:

一道dfs题的老朋友了。 

直接模拟就行:给个例子:中序ACGDBHZKX,后序CDGAHXKZB

找到主根B,然后左子树(ACGD,CDGA)重复操作,右子树(HZKX,HXKZ)重复操作即可。

然后注意输出是先序,所以先输出根,然后递归左边,最后是右边。

最最后注意结束条件就行了(长度为1就结束,不要进入错误操作)

#include <bits/stdc++.h>
using namespace std;void dfs(string s1,string s2){int sz1=s1.size(),sz2=s2.size();if(sz1==1){cout<<s1;return ;}string s11,s12,s21,s22;char ch=s2[sz2-1];int i=s1.find(ch);s11=s1.substr(0,i);s12=s1.substr(i+1);	s21=s2.substr(0,i);s22=s2.substr(i,sz2-i-1);cout<<ch;if(s11.size())dfs(s11,s21);if(s12.size())dfs(s12,s22);
}
int main(){string s1,s2;cin>>s1>>s2;dfs(s1,s2);
}

        

         

取数游戏 

思路:

我一开始的思路是dfs(x1,y1,x2,y2)然后转移到下一个(i,j,i+1,j+1)也就是用i+1,j+1来标记走过的状态。嗯~ 后面再一看,我去,这样子只能不与上一个状态冲突,那么和上上个状态就冲突。反正就是必须整个图都要进行标记,我标记的不够。

那么就dfs+回溯,进行全图标记。

 首先装模装样的分析一下dfs的时间复杂度:首先基于回溯的话,每个格子妥妥的要跑2种状态。一共36个格子,所以2^36??? 等等,因为是跳着走的,所以因为2^18左右吧(其是不是很会)

#include <bits/stdc++.h>
using namespace std;
const int d[8][2]={1,0,-1,0,0,1,0,-1,1,1,-1,1,1,-1,-1,-1};
int ans,n,m,t;
int vis[8][8],a[8][8];void dfs(int x,int y,int mx){if(y==m+1){dfs(x+1,1,mx);return ;}if(x==n+1){ans=max(ans,mx);return ;}dfs(x,y+1,mx);if(vis[x][y]==0){for(int i=0;i<8;i++){vis[x+d[i][0]][y+d[i][1]]++;}dfs(x,y+1,mx+a[x][y]);for(int i=0;i<8;i++){vis[x+d[i][0]][y+d[i][1]]--;}}
}
int main(){cin>>t;while(t--){ans=0;memset(a,0,sizeof(a));memset(vis,0,sizeof(vis));cin>>n>>m;for(int i=1;i<=n;i++)for(int j=1;j<=m;j++)cin>>a[i][j];dfs(1,1,0);cout<<ans<<'\n';}
}

        

         

数的划分 

思路:

两种做法:

第一种:动态规划:

题目可以理解成把n个相同球放入k个相同盒子,然后因为球都是相同的,就不能再对最后一个球进行讨论了。应该对应一类球:

设置a[i][j]表示i个球放入j个盒子的方案数。

第一种情况:有一个盒子只有一个球,那么就对应了a[i-1][j]

第二种情况:每个盒子都至少有两个球,那么就对应看a[i-j][j]

所以:a[i][j]=a[i-j][j]+a[i-1][j-1]

第二种:dfs:

在已经放了i时候,每次可以放1~n-i个,所有dfs(i)有n-i个分支,这个复杂度很高,别着急,只需要把无效分支剔除即可很快。

仔细观察7的拆法:

1 1 5

1 2 4

1 3 3

2 2 3

2 3 2(重复了哟)

所以你发现了,要想不重复 ,就必须后面选的数比前面的大,(很早的时候我们再输出组合数的时候就是这样子去重的,只需要安排升序即可),所以在dfs(i)也就是选了i的时候,后面选的数都必须比i大,那么有了sum+i*(k-cnt)<=n这个分支优化。

dfs的速度就变快了很多。

#include <bits/stdc++.h>
using namespace std;
int n,k,ans=0,a[205][70];
//int main(){
//	cin>>n>>k;
//	for(int i=1;i<=n;i++)a[i][1]=1;
//	for(int i=2;i<=n;i++)
//		for(int j=2;j<=(i,k);j++){
//			a[i][j]=a[i-1][j-1];
//			if(i>=2*j)a[i][j]+=a[i-j][j];
//		}
//	cout<<a[n][k];
//}
void dfs(int cnt,int up,int sum){if(cnt==k){if(sum==n)ans++;return ;}for(int i=up;sum+i*(k-cnt)<=n;i++){dfs(cnt+1,i,sum+i);}
}
int main(){cin>>n>>k;dfs(0,1,0);cout<<ans;
}

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Unity添加所有场景到BuildSettings

Unity添加所有场景到BuildSettings using UnityEngine; using UnityEditor; using System.Collections.Generic; using System.IO; public class Tools : Editor {[MenuItem("Tools/添加所有场景到BuildSettings")]static void CheckSceneSetting(){List<string&…

BOM,JS执行机制等

1.BOM 概述 1.1什么是 BOM BOM( Browser Object Model &#xff09;即浏览器对象模型&#xff0c;它提供了独立于内容而与浏览器窗口进行交互的对象&#xff0c;其核心对象是window. BOM由一系列相关的对象构成&#xff0c;并且每个对象都提供了很多方法与属性。 BOM缺乏标…

十九:爬虫最终篇-平安银行商城实战

平安银行商场实战 需求 获取该商城商品信息 目标网址 https://m.yqb.com/bank/product-item-50301196.html?mcId1583912328849970&loginModepab&historyy&sceneModem&traceid30187_4dXJVel1iop详细步骤 1、寻找数据接口 2、对比payload寻找可疑参数 3、多…

系列十四、while do...while switch模板代码

一、while & do...while & switch模板代码 1.1、while /*** 需求&#xff1a;使用while循环打印5遍Hello World!*/ Test public void print5() {int i 1;while (i < 5) {System.out.println("Hello World! " LocalDateTime.now());// 线程休眠&#x…

大模型笔记【2】 LLM in Flash

Apple最近发表了一篇文章&#xff0c;可以在iphone, MAC 上运行大模型&#xff1a;【LLM in a flash: Efficient Large Language Model Inference with Limited Memory】。 主要解决的问题是在DRAM中无法存放完整的模型和计算&#xff0c;但是Flash Memory可以存放完整的模型。…

05、Kafka ------ 各个功能的作用解释(主题和分区 详解,用命令行和图形界面创建主题和查看主题)

目录 CMAK 各个功能的作用解释&#xff08;主题&#xff09;★ 主题★ 分区★ 创建主题&#xff1a;★ 列出和查看主题 CMAK 各个功能的作用解释&#xff08;主题&#xff09; ★ 主题 Kafka 主题虽然也叫 topic&#xff0c;但它和 Pub-Sub 消息模型中 topic 主题及 AMQP 的 t…

【实用技巧】Windows 电脑向iPhone或iPad传输视频方法1:无线传输

一、内容简介 本文介绍如何使用 Windows 电脑向 iPhone 或 iPad 传输视频&#xff0c;以 iPhone 为例&#xff0c;iPad的操作方法类似&#xff0c;本文不作赘述。 二、所需原材料 Windows 电脑&#xff08;桌面或其它文件夹中存有要导入的视频&#xff09;、iPhone 14。 待…

如何通过PreMaint状态监测发现设备故障:以振动监测为例

在现代工业环境中&#xff0c;设备的健康状况对于维持生产效率至关重要。计划外停机可能导致巨大的成本损失&#xff0c;因此采用先进的监测技术成为预防性维护的核心策略之一。其中&#xff0c;振动监测作为一种早期故障检测手段&#xff0c;通过PreMaint状态监测系统的引入&a…

从虚拟到现实:数字孪生驱动智慧城市可持续发展

随着科技的飞速发展&#xff0c;智慧城市已经成为未来城市发展的重要趋势。数字孪生技术作为智慧城市建设中的关键技术之一&#xff0c;正在发挥着越来越重要的作用。本文将探讨数字孪生如何从虚拟走向现实&#xff0c;驱动智慧城市的可持续发展。 一、数字孪生技术&#xff1…

解决sublime中文符号乱码问题

效果图 原来 后来 问题不是出自encode文件编码&#xff0c;而是win10的字体问题。 解决方法 配置&#xff1a; { "font_face":"Microsoft Yahei", "dpi_scale": 1.0 } 参考自 Sublime 输入中文显示方框问号乱码_sublime中文问号-CSDN博…

【开源】基于JAVA+Vue+SpringBoot的大学计算机课程管理平台

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 实验课程档案模块2.2 实验资源模块2.3 学生实验模块 三、系统设计3.1 用例设计3.2 数据库设计3.2.1 实验课程档案表3.2.2 实验资源表3.2.3 学生实验表 四、系统展示五、核心代码5.1 一键生成实验5.2 提交实验5.3 批阅实…

系列十四、理解MySQL varchar(50)

一、理解MySQL varchar(50) 1.1、概述 日常开发中&#xff0c;数据库建表是必不可少的一个环节&#xff0c;建表的时候通常会看到设定某个字段的长度为varchar(50)&#xff0c;例如如下建表语句&#xff1a; 那么怎么理解varchar(50)&#xff1f;这个分情况的&#xff0c;MySQ…

关于“Python”的核心知识点整理大全61

目录 注意 20.1.4 使用 jumbotron 设置主页的样式 index.html 20.1.5 设置登录页面的样式 login.html 20.1.6 设置 new_topic 页面的样式 new_topic.html 20.1.7 设置 topics 页面的样式 topics.html 元素&#xff0c;让它们在页面上显得大些&#xff08;见2&#xf…

有趣的前端知识(二)

推荐阅读 智能化校园&#xff1a;深入探讨云端管理系统设计与实现&#xff08;一&#xff09; 智能化校园&#xff1a;深入探讨云端管理系统设计与实现&#xff08;二&#xff09; 文章目录 推荐阅读HTML元素元素属性头部元素列表元素区块元素表单元素 颜色字符实体 HTML元素 …

OpenHarmony之hdc

OpenHarmony之hdc 简介 hdc&#xff08;OpenHarmony Device Connector&#xff09;是 OpenHarmony 为开发人员提供的用于调试的命令行工具&#xff0c;通过该工具可以在Windows/Linux/MacOS等系统上与开发机或者模拟器进行交互。 类似于Android的adb&#xff0c;和adb类似&a…

BlogPark测试报告

目录 一&#xff0c;项目背景 二&#xff0c;项目功能 三&#xff0c;测试计划 3.1 测试用例的设计 3.2 功能测试 1.正常登录 2.正常写博客测试 &#xff08;输入完整的标题和内容&#xff09; 3.发布博客之后跳转到详情页观察是否有刚发布的博客 4.删除博客观察列表的…

Linux下从sqlite3源码编译出sqlite3库及相关可执行程序

目录 1. 下载sqlite3源码并编译 2. 下载Tcl库并编译 3. 再次编译sqlite源码 1. 下载sqlite3源码并编译 打开SQLite Download Page&#xff0c;滚动到页面的下面&#xff0c;找到源码量最大的那个&#xff08;其它的估计也行&#xff0c;但源码最大的本人感觉功能最全&#…

【unity小技巧】FPS游戏实现相机的偏移震动、武器射击后退和后坐力效果

最终效果 文章目录 最终效果前言相机偏移震动相机震动脚本换弹节点震动 武器射击后退效果武器后坐力效果完结 前言 关于后坐力之前其实已经分享了一个&#xff1a;FPS游戏后坐力制作思路 但是实现起来比较复杂&#xff0c;如果你只是想要简单的实现&#xff0c;可以看看这个&…

软件测试|Python对JSON的解析和创建详解

简介 JSON&#xff08;JavaScript Object Notation&#xff09;是一种轻量级的数据交换格式&#xff0c;已经成为当今互联网应用中广泛使用的数据格式之一。Python提供了内置的模块来解析和创建JSON数据&#xff0c;使得在Python中处理JSON变得非常简单。本文将详细介绍Python…

GB28181视频汇聚平台EasyCVR级联后,部分通道视频无法播放是什么原因?

GB28181协议智慧安防平台EasyCVR是基于各种IP流媒体协议传输的视频汇聚和融合管理平台。视频流媒体服务器EasyCVR采用了开放式的网络结构&#xff0c;支持高清视频的接入和传输、分发&#xff0c;平台提供实时远程视频监控、视频录像、录像回放与存储、告警、语音对讲、云台控制…