蓝桥杯国赛算法复习

2024-04-24 11:12
文章标签 算法 蓝桥 复习 杯国赛

本文主要是介绍蓝桥杯国赛算法复习,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

复习内容

1.spfa
2.背包问题
3.动态规划其他常考问题
4.dfs
5.bfs
6.并查集

一、基础题回顾
1.spfa
问题描述
蒜头君准备去参加骑车比赛,比赛在 n 个城市间进行,编号从 1 到 n。选手们都从城市 1 出发,终点在城市 n。
已知城市间有 m 条道路,每条道路连接两个城市,注意道路是双向的。现在蒜头君知道了他经过每条道路需要花费的时间,他想请你帮他计算一下,他这次比赛最少需要花多少时间完成。
输入格式
第一行输入两个整数\n,m(\1≤n≤1,000,1≤m≤5,000),分别代表城市个数和道路总数。接下来输入 m 行,每行输入三个数字 a,b,c(1≤a,b≤n,1≤c≤200),分别代表道路的起点和道路的终点,以及蒜头君骑车通过这条道路需要花费的时间。保证输入的图是连通的。
输出格式
输出一行,输出一个整数,输出蒜头君完成比赛需要的最少时间。
样例输入
5 6
1 2 2
2 3 3
2 5 5
3 4 2
3 5 1
4 5 1
样例输出
6

题解

#include<iostream>
#include<cstring>
#include<queue>
using namespace std;
const int N = 1e4+9;
const int M = 1e5+9;
struct edge{int u;int w;int next;edge(){}edge(int uu,int ww,int nn){u = uu;w = ww;next = nn;}
}e[N << 1];
int head[N],size;
void init(){memset(head,-1,sizeof(head));size = 0;
}
void add(int u,int v,int w){e[size] = edge(u,w,head[v]);head[v] = size++;
}
void add2(int u,int v,int w){add(u,v,w);add(v,u,w);
}
int n,m;
int dis[N];
bool vis[N];
void spfa(int u){memset(dis,0x3f,sizeof(dis));dis[u] = 0;memset(vis,false,sizeof(vis));vis[u] = true;queue<int> q;q.push(u);while(!q.empty()){u = q.front();q.pop();vis[u] = false;for(int j = head[u];~j;j = e[j].next){int v = e[j].u;int w = e[j].w;if(dis[v] > dis[u] + w){dis[v] = dis[u] + w;if(!vis[v]){q.push(v);vis[v] = true;}}}}
}
int main(){init();int u,v,w;cin >>n>>m;while(m--){cin >>u>>v>>w;add2(u,v,w);}spfa(1);cout <<dis[n]<<endl;return 0;
}

2.背包问题
01背包

问题描述:
有N件物品和一个容积为M的背包。第i件物品的体积为volume[i],价值为worth[i]。求解将哪些物品装入背包可使价值总和最大。每种物品只有一件,可以选择放或者不放。(N<=3500,M<=13000)输入格式:
第一行为物品数量N和背包容积M每行依次输入第i件物品的价值和体积样例输入:
3 103 42 66 7样例输出:
6
#include<iostream>
#include<algorithm>
using namespace std;
int dp[100][100];
int v[100],w[100];
int n,m;
int main(){cin >>n>>m;int maxs = -1;for(int i = 0;i<n;i++){cin >>v[i]>>w[i];}for(int i = 0;i<n;i++){for(int j = 0;j<m;j++){if(j < w[i]){dp[i][j] = dp[i-1][j];}else{dp[i][j] = max(dp[i-1][j],dp[i-1][j-w[i]]+v[i]);}maxs = max(maxs,dp[i][j]);}}cout <<maxs<<endl;return 0;
}

多重背包

题目描述:
有N种物品,第i种物品的体积是c[i],价值是w[i],每种物品的数量都是有限的,为n[i]。现有容量为V的背包,请你放入若干物品,在总体积不超过V的条件下,使总价值尽可能大。
#include<iostream>
#include<cstring>
using namespace std;
int dp[21][1010];
int w[21],c[21],n[21];
int main()
{int N,V;cin>>N>>V;for(int i=1;i<=N;++i)cin>>w[i]>>c[i]>>n[i];for(int i=1;i<=N;++i){for(int j=0;j<=V;++j){for(int k=0;k<=n[i];++k){if(j>=c[i]*k)dp[i][j]=max(dp[i-1][j-c[i]*k]+w[i]*k,dp[i][j]);}}}cout<<dp[N][V]<<endl;return 0;
}

完全背包

问题描述:
当前有N种物品,第i种物品的体积是c[i],价值是w[i]。
每种物品的数量都是无限的,可以任意选择若干件。
现有容量为V的背包,请你放入若干物品,在总体积不超过V的条件下,使总价值尽可能大。
与01背包问题的区别就是物品有无限多个.
#include<iostream>
#include<cstring>
using namespace std;
int dp[21][1010];
int w[21],c[21];
int main()
{int N,V;cin>>N>>V;for(int i=1;i<=N;++i)cin>>w[i]>>c[i];for(int i=1;i<=N;++i){for(int j=0;j<=V;++j){if(j>=c[i])dp[i][j]=max(dp[i][j-c[i]]+w[i],dp[i-1][j]);else dp[i][j]=dp[i-1][j];}}cout<<dp[N][V]<<endl;return 0;
}

3.动态规划其他常考问题

楼梯问题

贪心

最大字段和

给定N个数A1, A2, ... An,从中选出k(k不固定)个连续的数字 Ai, Ai+1, ... Ai+k-1,使得∑i+k−1iAt 达到最大,求该最大值。
#include<iostream>
#include<algorithm>
using namespace std;
int dp[100];
int a[100];
int n;
int main(){cin >>n;int maxs = -1;for(int i = 1;i<=n;i++){cin >>a[i];dp[i] = a[i];}for(int i = 2;i<=n;i++){dp[i] = max(dp[i],dp[i-1]+a[i]);maxs = max(maxs,dp[i]);}cout <<maxs<<endl;return 0;
}

最长上升子序列

#include<iostream>
#include<algorithm>
using namespace std;
int dp[100];
int a[100];
int n;
int main(){cin >>n;int maxs = -1;for(int i = 1;i<=n;i++){cin >>a[i];}for(int i = 1;i<=n;i++){dp[i] = 1;for(int j = 1;j<=i;j++){if(a[j] < a[i]){dp[i] = max(dp[i],dp[j]+1);}}maxs = max(maxs,dp[i]);}cout <<maxs<<endl;return 0;
}

数塔问题
在讲述DP算法的时候,一个经典的例子就是数塔问题,它是这样描述的:

有如下所示的数塔,要求从顶层走到底层,若每一步只能走到相邻的结点,则经过的结点的数字之和最大是多少?
在这里插入图片描述
Output
对于每个测试实例,输出可能得到的最大和,每个实例的输出占一行。

Sample Input
1 5 7 3 8 8 1 0 2 7 4 4 4 5 2 6 5

Sample Output
30

#include<iostream>
#include<algorithm>
using namespace std;
int dp[100][100];
int a[100][100];
int n,m;
int main(){cin >>n>>m;for(int i = 0;i<n;i++){for(int j = 0;j<m;j++){cin >>a[i][j];}}for(int i = 0;i<n;i++){dp[n][i] = a[n][i];}for(int i = n-1;i>1;i--){for(int j = 0;j<=i;j++){dp[i][j] = max(dp[i+1][j],dp[i+1][j+1])+a[i];}} cout <<dp[0][0]<<endl;
}
4 dfs
问题描述n个人参加某项特殊考试。为了公平,要求任何两个认识的人不能分在同一个考场。求是少需要分几个考场才能满足条件。
输入格式第一行,一个整数n(1<n<100),表示参加考试的人数。第二行,一个整数m,表示接下来有m行数据以下m行每行的格式为:两个整数a,b,用空格分开 (1<=a,b<=n) 表示第a个人与第b个人认识。
输出格式一行一个整数,表示最少分几个考场。
样例输入
5
8
1 2
1 3
1 4
2 3
2 4
2 5
3 4
4 5
样例输出
4
样例输入
5
10
1 2
1 3
1 4
1 5
2 3
2 4
2 5
3 4
3 5
4 5
样例输出
5
#include<bits/stdc++.h>
using namespace std;
const int MAXN = 1e2+2;
int g[MAXN][MAXN];//用邻接矩阵存储关系
int room[MAXN][MAXN]; //room[i][j]表示当前i号教室第j个人为room[i][j]
int MIN ;//优化一:当前状态下的最小教室
int n,m;//点数,边数void dfs(int v,int c){if(c>=MIN)return;//剪枝if(v>n){MIN=MIN>c?c:MIN;return;}for(int i=1;i<=c;++i){//循环判断每个教室的每个人是否与v有关系int k =0;while(room[i][k]&&g[v][room[i][k]]==0){//如果当前教室i,第k个人的编号room[i][k]与v有关系退出k++;}if(room[i][k]==0){//有当前教室i的所有人无关系,可加入教室room[i][k]=v;dfs(v+1,c);//把v+1加入到教室room[i][k]=0;//回溯}}room[c+1][0]=v;//或者新开一间教室放该v,(注不是所有教室不满足条件)dfs(v+1,c+1);room[c+1][0]=0;}int main(){//相当于图的着色问题,用dfs scanf("%d%d",&n,&m);MIN = n;//易知最多教室为nint a,b;for(int i=0;i<m;++i){scanf("%d%d",&a,&b);g[a][b]=g[b][a]=1;}dfs(1,0);printf("%d",MIN);return 0;
}
A - 滑雪 POJ - 1088
Michael喜欢滑雪百这并不奇怪, 因为滑雪的确很刺激。可是为了获得速度,滑的区域必须向下倾斜,而且当你滑到坡底,你不得不再次走上坡或者等待升降机来载你。Michael想知道载一个区域中最长底滑坡。区域由一个二维数组给出。数组的每个数字代表点的高度。下面是一个例子
1 2 3 4 516 17 18 19 615 24 25 20 714 23 22 21 813 12 11 10 9一个人可以从某个点滑向上下左右相邻四个点之一,当且仅当高度减小。在上面的例子中,一条可滑行的滑坡为24-17-16-1。当然25-24-23-…-3-2-1更长。事实上,这是最长的一条。Input
输入的第一行表示区域的行数R和列数C(1 <= R,C <= 100)。下面是R行,每行有C个整数,代表高度h,0<=h<=10000。Output
输出最长区域的长度。Sample Input
5 5
1 2 3 4 5
16 17 18 19 6
15 24 25 20 7
14 23 22 21 8
13 12 11 10 9Sample Output
25
#include<iostream>
#include<algorithm>
using namespace std;
int n,m;
int mp[100][100];
int ansm = -1;
int dir[4][2] = {{0,1},{0,-1},{1,0},{-1,0}};
int dfs(int x,int y){int ans = -1;if(ans > 0){return ans;}ans = 1;for(int i = 0;i<4;i++){int tx = x + dir[i][0];int ty = y + dir[i][1];if(tx >= 0 && tx < n && ty >= 0 && ty < m && mp[tx][ty] < mp[x][y]){int temp = dfs(tx,ty) + 1;ans = max(ans,temp);} }return ans; 
}
int main(){cin >>n>>m;for(int i = 0;i<n;i++){for(int j = 0;j<m;j++){cin >>mp[i][j];}}for(int i = 0;i<n;i++){for(int j = 0;j<m;j++){int t = dfs(i,j);ansm = max(ansm,t);}}cout <<ansm<<endl;
} 
5.bfs

迷宫最短路

#include<iostream>
#include<queue>
using namespace std;
struct node{int x,y,s;node(){}node(int xx,int yy,int ss){x = xx;y = yy;s = ss;}
};
int n,m;
char mp[100][100];
bool vis[100][100];
int dir[4][2] = {{1,0},{-1,0},{0,1},{0,-1}};
int bfs(int x,int y){queue<node> q;q.push(node(x,y,0));vis[x][y] = true;while(!q.empty()){node now = q.front();q.pop();for(int i = 0;i<4;i++){int tx = x + dir[i][0];int ty = y + dir[i][1];if(tx >= 0 && tx < n && ty >= 0 && ty < m && !vis[tx][ty] && mp[tx][ty] != '#'){if(mp[tx][ty] == 'T'){return now.s + 1;}else{vis[tx][ty] = true;q.push(node(tx,ty,now.s+1));}}}}
}
int main(){cin >>n>>m;for(int i = 0;i<n;i++){cin >>mp[i];}int sx,sy;for(int i = 0;i<n;i++){for(int j = 0;j<m;j++){if(mp[i][j] == 'S'){sx = i;sy = j;}}}cout <<bfs(sx,sy)<<endl;return 0;
}

拓展
dfs与bfs的抽象问题

例1:给定n nn个整数,要求选出K KK个数,使得选出来的K KK个数的和为s u m sumsum
#include <iostream>
using namespace std;
int a[40];
int n, k, sum, ans;
//i表示选择第i个数,cnt记录选择的个数,s表示选取数的和
void dfs(int i, int cnt, int s) {if (i == n) {if (cnt == k && s == sum) {ans++;}return;}dfs(i + 1, cnt, s); //不选该数dfs(i + 1, cnt + 1, s + a[i]); //选择该数
}
int main() {cin >> n >> k >> sum;for (int i = 0; i < n; ++i) {cin >> a[i];}ans = 0;dfs(0, 0, 0);cout << ans << endl;return 0;
}
例2:取木棍:
有4个木棍输入每个木棍长度;
判断是否可以拼成三角形:
比如 1 2 3 3可以拼成三角形:
输出yes;
#include <iostream>
using namespace std;
int n,m,sum;
int a[10010];
bool vis[10010];
bool f;
void dfs(int p,int s,int st)
{if(f){return ;}if(p==3){f=true;return ;}if(s=sum/3)//当前的边正好可以构成三角形的一个边 {dfs(p+1,0,0);return ;	}for(int i=0;i<n;++i){if(!vis[i]){vis[i]=true;dfs(p,s+a[i],i+1);vis[i]=false;}}
}
int main()
{cin>>n;for(int i=0;i<n;++i){cin>>a[i];sum+=a[i];}if(sum%3!=0)//如果不是3的倍数的话就已经明摆着不可以成功 {cout<<"no"<<endl;return 0;}else dfs(0,0,0);if(f) cout<<"yes"<<endl;else cout<<"no"<<endl;return 0; } 

这篇关于蓝桥杯国赛算法复习的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/931592

相关文章

Python中的随机森林算法与实战

《Python中的随机森林算法与实战》本文详细介绍了随机森林算法,包括其原理、实现步骤、分类和回归案例,并讨论了其优点和缺点,通过面向对象编程实现了一个简单的随机森林模型,并应用于鸢尾花分类和波士顿房... 目录1、随机森林算法概述2、随机森林的原理3、实现步骤4、分类案例:使用随机森林预测鸢尾花品种4.1

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

综合安防管理平台LntonAIServer视频监控汇聚抖动检测算法优势

LntonAIServer视频质量诊断功能中的抖动检测是一个专门针对视频稳定性进行分析的功能。抖动通常是指视频帧之间的不必要运动,这种运动可能是由于摄像机的移动、传输中的错误或编解码问题导致的。抖动检测对于确保视频内容的平滑性和观看体验至关重要。 优势 1. 提高图像质量 - 清晰度提升:减少抖动,提高图像的清晰度和细节表现力,使得监控画面更加真实可信。 - 细节增强:在低光条件下,抖

【数据结构】——原来排序算法搞懂这些就行,轻松拿捏

前言:快速排序的实现最重要的是找基准值,下面让我们来了解如何实现找基准值 基准值的注释:在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。 在此我们采用三数取中法,也就是取左端、中间、右端三个数,然后进行排序,将中间数作为枢纽值。 快速排序实现主框架: //快速排序 void QuickSort(int* arr, int left, int rig

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

秋招最新大模型算法面试,熬夜都要肝完它

💥大家在面试大模型LLM这个板块的时候,不知道面试完会不会复盘、总结,做笔记的习惯,这份大模型算法岗面试八股笔记也帮助不少人拿到过offer ✨对于面试大模型算法工程师会有一定的帮助,都附有完整答案,熬夜也要看完,祝大家一臂之力 这份《大模型算法工程师面试题》已经上传CSDN,还有完整版的大模型 AI 学习资料,朋友们如果需要可以微信扫描下方CSDN官方认证二维码免费领取【保证100%免费

dp算法练习题【8】

不同二叉搜索树 96. 不同的二叉搜索树 给你一个整数 n ,求恰由 n 个节点组成且节点值从 1 到 n 互不相同的 二叉搜索树 有多少种?返回满足题意的二叉搜索树的种数。 示例 1: 输入:n = 3输出:5 示例 2: 输入:n = 1输出:1 class Solution {public int numTrees(int n) {int[] dp = new int

Codeforces Round #240 (Div. 2) E分治算法探究1

Codeforces Round #240 (Div. 2) E  http://codeforces.com/contest/415/problem/E 2^n个数,每次操作将其分成2^q份,对于每一份内部的数进行翻转(逆序),每次操作完后输出操作后新序列的逆序对数。 图一:  划分子问题。 图二: 分而治之,=>  合并 。 图三: 回溯: