当前位置: 首页 > news >正文

洛谷每日一题——P1036 [NOIP2002 普及组] 选数、P1045 [NOIP2003 普及组] 麦森数(高精度快速幂)

P1036 [NOIP2002 普及组] 选数

题目描述

[NOIP2002 普及组] 选数 - 洛谷

运行代码

#include <stdio.h>
int n, k, a[25], t;
int ss(int b) {int i;if (b < 2)return 0;for (i = 2; i * i <= b; i++)if (b % i == 0)return 0;return 1;
}
void dfs(int num, int sum, int j) {int i;if (num == k) {if (ss(sum))t++;return;}for (i = j; i < n; i++)dfs(num + 1, sum + a[i], i + 1);return;
}
int main() {int i;scanf("%d %d", &n, &k);for (i = 0; i < n; i++) {scanf("%d", &a[i]);}dfs(0, 0, 0);printf("%d", t);return 0;
}
改进后
  • 将判断一个数是否为质数的函数 ss 改名为 isPrime,使其功能更清晰易懂。
  • 在 dfs 函数中,将原本在全局变量中的 k 和数组 a 以及用于计数的 t 作为参数传递进去,这样可以使函数的独立性更强,不需要依赖全局变量,降低了代码的耦合度。
  • 在 isPrime 函数中,使用 sqrt(b) 来代替 i * i <= b,在判断一个数是否为质数时,只需要遍历到该数的平方根即可,这样可以稍微提高一些效率。
#include <stdio.h>
#include <math.h>// 判断一个数是否为质数
int isPrime(int b) {if (b < 2) return 0;for (int i = 2; i <= sqrt(b); i++) {if (b % i == 0) return 0;}return 1;
}// 深度优先搜索函数
void dfs(int num, int sum, int j, int k, int *a, int *t) {if (num == k) {if (isPrime(sum)) (*t)++;return;}for (int i = j; i < n; i++) {dfs(num + 1, sum + a[i], i + 1, k, a, t);}
}int main() {int n, k, a[25], t = 0;scanf("%d %d", &n, &k);for (int i = 0; i < n; i++) {scanf("%d", &a[i]);}dfs(0, 0, 0, k, a, &t);printf("%d", t);return 0;
}

代码思路

这段代码的主要目的是从给定的一组整数中选取 k 个整数,将它们相加,然后判断相加的和是否为质数,统计满足条件的组合数量。

  1. 数据读取与初始化

    • 在 main 函数中,首先通过 scanf 读取两个整数 n 和 k,其中 n 表示给定的整数数组 a 的长度,k 表示要从数组中选取的整数个数。
    • 接着通过循环使用 scanf 读取数组 a 中的每一个元素。
    • 同时初始化一个变量 t 为 0,用于统计满足条件(即选取的 k 个整数相加和为质数)的组合数量。
  2. 深度优先搜索(DFS)实现组合选取dfs 函数用于实现深度优先搜索来找出所有可能的 k 个整数的组合。它接受几个参数:

    • t:用于统计满足条件的组合数量(通过指针传递,以便在函数内部修改其值)。
    • a:给定的整数数组(从 main 函数传递过来)。
    • k:要选取的整数个数(从 main 函数传递过来)。
    • j:表示下一个可供选取的整数在数组 a 中的索引。
    • sum:表示当前选取的整数相加的和。
    • num:表示当前已经选取的整数个数。
    • 在 dfs 函数内部:
      • 当 num 等于 k 时,说明已经选取了 k 个整数,此时调用 isPrime 函数判断 sum 是否为质数,如果是,则将 t 的值加 1,然后返回。
      • 如果 num 不等于 k,则通过循环从索引 j 开始遍历数组 a,对于每一个元素 a[i],递归调用 dfs 函数,将 num 加 1(表示又选取了一个整数),sum 加 a[i](更新选取的整数相加的和),i + 1(更新下一个可供选取的整数的索引)。
  3. 判断质数函数

    • isPrime 函数用于判断一个数是否为质数。它接受一个整数 b 作为参数。
    • 如果 b 小于 2,则直接返回 0,因为小于 2 的数不是质数。
    • 然后通过循环从 2 开始遍历到 sqrt(b),如果在这个范围内发现 b 能被某个数整除(即 b % i == 0),则返回 0,说明 b 不是质数;如果遍历完整个范围都没有发现这样的数,则返回 1,说明 b 是质数。
  4. 结果输出:最后,在 main 函数中,通过 printf 输出统计得到的满足条件的组合数量 t

综上所述,该代码通过深度优先搜索遍历所有可能的 k 个整数的组合,然后判断每个组合的和是否为质数,从而统计出满足条件的组合数量。

P1045 [NOIP2003 普及组] 麦森数(高精度快速幂)

题目描述

[NOIP2003 普及组] 麦森数 - 洛谷

运行代码

#include <algorithm>
#include <cmath>
#include <iostream>
#include <string.h>
#include <vector>
using namespace std;
const int N = 500;
typedef vector<int> VI;
VI a(N), res(N);
int p;
VI mul(VI& a, VI& b) {VI t(N * 2);for (int i = 0; i < N; i++)for (int j = 0; j < N; j++) {t[i + j] += a[i] * b[j];t[i + j + 1] += t[i + j] / 10;t[i + j] %= 10;}return t;
}
void quick_pow(int p) {res[0] = 1, a[0] = 2;while (p) {if (p & 1)res = mul(res, a);a = mul(a, a);p >>= 1;}res[0]--;
}
int main() {cin >> p;printf("%d\n", int(p * log10(2)) + 1);quick_pow(p);for (int i = 0, k = 499; i < 10; i++) {for (int j = 0; j < 50; j++, k--)printf("%d", res[k]);puts(" ");}return 0;
}
改进后
  • 在 mul 函数中,将结果向量 t 的初始化大小改为根据输入向量 a 和 b 的大小动态确定,更加灵活且避免了可能的空间浪费。同时,在函数结尾添加了去除前导 0 的操作,使结果更加规范。
  • 在 quick_pow 函数中,将 res 和 a 的初始化放在函数内部,使函数更加独立,不需要依赖全局变量。并且将 res 作为参数传入,这样可以在函数内部直接修改其值,而不是像原来那样通过全局变量来操作。
  • 在 main 函数中,使用 cout 代替 printf,使代码风格更加统一(因为前面已经使用了 iostream 库)。同时,在输出结果时,对超出向量范围的情况进行了处理,即当索引小于 0 时输出 0,保证了输出的完整性。
#include <iostream>
#include <vector>
#include <cmath>
using namespace std;// 高精度乘法函数,计算两个大整数向量的乘积
vector<int> mul(const vector<int>& a, const vector<int>& b) {vector<int> t(a.size() + b.size(), 0);for (size_t i = 0; i < a.size(); ++i) {for (size_t j = 0; j < b.size(); ++j) {t[i + j] += a[i] * b[j];t[i + j + 1] += t[i + j] / 10;t[i + j] %= 10;}}// 去除前导0while (t.size() > 1 && t.back() == 0) t.pop_back();return t;
}// 快速幂函数,用于计算2的p次方
void quick_pow(int p, vector<int>& res) {res = {1};vector<int> a = {2};while (p) {if (p & 1) res = mul(res, a);a = mul(a, a);p >>= 1;}res[0]--;
}int main() {int p;cin >> p;// 先输出结果的位数cout << static_cast<int>(p * log10(2)) + 1 << endl;vector<int> res;quick_pow(p, res);// 输出结果,每50个数字为一行for (int i = 0, k = res.size() - 1; i < 10; ++i) {for (int j = 0; j < 50; ++j, k--) {if (k >= 0) cout << res[k];else cout << 0;}cout << endl;}return 0;
}

代码思路

这段代码主要实现了两个功能:一是计算并输出 2 的 p 次方减 1 的结果(以高精度整数的形式),二是先输出这个结果的位数。

  1. 高精度乘法函数 mul

    • 目的是实现两个大整数(以 vector<int> 形式存储,每个元素代表整数的一位)的乘法运算。
    • 首先创建一个大小为 a.size() + b.size() 的结果向量 t,初始值都为 0
    • 然后通过两层嵌套的循环遍历两个输入向量 a 和 b 的每一位。对于每一对位 a[i] 和 b[j],将它们的乘积加到 t[i + j] 上。接着处理进位,将 t[i + j] 除以 10 的商加到 t[i + j + 1] 上,同时将 t[i + j] 除以 10 的余数保留在 t[i + j] 上。
    • 最后,通过循环去除结果向量 t 中的前导 0,得到规范的乘法结果并返回。
  2. 快速幂函数 quick_pow

    • 基于快速幂算法来计算 2 的 p 次方。快速幂算法的基本思想是通过不断地将指数减半,同时根据指数的奇偶性来决定是否将当前的中间结果与底数相乘,从而减少乘法运算的次数,提高计算效率。
    • 首先初始化 res 为只包含一个元素 1 的向量,表示 2 的 0 次方结果;初始化 a 为只包含一个元素 2 的向量,表示底数。
    • 然后在循环中,当 p 不为 0 时:
      • 如果 p 是奇数(即 p & 1 为真),则将当前的中间结果 res 与底数 a 相乘,更新 res 的值。
      • 然后将底数 a 自身相乘,更新 a 的值。
      • 最后将 p 除以 2(通过 p >>= 1 实现),继续下一轮循环。
    • 循环结束后,将 res 的第一个元素减 1,得到 2 的 p 次方减 1 的结果。
  3. 主函数 main

    • 首先通过 cin 读取输入的整数 p
    • 接着计算并输出 2 的 p 次方减 1 的结果的位数。根据对数的性质,一个数 N 的位数可以通过 log10(N) 来估算,对于 2 的 p 次方,其位数大约为 p * log10(2),再加上 1 是因为可能存在进位情况,所以通过 cout 输出 int(p * log10(2)) + 1
    • 然后调用 quick_pow 函数计算 2 的 p 次方减 1 的结果,并将结果存储在 res 向量中。
    • 最后,通过两层嵌套的循环将 res 中的结果以每 50 个数字为一行的方式输出。在循环中,先从 res 的末尾开始向前遍历,对于每一行,输出 50 个数字,如果遇到索引小于 0 的情况(即已经遍历完所有数字),则输出 0,保证每行输出的数字数量固定为 50 个。

综上所述,该代码通过高精度乘法和快速幂算法实现了对 2 的 p 次方减 1 的计算和输出,同时也给出了结果的位数估算。

相关文章:

洛谷每日一题——P1036 [NOIP2002 普及组] 选数、P1045 [NOIP2003 普及组] 麦森数(高精度快速幂)

P1036 [NOIP2002 普及组] 选数 题目描述 [NOIP2002 普及组] 选数 - 洛谷 运行代码 #include <stdio.h> int n, k, a[25], t; int ss(int b) {int i;if (b < 2)return 0;for (i 2; i * i < b; i)if (b % i 0)return 0;return 1; } void dfs(int num, int sum, …...

OpenHarmony开源鸿蒙

OpenHarmony_百度百科 2024年4 月 1 日&#xff0c;开源鸿蒙 OpenHarmony 4.1 Release 版本于昨日发布&#xff0c;开发套件同步升级到 API 11 Release...

2024.11.4 STM32点灯和简单的数据收发

1.发送函数 HAL_StatusTypeDef HAL_UART_Transmit(UART_HandleTypeDef *huart, uint8_t *pData, uint16_t Size, uint32_t Timeout); 参数1&#xff1a; UART 处理结构体的指针&#xff0c;该结构体包含了 UART 的所有配置参数。 参数2&#xff1a;要发送的数据指针 参数3&…...

Android Studio jcenter 停止服务,改用mavenCentral

随着jcenter在2021年2月28日停止服务&#xff0c;Android和Java开发者需寻找替代方案。推荐使用MavenCentral&#xff0c;可借助国内镜像加速。此外&#xff0c;jitpack.io也是一个选项&#xff0c;但对于大型项目&#xff0c;自建Nexus或MavenCentral更合适。迁移步骤包括更新…...

EasyPOI使用详解

EasyPOI 简介 easypoi功能如同名字easy,主打的功能就是容易,让一个没见接触过poi的人员 就可以方便的写出Excel导出,Excel模板导出,Excel导入,Word模板导出,通过简单的注解和模板 语言(熟悉的表达式语法),完成以前复杂的写法 文档&#xff1a;http://easypoi.mydoc.io/#categor…...

【云原生开发】K8S多集群资源管理平台架构设计

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…...

基于SpringBoot的城镇住房保障系统开发

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

一文解秘Rust如何与Java互操作

本博客所有文章除特别声明外&#xff0c;均采用CC BY-NC-SA 4.0许可协议。转载请注明来自 唯你 使用场景 JAVA 与 Rust 互操作让 Rust 可以背靠 Java 大生态来做更多事情&#xff0c;而 Java 也可以享受 Rust 语言特性的内存安全&#xff0c;所有权机制&#xff0c;无畏并发。…...

手机发展史介绍

手机&#xff0c;这个曾经在电影和科幻小说中出现的高科技产品&#xff0c;如今已经渗透进了我们生活的每个角落。从单纯的通讯工具到如今集成了通讯、娱乐、工作、社交等多种功能的智能终端&#xff0c;手机的发展史也是人类科技进步的缩影。本文将从手机的发展历程、技术革新…...

【ArcGISPro】单次将自己建立的工具箱添加至Arcpy中

新建工具箱 添加至Arcpy中 调用刚添加的工具箱...

docker镜像仓库常用命令

docker镜像仓库常用命令 docker logindocker logoutdocker pulldocker pushdocker searchdocker imagesdocker image inspectdocker tagdocker rmidocker image prunedocker savedocker loaddocker history docker login 语法: docker login [options] [server] 功能&#xff…...

springboot 传统应用程序,适配云原生改造

概述 2024年传统应用程序上云&#xff0c;改造方案 1、mysql 云环境高可用方案 2、redis 云环境高可用方案 3、nginx 云环境高可用方案 4、应用 云环境高可用方案1、mysql 云环境高可用方案 1.1 你先了解 1.1.1 你先了解“mysql高可用方案” 主从复制&#xff08;Master-S…...

D61【python 接口自动化学习】- python基础之数据库

day61 数据库定义 学习日期&#xff1a;20241107 学习目标&#xff1a;MySQL数据库-- 130&#xff1a;MySQL入门使用 学习笔记&#xff1a; 在命令提示符内先试用MySQL 使用图形化工具操作MySQL DBeaver安装 DBeaver连接MySQL 总结 MySQL安装成功后&#xff0c;可以使用命…...

数据库期末考试简答题

1&#xff0e;试述数据、数据库、数据库管理系统、数据库系统的概念。 答&#xff1a;&#xff08;1&#xff09;数据是数据库中存储的基本对象&#xff0c;是描述事物的符号记录。数据有多种表现形式&#xff0c;它们都可以经过数字化后存入计算机。数据的种类有数字、文字、…...

Java[面试题]-真实面试

1.什么是IOC和AOP&#xff1f;了解么&#xff1f; IOC&#xff08;控制反转&#xff09;和AOP&#xff08;面向切面编程&#xff09; 1. IOC&#xff08;控制反转&#xff09; 概念 IOC&#xff08;Inversion of Control&#xff09;是面向对象编程中的一个设计原则&#xf…...

HTML5新增多媒体支持

一、引言 在当今数字化时代&#xff0c;丰富的多媒体内容对于网页的吸引力和用户体验至关重要。HTML5 的出现为网页带来了强大的多媒体支持&#xff0c;尤其是在音频和视频方面&#xff0c;为开发者和用户带来了全新的可能性。 二、音频audio标签 2.1 定义与属性详解 <a…...

K8S群集调度二

一、污点(Taint) 和 容忍(Tolerations) 1.1、污点(Taint) 设置在node上是对pod的一种作用 节点的亲和性&#xff0c;是Pod的一种属性&#xff08;偏好或硬性要求&#xff09;&#xff0c;它使Pod被吸引到一类特定的节点 而Taint 则相反&#xff0c;它使节点能够排斥一类特…...

43.第二阶段x86游戏实战2-提取游戏里面的lua

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 本次游戏没法给 内容参考于&#xff1a;微尘网络安全 本人写的内容纯属胡编乱造&#xff0c;全都是合成造假&#xff0c;仅仅只是为了娱乐&#xff0c;请不要…...

debian系统安装qt的时候 显示xcb相关文件缺失

如果是安装之后的问题 我们可以选择使用ldd的命令查看当前依赖的so那些文件确实 ldd /home/yinsir/Qt/5.15.2/gcc_64/plugins/platforms/libqxcb.so 本人在进行打包的时候 出现则会个报错 ERROR: ldd outputLine: “libxcb-util.so.1 > not found” ERROR: for binary: “/…...

得物多模态大模型在重复商品识别上的应用和架构演进

重复商品治理介绍 根据得物的平台特性&#xff0c;同一个商品在平台上不能出现多个链接&#xff0c;原因是平台需要保证一品一链的特点&#xff0c;以保障商品的集中竞价&#xff0c;所以说一个商品在整个得物平台上只能有一个商详链接&#xff0c;因此我们需要对一品多链的情…...

第19节 Node.js Express 框架

Express 是一个为Node.js设计的web开发框架&#xff0c;它基于nodejs平台。 Express 简介 Express是一个简洁而灵活的node.js Web应用框架, 提供了一系列强大特性帮助你创建各种Web应用&#xff0c;和丰富的HTTP工具。 使用Express可以快速地搭建一个完整功能的网站。 Expre…...

Android Wi-Fi 连接失败日志分析

1. Android wifi 关键日志总结 (1) Wi-Fi 断开 (CTRL-EVENT-DISCONNECTED reason3) 日志相关部分&#xff1a; 06-05 10:48:40.987 943 943 I wpa_supplicant: wlan0: CTRL-EVENT-DISCONNECTED bssid44:9b:c1:57:a8:90 reason3 locally_generated1解析&#xff1a; CTR…...

12.找到字符串中所有字母异位词

&#x1f9e0; 题目解析 题目描述&#xff1a; 给定两个字符串 s 和 p&#xff0c;找出 s 中所有 p 的字母异位词的起始索引。 返回的答案以数组形式表示。 字母异位词定义&#xff1a; 若两个字符串包含的字符种类和出现次数完全相同&#xff0c;顺序无所谓&#xff0c;则互为…...

Redis数据倾斜问题解决

Redis 数据倾斜问题解析与解决方案 什么是 Redis 数据倾斜 Redis 数据倾斜指的是在 Redis 集群中&#xff0c;部分节点存储的数据量或访问量远高于其他节点&#xff0c;导致这些节点负载过高&#xff0c;影响整体性能。 数据倾斜的主要表现 部分节点内存使用率远高于其他节…...

C++使用 new 来创建动态数组

问题&#xff1a; 不能使用变量定义数组大小 原因&#xff1a; 这是因为数组在内存中是连续存储的&#xff0c;编译器需要在编译阶段就确定数组的大小&#xff0c;以便正确地分配内存空间。如果允许使用变量来定义数组的大小&#xff0c;那么编译器就无法在编译时确定数组的大…...

基于Java+MySQL实现(GUI)客户管理系统

客户资料管理系统的设计与实现 第一章 需求分析 1.1 需求总体介绍 本项目为了方便维护客户信息为了方便维护客户信息&#xff0c;对客户进行统一管理&#xff0c;可以把所有客户信息录入系统&#xff0c;进行维护和统计功能。可通过文件的方式保存相关录入数据&#xff0c;对…...

Python基于历史模拟方法实现投资组合风险管理的VaR与ES模型项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档&#xff09;&#xff0c;如需数据代码文档可以直接到文章最后关注获取。 1.项目背景 在金融市场日益复杂和波动加剧的背景下&#xff0c;风险管理成为金融机构和个人投资者关注的核心议题之一。VaR&…...

LangChain知识库管理后端接口:数据库操作详解—— 构建本地知识库系统的基础《二》

这段 Python 代码是一个完整的 知识库数据库操作模块&#xff0c;用于对本地知识库系统中的知识库进行增删改查&#xff08;CRUD&#xff09;操作。它基于 SQLAlchemy ORM 框架 和一个自定义的装饰器 with_session 实现数据库会话管理。 &#x1f4d8; 一、整体功能概述 该模块…...

JavaScript 数据类型详解

JavaScript 数据类型详解 JavaScript 数据类型分为 原始类型&#xff08;Primitive&#xff09; 和 对象类型&#xff08;Object&#xff09; 两大类&#xff0c;共 8 种&#xff08;ES11&#xff09;&#xff1a; 一、原始类型&#xff08;7种&#xff09; 1. undefined 定…...

C++课设:简易日历程序(支持传统节假日 + 二十四节气 + 个人纪念日管理)

名人说:路漫漫其修远兮,吾将上下而求索。—— 屈原《离骚》 创作者:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 专栏介绍:《编程项目实战》 目录 一、为什么要开发一个日历程序?1. 深入理解时间算法2. 练习面向对象设计3. 学习数据结构应用二、核心算法深度解析…...