第十五届蓝桥杯C++B组省赛

news2024/10/25 7:18:49

在这里插入图片描述

文章目录

  • 1.握手问题
    • 解题思路1(组合数学)
    • 解题思路2(暴力枚举)
  • 2.小球反弹
    • 做题思路
  • 3.好数
    • 算法思路(暴力解法)---不会超时
  • 4.R格式
    • 算法思路
  • 5.宝石组合
    • 算法思路---唯一分解定理
  • 6.数字接龙
    • 算法思路----DFS
  • 7.拔河
    • 算法思路

1.握手问题

题目描述:
在这里插入图片描述

解题思路1(组合数学)

按照题目描述来说,会议有五十人,如果不加任何限制条件,这五十个人两两握手的次数是:
t o t a l = 49 + 48 + 47 + . . . . . . . . + 1 total=49+48+47+........+1 total=49+48+47+........+1
利用高斯求和的得出: t o t a l = 50 ∗ 49 / 2 total=50*49/2 total=5049/2
如果加上限制条件的话,题目给定的其中有七个人不会相互握手,需要用上面总的不加限制的减去七个人相互握手的次数。
c n t = 6 + 5 + . . . . . . + 1 = 7 ∗ 6 / 2 cnt=6+5+......+1=7*6/2 cnt=6+5+......+1=76/2
上述两式作差即可
编写代码:

#include<iostream>
using namespace std;
int main()
{
	int total = 50 * 49 / 2;
	int cnt = 7 * 6 / 2;
	cout << total - cnt << endl;
	return 0;
}

解题思路2(暴力枚举)

将每个人握手的情况枚举出来即可。

#include<iostream>
using namespace std;
int main()
{
	int ans = 0;
	for (int i = 1;i <= 50;i++)
	{
		for (int j = i + 1;j <= 50;j++)
		{
			//排除掉七人的情况
			if (!(i >= 1 && i <= 7 && j >= 1 && j <= 7))
			{
				ans++;
			}
		}
	}
	cout << ans << endl;
	return 0;
}

2.小球反弹

问题描述:
在这里插入图片描述

做题思路

这道题我们肯定不能直接做的,这道题给定了 d x : d y dx:dy dx:dy的值说明这道题我们应该分解来做,将小球的反弹的路径分解为x方向和y方向来做。
我们首先假设x方向上经过了p个来回,y方向上经历了q个来回,因为是分解的缘故,所以两个分解方向上的时间是相同的。
所以可以得出两个等式:
d x ∗ t = 2 p x dx*t=2px dxt=2px(由于这里一半的路程是x,所以一个来回的路程是2x,乘以来回就是总路程)
d y ∗ t = 2 q x dy*t=2qx dyt=2qx

将这两个式子进行比例
d x d y = p x q y \frac{dx}{dy}=\frac{px}{qy} dydx=qypx
得到这个式子之后我们可以利用gcd对这个式子的左边进行约分。
可以得出: p = d x ∗ y p=dx*y p=dxy q = d y ∗ x q=dy*x q=dyx
算出q或者p之后可以利用公式计算t: t = 2 p x / d x t=2px/dx t=2px/dx
最后得出总路程: 总路程 = t ∗ ( s q r t ( 1 5 2 + 1 7 2 ) ) 总路程=t*(sqrt(15^2+17^2)) 总路程=t(sqrt(152+172))

编写代码:

//求最大公约数
int gcd(int a, int b)
{
	return b == 0 ? a : gcd(b, a % b);
}
int main()
{
	//给出x方向和y方向的速度 
	int dx = 15, dy = 17;
	//给出x方向和y方向上的距离
	int x = 343720, y = 233333;
	//求出多少来回
	int q = dy * x, p = dx * y;
	//求最大公约数
	int g = gcd(p, q);
	p /= g, q /= g;

	//计算时间
	int t = 2 * p * x / dx;

	//求路程
	double ans = t * sqrt(15 * 15 + 17 * 17);
	printf("%.2lf\n", ans);
	return 0;
}

3.好数

问题描述:
在这里插入图片描述

数据量:
在这里插入图片描述

算法思路(暴力解法)—不会超时

遍历1到n的数,然后写一个check函数判断每个数是否是好数,这里的时间复杂度是 n ∗ l o g n n*logn nlogn
编写代码:

#include <iostream>
using namespace std;
int N,count;

bool Check(int n)
{
  int i=1;
  while(n!=0)
  {
    int tail=n%10;
    if(i%2==1)
    {
      if(tail%2!=1)return false;
    }
    else
    {
      if(tail%2!=0)return false;
    }
    i++;
    n/=10;
  }
  return true;
}

int main()
{
  // 请在此输入您的代码
  cin>>N;
  for(int i=1;i<N;i++)
  {
    if(Check(i))
    {
      count++;
    }
  }
  cout<<count<<endl;
  return 0;
}

4.R格式

题目描述:
在这里插入图片描述

数据量:
在这里插入图片描述

可以看到这道题的数据量是很大的,涉及到了幂次,肯定不可能直接去算,这道题很显然是考察的是高精度算法(高精度*低精度)

算法思路

高精度模版题:

编写代码:

#include<iostream>
#include<algorithm>
#include<string>
#include<cmath>
using namespace std;

//数组模拟高精度:高精度*低精度
const int N = 2e3 + 10;
string s;
int a[N];
int main()
{
	int n;
	cin >> n >> s;
	//反转操作
	reverse(s.begin(), s.end());

	//确定小数点的位置
	int pos = s.find(".");
	s.erase(pos, 1);//删除小数点,方便后续计算
	int len = s.size();
	for (int i = 0;i < len;i++)  a[i + 1] = s[i] - '0';

	//高精度*低精度
	for (int i = 1;i <= n;i++)
	{
		//顺序扫描,均*2
		for (int j = 1;j <= len;j++) a[j] *= 2;
		//处理大于10的位数
		for (int j = 1;j <= len;j++)
		{
			if (a[j] >= 10)
			{
				a[j + 1]++;
				a[j] %= 10;
				if (j == len) len++;
			}
		}
	}

	//处理小数点后的第一位,进行四舍五入

	if (a[pos] >= 5)a[pos + 1]++;

	for (int i = pos + 1;i <= len;i++)
	{
		if (a[i] >= 10)
		{
			a[i + 1]++;
			a[i] %= 10;
			if (i == len)len++;
		}
	}
	
	//打印的时候倒序打印
	for (int i = len;i >= pos + 1;i--) cout << a[i];
	return 0;
}

5.宝石组合

题目描述:

在这里插入图片描述

数据范围:
在这里插入图片描述

首先从数据量来看这道题是不能用暴力枚举的,因为暴力枚举三个数的时间复杂度是 O ( N 3 ) O(N^3) O(N3)了。

算法思路—唯一分解定理

首先我们要知道什么是唯一分解定理,简单来说唯一分解定理就是,任意一个大于1的正整数 ,都可以唯一地表示为若干个质数的乘积,且这些质数的顺序不影响分解的唯一性。
那么可以得出:
N 1 = p 1 a 1 ⋅ p 2 a 2 ⋅ … ⋅ p n a n N_1 = p_1^{a_1} \cdot p_2^{a_2} \cdot \ldots \cdot p_n^{a_n} N1=p1a1p2a2pnan

N 2 = p 1 b 1 ⋅ p 2 b 2 ⋅ … ⋅ p n b n N_2 = p_1^{b_1} \cdot p_2^{b_2} \cdot \ldots \cdot p_n^{b_n} N2=p1b1p2b2pnbn

从上面两个式子可以得出:
gcd ⁡ ( N 1 , N 2 ) = p 1 min ⁡ ( a 1 , b 1 ) ⋅ p 2 min ⁡ ( a 2 , b 2 ) ⋅ … ⋅ p n min ⁡ ( a n , b n ) \gcd(N_1,N_2) = p_1^{\min(a_1,b_1)} \cdot p_2^{\min(a_2,b_2)} \cdot \ldots \cdot p_n^{\min(a_n,b_n)} gcd(N1,N2)=p1min(a1,b1)p2min(a2,b2)pnmin(an,bn)

lcm ⁡ ( N 1 , N 2 ) = p 1 max ⁡ ( a 1 , b 1 ) ⋅ p 2 max ⁡ ( a 2 , b 2 ) ⋅ … ⋅ p n max ⁡ ( a n , b n ) \operatorname{lcm}(N_1,N_2) = p_1^{\max(a_1,b_1)} \cdot p_2^{\max(a_2,b_2)} \cdot \ldots \cdot p_n^{\max(a_n,b_n)} lcm(N1,N2)=p1max(a1,b1)p2max(a2,b2)pnmax(an,bn)

假设Ha,Hb,Hc的分解出来的相同的质数的幂次分别是x,y,z那么可以得出:

在这里插入图片描述

上面的式子可以转换为幂次是:

x + y + z + max ⁡ ( x , y , z ) − max ⁡ ( x , y ) − max ⁡ ( x , z ) − max ⁡ ( y , z ) x+y+z+\max(x,y,z)-\max(x,y)-\max(x,z)-\max(y,z) x+y+z+max(x,y,z)max(x,y)max(x,z)max(y,z)
相当于我们只需要求出上面式子的最大值即可。

编写代码:

#include<iostream>
#include<vector>
#include<algorithm>
using namespace std;
const int N = 1e5 + 10;
int a[N];
//fac是存储因子的二维数组,s是求的最大值
vector<int> fac[N], s[N];
int main()
{
	//遍历数组
	for (int i = 1;i <= 1e5;i++)
	{
		for (int j = i;j <= 1e5;j += i)
		{
			//i是j的因子
			fac[j].push_back(i);
		}
	}
	//输入n个数
	int n;cin >> n;
	for (int i = 1;i <= n;i++)cin >> a[i];
	//保证字典序最小
	sort(a + 1, a + n + 1);

	for (int i = 1;i <= n;i++)
	{
		//处理i的每个因子
		for (int j = 0;j < fac[a[i]].size();j++)
		{
			//
			s[fac[a[i]][j]].push_back(a[i]);
		}
	}
	for (int i = 1e5;i >= 0;i--)
	{
		if (s[i].size() >= 3)
		{
			cout << s[i][0] << ' ' << s[i][1] << ' ' << s[i][2] << endl;
			break;
		}
	}
	return 0;
}

6.数字接龙

问题描述:

在这里插入图片描述
在这里插入图片描述
数据量:
在这里插入图片描述
根据数据量来看这道题考察的应该是DFS,但是在DFS中应该还涉及到回溯,因为当走到不满足条件的时候需要进行回溯。

算法思路----DFS

编写代码:

#include<iostream>
#include<string>
using namespace std;
const int N = 20;
int a[N][N];
bool vis[N][N];
int n, k;
//方向数组:   0  1 2 3 4 5 6  7
int dx[8] = { -1,-1,0,1,1,1,0,-1 };
int dy[8] = { 0,1,1,1,0,-1,-1,-1 };
string res;

void dfs(int x, int y, int prev, string s, int dep)
{
	//当搜到终点的时候,且搜索深度是n的时候,意思就是每种情况都搜索完了
	if (x == n && y == n && dep == n * n) {
		if (res.empty())res = s;
		return;
	}
	for (int i = 0;i < 8;i++)
	{
		//生成邻接点
		int bx = x + dx[i];
		int by = y + dy[i];
		//过滤越界节点
		if (bx<1 || bx>n || by<1 || by>n)continue;
		//过滤访问过的节点
		if (vis[bx][by] == true)continue;
		//防止交叉搜索
		if (i == 1 && vis[x - 1][y] && vis[x][y + 1])continue;
		if (i == 3 && vis[x + 1][y] && vis[x][y + 1])continue;
		if (i == 5 && vis[x + 1][y] && vis[x][y - 1])continue;
		if (i == 7 && vis[x - 1][y] && vis[x][y - 1])continue;
		//保证路径数值为0 1 2 3 .....k-1
		if ((a[bx][by] < k && a[bx][by] == prev + 1) || (prev + 1 == k && a[bx][by] == 0))
		{
			//可以搜索,将点标记为true
			vis[bx][by] = true;
			dfs(bx, by, a[bx][by], s + to_string(i), dep + 1);
			//最优性剪枝
			if (!res.empty())return;
			vis[bx][by] = false;//回溯
		}

	}
}

int main()
{
	cin >> n >> k;
	for (int i = 1;i <= n;i++)
		for (int j = 1;j <= n;j++)
			cin >> a[i][j];
	string emp;
	//标记起点
	vis[1][1] = true;
	dfs(1, 1, 0, emp, 1);
	if (res.empty())cout << -1;
	else cout << res << endl;
	return 0;
}

7.拔河

问题描述:
在这里插入图片描述
数据量:
在这里插入图片描述
对于这种涉及到区间和的题来说,大概率都是用前缀和算法解决

算法思路

编写代码:

#include<iostream>
#include<set>
#include<climits>
using namespace std;

#define ll long long

const int N = 1e3 + 10;
int a[N], s[N];//前缀和和数组
multiset<int> ms;


int main()
{
	int n;cin >> n;
	for (int i = 1;i <= n;i++)
	{
		cin >> a[i];
		//前缀和
		s[i] = s[i - 1] + a[i];
	}

	//用set去维护所有的区间和
	for (int i = 1;i <= n;i++)
	{
		for (int j = 1;j <= n;j++)
		{
			//维护区间和
			ms.insert(s[j] - s[i - 1]);
		}
	}

	int ans = LONG_MAX;
	for (int i = 1;i <= n;i++)
	{
		for (int j = 1;j < i;j++)
		{
			//枚举以i结尾的区间,因为这里i-1只会有一个人,所以应该是j-1
			int sum = s[i] - s[j - 1];
			//找到与该区间和sum相似的区间和
			auto it = ms.lower_bound(sum);
			if (it != ms.end())
			{
				ans = min(ans, abs(*it - sum));
			}
			if (it != ms.begin())
			{
				it--;
				ans = min(ans, abs(*it - sum));
			}
		}
		//删除以i开头且以j结尾的区间,防止后续查询区间的时候出现区间重叠/交叉问题
		for (int j = i;j <= n;j++) ms.erase(ms.find(s[j] - s[i - 1]));
	}
	cout << ans << endl;
	return 0;
}

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

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

相关文章

【Oracle数据库进阶】001.SQL基础查询_查询语句

课 程 推 荐我 的 个 人 主 页&#xff1a;&#x1f449;&#x1f449; 失心疯的个人主页 &#x1f448;&#x1f448;入 门 教 程 推 荐 &#xff1a;&#x1f449;&#x1f449; Python零基础入门教程合集 &#x1f448;&#x1f448;虚 拟 环 境 搭 建 &#xff1a;&#x1…

Egg考古系列-EggCore的生命周期

关于EGG egg框架的第一个版本还是2017-03-21&#xff0c;距今已有7年了。虽然最近几年没有什么更新&#xff0c;但它在国内的使用还是挺多的&#xff0c;mvc的分层模式也很受大家喜欢。虽然声称是面向企业级、中大型项目场景的框架&#xff0c;但这种约定式在大型项目中其实也很…

高校学科竞赛管理:SpringBoot实现的高效策略

2相关技术 2.1 MYSQL数据库 MySQL是一个真正的多用户、多线程SQL数据库服务器。 是基于SQL的客户/服务器模式的关系数据库管理系统&#xff0c;它的有点有有功能强大、使用简单、管理方便、安全可靠性高、运行速度快、多线程、跨平台性、完全网络化、稳定性等&#xff0c;非常…

【M2-Mixer】核心方法解读

abstract&#xff1a; 在本文中&#xff0c;我们提出了M2-Mixer&#xff0c;这是一种基于MLPMixer的结构&#xff0c;具有多头损失&#xff0c;用于多模态分类。它比基于卷积、循环或神经结构搜索的基线模型具有更好的性能&#xff0c;其主要优势是概念和计算简单。所提出的多…

【电子电力】LCL滤波器设计,包括电流控制调谐

摘要 LCL 滤波器是电力电子领域中广泛应用于并网逆变器的滤波器之一&#xff0c;其主要功能是减少高频开关的谐波&#xff0c;确保输出电流的质量。本文设计并实现了基于 MATLAB 的 LCL 滤波器模型&#xff0c;结合电流控制器和调谐技术&#xff0c;验证了其在谐波抑制方面的效…

从RNN讲起(RNN、LSTM、GRU、BiGRU)——序列数据处理网络

文章目录 RNN&#xff08;Recurrent Neural Network&#xff0c;循环神经网络&#xff09;1. 什么是RNN&#xff1f;2. 经典RNN的结构3. RNN的主要特点4. RNN存在问题——长期依赖&#xff08;Long-TermDependencies&#xff09;问题 LSTM&#xff08;Long Short-Term Memory&a…

PostgreSQL学习笔记七:常规SQL操作

PostgreSQL 支持标准的 SQL 语句&#xff0c;同时也扩展了一些特有的功能。以下是一些常规的 SQL 语句示例&#xff0c;这些示例涵盖了数据定义、数据操作和数据查询的基本操作&#xff1a; 数据定义语言 (DDL 创建数据库&#xff1a; CREATE DATABASE mydatabase;创建表&#…

stm32单片机个人学习笔记9(TIM输入捕获)

前言 本篇文章属于stm32单片机&#xff08;以下简称单片机&#xff09;的学习笔记&#xff0c;来源于B站教学视频。下面是这位up主的视频链接。本文为个人学习笔记&#xff0c;只能做参考&#xff0c;细节方面建议观看视频&#xff0c;肯定受益匪浅。 STM32入门教程-2023版 细…

AWD入门

一、简介 AWD(Attack With Defense&#xff0c;攻防兼备)模式。你需要在一场比赛里要扮演攻击方和防守方&#xff0c;攻者得分&#xff0c;失守者会被扣分。也就是说攻击别人的靶机可以获取 Flag 分数时&#xff0c;别人会被扣分&#xff0c;同时你也要保护自己的主机不被别人…

Docker 教程四 (Docker 镜像加速)

Docker 镜像加速 国内从 DockerHub 拉取镜像有时会遇到困难&#xff0c;此时可以配置镜像加速器。 目前国内 Docker 镜像源出现了一些问题&#xff0c;基本不能用了&#xff0c;后期能用我再更新下。* Docker 官方和国内很多云服务商都提供了国内加速器服务&#xff0c;例如…

Python网络爬虫入门指南

✅作者简介&#xff1a;2022年博客新星 第八。热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏…

MPA-SVM多变量回归预测|海洋捕食者优化算法-支持向量机|Matalb

目录 一、程序及算法内容介绍&#xff1a; 基本内容&#xff1a; 亮点与优势&#xff1a; 二、实际运行效果&#xff1a; 三、算法介绍&#xff1a; 四、完整程序下载&#xff1a; 一、程序及算法内容介绍&#xff1a; 基本内容&#xff1a; 本代码基于Matlab平台编译&am…

2024年网络安全进阶学习路径-2024年进阶学习指南

&#x1f91f; 基于入门网络安全/黑客打造的&#xff1a;&#x1f449;黑客&网络安全入门&进阶学习资源包 前言 什么是网络安全 网络安全可以基于攻击和防御视角来分类&#xff0c;我们经常听到的 “红队”、“渗透测试” 等就是研究攻击技术&#xff0c;而“蓝队”、…

KDD 2024论文分享┆用于序列推荐的数据集再生

论文简介 本推文介绍了2024 KDD的最佳学生论文《Dataset Regeneration for Sequential Recommendation》。该论文提出了一种基于数据中心化范式的新框架&#xff0c;称为DR4SR&#xff0c;该框架通过模型无关的数据再生机制&#xff0c;能够生成具有出色跨架构泛化能力的理想训…

git(版本回退,分支管理,vscode集成git)

一、安装与简单命令 1.官网 https://git-scm.com/downloads 2.查看版本号git --version 3.设置用户签名&#xff08;用户名和邮箱&#xff09; 用来标识用户&#xff0c;以区分不同的开发人员 git config --global user.name "Your Name" git config --global u…

2024年最新算法:青蒿素优化算法(Artemisinin Optimization Algorithm, AOA)原理介绍

青蒿素优化算法&#xff08;Artemisinin Optimization Algorithm, AOA&#xff09;是2024年提出的一种受青蒿素抗疟疾特性启发的元启发式优化算法。青蒿素是一种从中草药青蒿中提取的化合物&#xff0c;因其在治疗疟疾方面的显著效果而闻名。AOA算法的设计者将青蒿素的这一特性…

【机器学习】深入浅出讲解贝叶斯分类算法

0. 前言 1.贝叶斯分类器介绍 贝叶斯分类是一类分类算法的总称&#xff0c;这类算法均以贝叶斯定理为基础&#xff0c;故统称为贝叶斯分类。而朴素贝叶斯&#xff08;Naive Bayes&#xff09;分类是贝叶斯分类中最简单&#xff0c;也是常见的一种分类方法。 一些很常见的分类…

动态规划最大子段和讲解和【题解】——最大子段和

动态规划最大子段和讲解和【题解】——最大子段和 1.详细讲解最大子段和题目描述输入格式输出格式输入输出样例输入 #1输出 #1 提示样例 1 解释数据规模与约定 1.1.思路解析1.2.AC代码 2.优化3.别言 1.详细讲解 最大子段和 题目描述 给出一个长度为 n n n 的序列 a a a&am…

cursor: mutex X 等待事件分析

背景&#xff1a; v$session中同一个sql语句bhaku1zp2w5v7大量等待cursor: mutex X &#xff0c;且等待事件较长。 分析&#xff1a; 什么是cursor: mutex X&#xff1f; 任何操作或访问游标的操作都可能需要等待访问共享池中支持游标的结构。在极端争用的情况下&#xff0c…

MySQL 【数字】函数大全(一)

ABSCEILCEILINGCONVDIVFLOORCREATESTLEAST 1、ABS ABS(number) &#xff1a;返回指定数字的绝对值 如果参数 number 为字符串&#xff0c;ABS() 将按照如下规则尝试转为数字&#xff1a; 如果以数字开头&#xff0c;则将开头的数字部分转为数字。如果不能转为数字&#xff0c;…