第十四届蓝桥杯c++b组

2024-04-12 22:12
文章标签 c++ 蓝桥 第十四届

本文主要是介绍第十四届蓝桥杯c++b组,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.日期统计

前四位已经固定,四重循环暴力生成后四位,并将合法的日期记录下来,然后检验这些日期能否被生成

具体如何检验:

由于题目说的是子序列,所以可以跳着选,但是相对顺序是不能变的,所以对于给定的100个数,从前往后一位一位和日期的8位匹配,如果一位匹配成功就匹配下一位,如果日期的8位都匹配成功了,那么就说明该日期合法

#include<bits/stdc++.h>
#define endl '\n'
#define int long long
using namespace std;
int mon[13]={-1,31,29,31,30,31,30,31,31,30,31,30,31};
void solve() {set<int>s;int a[100]={5,6,8,6,9,1,6,1,2,4,9,1,9,8,2,3,6,4,7,7,5,9,5,0,3,8,7,5,8,1,5,8,6,1,8,3,0,3,7,9,2,7,0,5,8,8,5,7,0,9,9,1,9,4,4,6,8,6,3,3,8,5,1,6,3,4,6,7,0,7,8,2,7,6,8,9,5,6,5,6,1,4,0,1,0,0,9,4,8,0,9,1,2,8,5,0,2,5,3,3};for(int i=0;i<=9;i++){for(int j=0;j<=9;j++){for(int k=0;k<=9;k++){for(int m=0;m<=9;m++){int tmp=2023;tmp=tmp*10+i;tmp=tmp*10+j;tmp=tmp*10+k;tmp=tmp*10+m;string x=to_string(tmp);int month=(x[4]-'0')*10+x[5]-'0';int day=(x[6]-'0')*10+x[7]-'0';if(month>=1&&month<=12&&day>=1&&day<=mon[month]) s.insert(tmp);}}}}int ans=0;for(auto v:s){string t=to_string(v);int cnt=0;for(int i=0;i<100;i++){if(a[i]==t[cnt]-'0') cnt++;}if(cnt==(int)t.size()) ans++;}cout<<ans<<endl;
}
signed main() {ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);int t=1;
//    cin>>t;while(t--) {solve();}return 0;
}

2.01串的熵

就是很简单的式子,枚举0的个数和1的个数,检验能否满足给定式子

#include<bits/stdc++.h>
#define endl '\n'
#define eps 1e-4
#define int long long
using namespace std;
void solve() {for(int i=0;i<=23333333;i++){//枚举0的个数int j=23333333-i;//1的个数if(i>j) break;double p0=(double)i/23333333;//0的占比double p1=(double)j/23333333;//1的占比double sum0=(-1)*i*p0*log2(p0);double sum1=(-1)*j*p1*log2(p1);double sum=sum0+sum1;if(abs(sum-11625907.5798)<eps){cout<<i<<endl;return;}}
}
signed main() {ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);int t=1;
//    cin>>t;while(t--) {solve();}return 0;
}

3.冶炼金属

法一:

找一下规律,比如题目中的样例

3
75 3
53 2
59 2
75/3=25
75/4=1875/25=3
75/24=3
75/23=3
...
75/19=3
75/18=4
#include<bits/stdc++.h>
#define endl '\n'
#define int long long
using namespace std;
const int N=1e4+10;
int a[N],b[N];
int n;
void solve() {cin>>n;for(int i=1;i<=n;i++) cin>>a[i]>>b[i];int maxn=2e9,minn=0;for(int i=1;i<=n;i++){int u=a[i]/b[i];int v=a[i]/(b[i]+1)+1;maxn=min(maxn,u);minn=max(minn,v);}cout<<minn<<' '<<maxn<<endl;
}
signed main() {ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);int t=1;
//    cin>>t;while(t--) {solve();}return 0;
}

法二:二分答案

在[1,1e9]二分出一个答案

如果是求最小的转换率,那么对于每个答案,check每条记录是否满足产出均小于等于b[i],如果是返回true,找到最小的返回为true的转换率

产出的均小于等于b,需要找到最小的x满足产出均等于b[i],一定存在,因为题目说一定有解

求最大的转换率同理

#include<bits/stdc++.h>
#define endl '\n'
#define int long long
using namespace std;
const int N=1e4+10;
int a[N],b[N];
int n;
bool check1(int x){for(int i=1;i<=n;i++){if(a[i]/x>b[i]) return false;}return true;//产出的均小于等于b,需要找到最小的x满足产出均等于b[i],一定存在,因为题目说一定有解
}
bool check2(int x){for(int i=1;i<=n;i++){if(a[i]/x<b[i]) return false;}return true;//产出的均大于等于b,需要找到最大的x满足产出均等于b[i],一定存在,因为题目说一定有解
}
void solve() {cin>>n;for(int i=1;i<=n;i++) cin>>a[i]>>b[i];int l=1,r=1e9;while(l<r){int mid=(l+r)>>1;if(check1(mid)) r=mid;else l=mid+1;}int minn=l;l=1,r=1e9;while(l<r){int mid=(l+r+1)>>1;if(check2(mid)) l=mid;else r=mid-1;}int maxn=l;cout<<minn<<' '<<maxn<<endl;
}
signed main() {ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);int t=1;
//    cin>>t;while(t--) {solve();}return 0;
}

4.飞机降落

数据很小,直接暴力枚举每种情况,并check是否至少一种情况可以使得飞机成功降落

全排列即可

#include<bits/stdc++.h>
#define endl '\n'
#define int long long
using namespace std;
const int N=15;
int t[N],d[N],l[N];
int a[N];
int n;
void solve() {cin>>n;for(int i=1;i<=n;i++) cin>>t[i]>>d[i]>>l[i],a[i]=i;do{bool ok=true;int ed=0;//上一架飞机降落结束时间for(int i=1;i<=n;i++){if(t[a[i]]+d[a[i]]<ed){ok=false;break;}int st=max(ed,t[a[i]]);//开始降落时间ed=st+l[a[i]];}if(ok){cout<<"YES"<<endl;return;}}while(next_permutation(a+1,a+1+n));cout<<"NO"<<endl;
}
signed main() {ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);int t=1;cin>>t;while(t--) {solve();}return 0;
}

5.接龙数列

问最少删除几个,也就是说最多保留几个

其实可以先求最长接龙子序列的长度,再用总长度减去它

从前往后枚举每个数,当以这个数作为接龙子序列的结尾时,它前面一个数的尾必须是它的头,所以dp[尾]=max(dp[尾],dp[头]+1)

dp[x]表示x作为尾的接龙子序列的最长长度

#include<bits/stdc++.h>
#define endl '\n'
#define int long long
using namespace std;
const int N=1e5+10;
int dp[N];
int n;
void solve() {cin>>n;int ans=0;for(int i=1;i<=n;i++){string s;cin>>s;int len=s.size()-1;int x=s[0]-'0';int y=s[len-1]-'0';dp[y]=max(dp[y],dp[x]+1);ans=max(ans,dp[y]);}cout<<n-ans<<endl;
}
signed main() {ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);int t=1;
//    cin>>t;while(t--) {solve();}return 0;
}

6.岛屿个数

和外海接触的岛屿肯定不是子岛屿

先将外围一圈标记为外海,从(0,0)开始搜,将所有的外海搜出来(8方向),标记为2

然后对于那些内海,仍为0,将它们都标记为陆地

最后搜陆地一共有几个连通块即可

#include<bits/stdc++.h>
#define endl '\n'
#define int long long
using namespace std;
typedef pair<int,int>PII;
const int N=55;
char s[N][N];
int m,n;
int dx1[8]={0,0,1,-1,1,1,-1,-1},dy1[8]={1,-1,0,0,-1,1,-1,1};
int dx[4]={0,0,1,-1},dy[4]={1,-1,0,0};
void dfs1(int x,int y){//找和外海相连的海,将所有外海标记为2s[x][y]='2';for(int i=0;i<8;i++){int tx=x+dx1[i],ty=y+dy1[i];if(tx<0||tx>m+1||ty<0||ty>n+1||s[tx][ty]!='0') continue;dfs1(tx,ty);}
}
void dfs2(int x,int y){//对于找到的所有外海,去找和外海接壤的陆地s[x][y]='3';for(int i=0;i<4;i++){int tx=x+dx[i],ty=y+dy[i];if(tx<0||tx>m+1||ty<0||ty>n+1||s[tx][ty]!='1') continue;dfs2(tx,ty);}
}
void solve() {cin>>m>>n;for(int i=1;i<=m;i++){for(int j=1;j<=n;j++){cin>>s[i][j];}}//外海一圈,朝8个方向搜,搜到所有的外海for(int i=0;i<=m+1;i++){for(int j=0;j<=n+1;j++){if(s[i][j]=='1'||s[i][j]=='0') continue;s[i][j]='0';}}dfs1(0,0);for(int i=0;i<m;i++){for(int j=0;j<n;j++){if(s[i][j]=='0') s[i][j]='1';//将内海变成陆地}}int ans=0;for(int i=1;i<=m;i++){for(int j=1;j<=n;j++){if(s[i][j]=='1'){dfs2(i,j);ans++;}}}cout<<ans<<endl;
}
signed main() {ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);int t=1;cin>>t;while(t--) {solve();}return 0;
}

7.子串简写

将字符为c1的位置都存储下来,然后枚举到字符为c2时,二分到最大的满足以c2结尾,c1开头的长度大于等于k的位置,然后该位置以及前面的c1都满足,作为对答案的贡献

#include<bits/stdc++.h>
#define endl '\n'
#define int long long
using namespace std;
int k;
string s;
char c1,c2;
void solve() {cin>>k;cin>>s;cin>>c1>>c2;map<char,int>mp;vector<int>a;int ans=0;for(int i=0;i<(int)s.size();i++){if(s[i]==c1) a.push_back(i);else if(s[i]==c2){if(!a.size()) continue;int l=0,r=a.size()-1;while(l<r){int mid=(l+r+1)>>1;if(i-a[mid]+1>=k) l=mid;else r=mid-1;}if(i-a[l]+1>=k) ans+=l+1;}}cout<<ans<<endl;
}
signed main() {ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);int t=1;
//    cin>>t;while(t--) {solve();}return 0;
}

8.整数删除

线性存储结构进行删除操作时间复杂度为O(n),使用链式存储结构更适合

用双链表,这里用数组模拟

利用优先队列(小根堆),方便取出值最小的元素

这里我们删除一个元素时,需要将它左右两边的元素都加上它的值,但是我们无法在优先队列随意修改元素,所以这里有一个技巧,先记录下来下标为x的元素更新后的值,先不急着更新优先队列,类似于线段树的懒标记,等到用到了才更新优先队列,当队头的值是最新的值的时候,那么说明更新过了,那么执行删除操作,否则,需要在优先队列中更新它的值

#include<bits/stdc++.h>
#define endl '\n'
#define int long long
using namespace std;
typedef pair<int,int>PII;
const int N=5e5+10;
int a[N];
int l[N],r[N];//用数组模拟双链表
int n,k;
void del(int x){l[r[x]]=l[x],r[l[x]]=r[x];a[l[x]]+=a[x],a[r[x]]+=a[x];
}
void solve() {cin>>n>>k;for(int i=1;i<=n;i++) cin>>a[i];for(int i=1;i<=n;i++) l[i]=i-1;r[0]=1;for(int i=n;i>=1;i--) r[i]=i+1;priority_queue<PII,vector<PII>,greater<PII>>q;for(int i=1;i<=n;i++) q.push({a[i],i});while(k--){auto t=q.top();q.pop();if(t.first!=a[t.second]){q.push({a[t.second],t.second});k++;}else del(t.second);}for(int i=r[0];i<=n;i=r[i]) cout<<a[i]<<' ';cout<<endl;
}
signed main() {ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);int t=1;
//    cin>>t;while(t--) {solve();}return 0;
}

这篇关于第十四届蓝桥杯c++b组的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

C/C++错误信息处理的常见方法及函数

《C/C++错误信息处理的常见方法及函数》C/C++是两种广泛使用的编程语言,特别是在系统编程、嵌入式开发以及高性能计算领域,:本文主要介绍C/C++错误信息处理的常见方法及函数,文中通过代码介绍... 目录前言1. errno 和 perror()示例:2. strerror()示例:3. perror(

C++变换迭代器使用方法小结

《C++变换迭代器使用方法小结》本文主要介绍了C++变换迭代器使用方法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1、源码2、代码解析代码解析:transform_iterator1. transform_iterat

详解C++中类的大小决定因数

《详解C++中类的大小决定因数》类的大小受多个因素影响,主要包括成员变量、对齐方式、继承关系、虚函数表等,下面就来介绍一下,具有一定的参考价值,感兴趣的可以了解一下... 目录1. 非静态数据成员示例:2. 数据对齐(Padding)示例:3. 虚函数(vtable 指针)示例:4. 继承普通继承虚继承5.

C++中std::distance使用方法示例

《C++中std::distance使用方法示例》std::distance是C++标准库中的一个函数,用于计算两个迭代器之间的距离,本文主要介绍了C++中std::distance使用方法示例,具... 目录语法使用方式解释示例输出:其他说明:总结std::distance&n编程bsp;是 C++ 标准

C++ 中的 if-constexpr语法和作用

《C++中的if-constexpr语法和作用》if-constexpr语法是C++17引入的新语法特性,也被称为常量if表达式或静态if(staticif),:本文主要介绍C++中的if-c... 目录1 if-constexpr 语法1.1 基本语法1.2 扩展说明1.2.1 条件表达式1.2.2 fa

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++从序列容器中删除元素的四种方法

《C++从序列容器中删除元素的四种方法》删除元素的方法在序列容器和关联容器之间是非常不同的,在序列容器中,vector和string是最常用的,但这里也会介绍deque和list以供全面了解,尽管在一... 目录一、简介二、移除给定位置的元素三、移除与某个值相等的元素3.1、序列容器vector、deque

C++常见容器获取头元素的方法大全

《C++常见容器获取头元素的方法大全》在C++编程中,容器是存储和管理数据集合的重要工具,不同的容器提供了不同的接口来访问和操作其中的元素,获取容器的头元素(即第一个元素)是常见的操作之一,本文将详细... 目录一、std::vector二、std::list三、std::deque四、std::forwa

C++字符串提取和分割的多种方法

《C++字符串提取和分割的多种方法》在C++编程中,字符串处理是一个常见的任务,尤其是在需要从字符串中提取特定数据时,本文将详细探讨如何使用C++标准库中的工具来提取和分割字符串,并分析不同方法的适用... 目录1. 字符串提取的基本方法1.1 使用 std::istringstream 和 >> 操作符示

C++原地删除有序数组重复项的N种方法

《C++原地删除有序数组重复项的N种方法》给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度,不要使用额外的数组空间,你必须在原地修改输入数组并在使用O(... 目录一、问题二、问题分析三、算法实现四、问题变体:最多保留两次五、分析和代码实现5.1、问题分析5.