第十六届蓝桥杯省赛(软件类真题)C/C++ 大学A组

第十六届蓝桥杯省赛(软件类真题)C/C++ 大学A组

大纲:

A.寻找质数
B:黑白棋

题目&解析&代码

A题

在这里插入图片描述

题目解析

本题的目标是枚举质数并计数,直到数到第2025个。由于2025不算太大,第2025个质数大约在17000~18000之间,完全可以在合理时间内通过简单枚举得到。

解题步骤:

从2开始遍历每个整数,判断它是否是质数。

质数判断采用试除法:对于一个数n,只需检查从2到√n的所有整数是否能整除n。若存在能整除的数,则n不是质数;否则是质数。

每找到一个质数,计数器加1。

当计数器达到2025时,输出当前的质数并结束。

优化点:

除了2以外,偶数不可能是质数,因此可以跳过偶数判断(直接步进2)。

在isPrime函数中,可以先处理特殊情况(n<2返回false),然后单独判断偶数,再对奇数进行试除,步进也可以设为2。

C++ 参考代码
以下代码实现了上述算法,并输出第2025个质数。

cpp
#include
#include
using namespace std;

// 判断一个整数是否为质数
bool isPrime(int n) {
if (n < 2) return false;
if (n == 2) return true;
if (n % 2 == 0) return false; // 偶数直接排除
int limit = sqrt(n); // 只需检查到平方根
for (int i = 3; i <= limit; i += 2) { // 只检查奇数因子
if (n % i == 0) return false;
}
return true;
}

int main() {
int count = 0; // 已找到的质数个数
int num = 2; // 从2开始检查
while (true) {
if (isPrime(num)) {
count++;
if (count == 2025) {
cout << num << endl;
break;
}
}
num++;
}
return 0;
}
代码说明:

isPrime函数做了常规的优化:先排除2以外的偶数,然后只检查奇数因子到平方根。

main函数中循环递增num,每遇到一个质数计数器加1,直到第2025个时输出该数。

运行结果
在本地运行上述代码,最终输出的数字即为第2025个质数。根据已知质数表和实际计算,该数为:

17627

因此,本题的答案是 17627。

B题

在这里插入图片描述


在这里插入图片描述


在这里插入图片描述


🧩 题目回顾与核心规则
我们首先要明确游戏的胜利条件。在一个 6x6 的棋盘上,已有部分格子填有黑色 (1) 或白色 (0) 棋子,需要填满所有空格,并满足以下规则:

数量相等:每一行和每一列中,黑棋和白棋的数量必须相等(即各有3个)。

无三连:在任何一行或一列中,不能有超过两个相同颜色的棋子连续排列(即禁止出现“111”或“000”)。

行列唯一:每一行的排列方式必须是唯一的(不能与其他任何行相同);每一列的排列方式也必须是唯一的(不能与其他任何列相同)。但行与列之间可以相同。

题目给定的初始棋盘状态(基于选手回忆和题解推导)如下所示,其中 -1 代表空格:

cpp
int grid[6][6] = {
{ 1, 0, 1, 0,-1,-1},
{-1,-1,-1, 0,-1,-1},
{-1,-1,-1, 1, 0, 0},
{-1,-1,-1,-1,-1,-1},
{-1,-1, 1,-1,-1, 1},
{-1, 0,-1,-1, 1,-1}
};
💡 核心解题思路:深度优先搜索(DFS) + 剪枝
程序的核心思想是深度优先搜索(DFS),逐个格子地去尝试填入黑棋(1)或白棋(0)。但6x6的棋盘总共有36个格子,如果盲目地尝试所有可能,计算量是天文数字(2^36 ≈ 687亿种可能)。因此,必须在搜索过程中进行严格的剪枝,提前排除那些不可能满足规则的分支,才能快速找到唯一解。

主要的剪枝策略有:

数量限制:在放置一个棋子前,检查其所在行和列中,该颜色棋子数量是否已达到3个。如果已达到,就不能再放。

无三连限制:在放置一个棋子后,立即检查它是否与其左边两个(或上边两个)棋子构成了连续三个相同颜色。如果构成了,这个分支就是无效的,必须回退。

行列唯一性剪枝:每当填完一整行或一整列时,立即检查该行(或列)的排列是否与之前已完成的任何一行(或列)重复。如果重复,则剪枝。

💻 完整C++源码与逐行解析
下面这个程序正是基于上述思路实现的,代码中包含了详细的注释,可以帮助你理解每一步的作用。

cpp
#include<bits/stdc++.h>
using namespace std;

// 初始化棋盘,-1表示空格。这个状态是根据题目描述和选手回忆确定的。
int grid[6][6] = {
{ 1, 0, 1, 0,-1,-1},
{-1,-1,-1, 0,-1,-1},
{-1,-1,-1, 1, 0, 0},
{-1,-1,-1,-1,-1,-1},
{-1,-1, 1,-1,-1, 1},
{-1, 0,-1,-1, 1,-1}
};

// 用于记录当前每行和每列中黑棋(1)和白棋(0)的数量
int black_row[6] = {0}, black_col[6] = {0};
int white_row[6] = {0}, white_col[6] = {0};

// 用于记录已经出现过的行排列和列排列,以实现“行列唯一性”的剪枝
unordered_set vis_row, vis_col;
string ans; // 存储最终的答案字符串

// 函数:在搜索结束时,对整个棋盘进行一次最终检查,确保所有规则都被满足
int check() {
unordered_set r, c;
// 检查所有行是否唯一
for (int i = 0; i < 6; i++) {
string rr = “”;
for (int j = 0; j < 6; j++) rr += to_string(grid[i][j]);
if (r.count(rr)) return 0; // 发现重复行,无效
r.insert(rr);
}
// 检查所有列是否唯一
for (int i = 0; i < 6; i++) {
string cc = “”;
for (int j = 0; j < 6; j++) cc += to_string(grid[j][i]);
if (c.count(cc)) return 0; // 发现重复列,无效
c.insert(cc);
}
return 1; // 所有检查通过
}

// 核心:深度优先搜索函数,pos 表示当前正在处理的格子编号(从0到35)
int solve(int pos) {
// 如果所有格子都处理完了,进行最终检查
if (pos == 36) {
if (check()) {
// 将最终的棋盘状态转换为答案字符串
ans = “”;
for (int i = 0; i < 6; i++)
for (int j = 0; j < 6; j++) ans += to_string(grid[i][j]);
return 1; // 找到答案,返回1
}
return 0;
}

int row = pos / 6; // 计算当前格子所在行 int col = pos % 6; // 计算当前格子所在列 // 如果当前格子已经有棋子(非空格),则直接跳过,处理下一个格子 if (grid[row][col] != -1) return solve(pos + 1); // 尝试在当前空格放入两种颜色的棋子:val = 0 (白棋) 或 1 (黑棋) for (int val = 0; val <= 1; val++) { // 剪枝1:检查行和列中该颜色的数量是否已达上限(3个) if (!val) { // 尝试放白棋 if (white_row[row] >= 3 || white_col[col] >= 3) continue; } else { // 尝试放黑棋 if (black_row[row] >= 3 || black_col[col] >= 3) continue; } // 剪枝2:检查是否会形成横向或纵向的三个连续相同棋子 // 横向检查:当前格子的左边两个是否和当前尝试的val相同 if (col >= 2 && grid[row][col - 2] == val && grid[row][col - 1] == val) continue; // 纵向检查:当前格子的上边两个是否和当前尝试的val相同 if (row >= 2 && grid[row - 2][col] == val && grid[row - 1][col] == val) continue; // --- 执行放置操作,并更新相关计数 --- grid[row][col] = val; if (val) { black_row[row]++; black_col[col]++; } else { white_row[row]++; white_col[col]++; } // --- 剪枝3:如果当前放完了一整行或一整列,检查其唯一性 --- int flag = 1; // 标记是否通过唯一性检查 int full_row = (col == 5); // 是否刚填完一行(当col为5时,说明当前格子是行尾) int full_col = (row == 5); // 是否刚填完一列(当row为5时,说明当前格子是列尾) string,; if (full_row) { // 生成当前行的字符串表示 for (int i = 0; i < 6; i++) rowstr += to_string(grid[row][i]); if (vis_row.count(rowstr)) flag = 0; // 如果该行已出现过,则剪枝 else vis_row.insert(rowstr); // 否则,将其加入已出现行的集合 } if (full_col) { // 生成当前列的字符串表示 for (int i = 0; i < 6; i++) colstr += to_string(grid[i][col]); if (vis_col.count(colstr)) flag = 0; // 如果该列已出现过,则剪枝 else vis_col.insert(colstr); // 否则,将其加入已出现列的集合 } // 如果通过了所有剪枝,则递归地搜索下一个格子 if (flag && solve(pos + 1)) return 1; // --- 回溯:撤销当前尝试的操作,恢复现场,以便尝试另一种颜色的棋子 --- if (full_row) vis_row.erase(rowstr); if (full_col) vis_col.erase(colstr); grid[row][col] = -1; if (val) { black_row[row]--; black_col[col]--; } else { white_row[row]--; white_col[col]--; } } return 0; // 如果当前格子尝试了两种颜色都无法得到解,则返回0 

}

int main() {
// 程序开始前,先根据初始棋盘,初始化每行每列的黑白棋数量
for (int i = 0; i < 6; i++) {
for (int j = 0; j < 6; j++) {
if (grid[i][j] == 1) {
black_row[i]++;
black_col[j]++;
} else if (grid[i][j] == 0) {
white_row[i]++;
white_col[j]++;
}
}
}
// 从第一个格子(编号0)开始深度优先搜索
if (solve(0)) {
cout << ans << endl; // 输出最终找到的答案字符串
}
return 0;
}

谢谢老铁们支持,下一期继续分享

Read more

Flutter 三方库 json_extractor 的鸿蒙化适配指南 - 支持声明式 JSON 数据提取、复杂嵌套结构解析与强类型转换

Flutter 三方库 json_extractor 的鸿蒙化适配指南 - 支持声明式 JSON 数据提取、复杂嵌套结构解析与强类型转换

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 json_extractor 的鸿蒙化适配指南 - 支持声明式 JSON 数据提取、复杂嵌套结构解析与强类型转换 前言 在 Flutter for OpenHarmony 的日常开发中,处理后端返回的“排山倒海”般的 JSON 数据是每个开发者的必经之路。虽然 json_serializable 很强大,但如果你只需要从一个极其庞大且嵌套复杂的 JSON 中提取特定的几个字段,定义完整的 Model 类就显得过于繁琐。json_extractor 提供了一种基于声明式路径的轻量级提取方案。本文将指导大家如何在鸿蒙端利用该库高效“榨取”JSON 数据。 一、原理解析 / 概念介绍 1.1 基础原理 json_

By Ne0inhk
Flutter for OpenHarmony: Flutter 三方库 shamsi_date 助力鸿蒙应用精准适配波斯历法(中东出海必备)

Flutter for OpenHarmony: Flutter 三方库 shamsi_date 助力鸿蒙应用精准适配波斯历法(中东出海必备)

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net 前言 在进行 OpenHarmony 的全球化(Internationalization)应用开发时,进军中东市场(尤其是波斯语地区)是一项充满潜力的战略。但在这些地区,用户习惯使用的并非公历(Gregorian),而是 波斯历(Shamsi/Jalali)。 1. 如何将用户的生日从公历转换成波斯历? 2. 鸿蒙应用的时间轴、日历选择器如何呈现 Jalali 格式? 3. 业务系统中的合同到期日如何按波斯历进行逻辑计算? shamsi_date 是 Dart 生态中处理波斯历法的权威库。它提供了极其简单的转换 API,是你开发鸿蒙出海应用、打入中东市场的关键技术补丁。 一、历法转换算法模型 shamsi_date 实现了公历与波斯历之间的双向精准映射。 Conversion Conversion 公历 (2024-02-20) 波斯历 (1402-12-01)

By Ne0inhk
Flutter 三方库 hooks_runner 的鸿蒙化适配指南 - 实现声明式的生命周期 Hook 任务管理、支持端侧自动化脚本触发与执行流精准编排实战

Flutter 三方库 hooks_runner 的鸿蒙化适配指南 - 实现声明式的生命周期 Hook 任务管理、支持端侧自动化脚本触发与执行流精准编排实战

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 hooks_runner 的鸿蒙化适配指南 - 实现声明式的生命周期 Hook 任务管理、支持端侧自动化脚本触发与执行流精准编排实战 前言 在进行 Flutter for OpenHarmony 的自动化工具、CI/CD 插件或具备高度动态逻辑的业务系统开发时,如何有序、可控地执行一系列相互依赖的“任务钩子(Hooks)”?hooks_runner 是一个专为任务生命周期编排设计的轻量级引擎。它能将离散的函数逻辑拆解并组装成一条健壮的执行流水线。本文将介绍如何在鸿蒙端利用该库构建极致的任务执行闭环。 一、原理解析 / 概念介绍 1.1 基础原理 hooks_runner 采用了“注册-触发(Register & Trigger)”模式。它允许开发者在不同的生命周期阶段(如 pre_

By Ne0inhk