【CSP试题回顾】202112-2-序列查询新解

2024-03-09 06:04

本文主要是介绍【CSP试题回顾】202112-2-序列查询新解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

CSP-202112-2-序列查询新解

关键点:时间复杂度

本题关键在于它避免暴力枚举法的时间复杂度。暴力枚举法可能会对每一对可能的 f(x)g(x) 组合进行比较,其时间复杂度为 O ( n 2 ) O(n^2) O(n2),这对于大数据集来说是不可行的。

  • 【解决思路】:通过逐步遍历 f(x)g(x) 的值(每次只移动到下一个 f(x)g(x) 值),将时间复杂度降低到了 O ( n ) O(n) O(n)。这种方法只需要遍历一次 f(x)g(x) 的所有值,大大减少了计算量。

  • 这种方法的效率提高主要是因为它保持了 f(x)g(x) 索引的逻辑顺序,这样就能在单个循环内完成所有的计算,避免了不必要的重复比较。

解题思路

1.初始化

  • FiGi 分别是 f(x)g(x) 当前索引的指针,初始值都设置为 0。
  • lastIndex 表示最近处理的点的位置,初始化为 0。
  • error 用于累积总误差,也是从 0 开始的。

2.遍历过程

  1. 循环条件while 循环会持续,直到 FiGi 其中一个超过各自数组的长度。确保所有的 f(x)g(x) 值都被遍历

  2. 找到下一个处理点(重要):循环中,首先确定下一个处理点的位置 nextIndex。这是当前 f(x)(由 arrF_x[Fi] 给出)和 g(x)(由 arrG_x[Gi] 给出)值中较小的一个。这个点是当前关注的位置,因为它表示下一个即将处理的 f(x)g(x) 的值。

  3. 计算长度和误差(重要):根据 nextIndexlastIndex(上一个点的位置),计算这两点之间的距离 length = nextIndex - lastIndex。然后,利用这段长度和 FiGi 之间的差的绝对值更新总误差 error。这个差的绝对值代表在此段中 f(x)g(x) 之间的偏差。

  4. 更新索引(重要):如果 arrF_x[Fi](当前 f(x) 的值)是 nextIndex,那么 Fi 自增以移动到 f(x) 的下一个值。同样,如果 arrG_x[Gi](当前 g(x) 的值)是 nextIndex,那么 Gi 自增以移动到 g(x) 的下一个值。这个步骤确保了在每次迭代后,FiGi 至少有一个向前移动。

  5. 重复过程:更新 lastIndexnextIndex,然后重复循环,直到遍历完所有的 f(x)g(x) 值。

3.逐步遍历的意义

  • 通过逐步遍历,代码能够有效地计算在每一小段上 f(x)g(x) 的近似误差,并累积这些误差以得到整体误差。这种方法利用了线性遍历的策略,减少了计算复杂度,相比暴力方法有显著的性能提升。

完整代码

#include<iostream>
#include<vector>
using namespace std;long long n, N, error, r;int main() {   cin >> n >> N; // 输入n和Nr = N / (n + 1); // 计算每个区间的长度vector<long long> arrF_x(n); // 存储f(x)的值for (int i = 0; i < n; i++) {cin >> arrF_x[i]; // 输入f(x)的值}arrF_x.push_back(N); // 将N作为最后一个f(x)的值,确保能够覆盖全部范围vector<long long> arrG_x; // 存储g(x)的值for (int i = r; i < N; i += r) {arrG_x.push_back(i); // 按照间隔r生成g(x)的值}if (arrG_x.back() != N) { // 确保N也作为g(x)的一个值arrG_x.push_back(N);}long long Fi = 0, Gi = 0, lastIndex = 0; // 初始化索引和上一个点的位置while (Fi < arrF_x.size() && Gi < arrG_x.size()) { // 遍历f(x)和g(x)long long nextIndex = min(arrF_x[Fi], arrG_x[Gi]); // 找到下一个处理点的位置long long length = nextIndex - lastIndex; // 计算两点之间的长度lastIndex = nextIndex; // 更新上一个点的位置error += length * abs(Fi - Gi); // 更新误差if (arrF_x[Fi] == nextIndex) Fi++; // 如果f(x)到达下一个点,则移动Fiif (arrG_x[Gi] == nextIndex) Gi++; // 如果g(x)到达下一个点,则移动Gi}cout << error; // 输出总误差return 0;
}

请添加图片描述

这篇关于【CSP试题回顾】202112-2-序列查询新解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

mysql线上查询之前要性能调优的技巧及示例

《mysql线上查询之前要性能调优的技巧及示例》文章介绍了查询优化的几种方法,包括使用索引、避免不必要的列和行、有效的JOIN策略、子查询和派生表的优化、查询提示和优化器提示等,这些方法可以帮助提高数... 目录避免不必要的列和行使用有效的JOIN策略使用子查询和派生表时要小心使用查询提示和优化器提示其他常

SQL 中多表查询的常见连接方式详解

《SQL中多表查询的常见连接方式详解》本文介绍SQL中多表查询的常见连接方式,包括内连接(INNERJOIN)、左连接(LEFTJOIN)、右连接(RIGHTJOIN)、全外连接(FULLOUTER... 目录一、连接类型图表(ASCII 形式)二、前置代码(创建示例表)三、连接方式代码示例1. 内连接(I

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

关于最长递增子序列问题概述

《关于最长递增子序列问题概述》本文详细介绍了最长递增子序列问题的定义及两种优化解法:贪心+二分查找和动态规划+状态压缩,贪心+二分查找时间复杂度为O(nlogn),通过维护一个有序的“尾巴”数组来高效... 一、最长递增子序列问题概述1. 问题定义给定一个整数序列,例如 nums = [10, 9, 2

轻松上手MYSQL之JSON函数实现高效数据查询与操作

《轻松上手MYSQL之JSON函数实现高效数据查询与操作》:本文主要介绍轻松上手MYSQL之JSON函数实现高效数据查询与操作的相关资料,MySQL提供了多个JSON函数,用于处理和查询JSON数... 目录一、jsON_EXTRACT 提取指定数据二、JSON_UNQUOTE 取消双引号三、JSON_KE

查询SQL Server数据库服务器IP地址的多种有效方法

《查询SQLServer数据库服务器IP地址的多种有效方法》作为数据库管理员或开发人员,了解如何查询SQLServer数据库服务器的IP地址是一项重要技能,本文将介绍几种简单而有效的方法,帮助你轻松... 目录使用T-SQL查询方法1:使用系统函数方法2:使用系统视图使用SQL Server Configu

MYSQL关联关系查询方式

《MYSQL关联关系查询方式》文章详细介绍了MySQL中如何使用内连接和左外连接进行表的关联查询,并展示了如何选择列和使用别名,文章还提供了一些关于查询优化的建议,并鼓励读者参考和支持脚本之家... 目录mysql关联关系查询关联关系查询这个查询做了以下几件事MySQL自关联查询总结MYSQL关联关系查询

Java实现Elasticsearch查询当前索引全部数据的完整代码

《Java实现Elasticsearch查询当前索引全部数据的完整代码》:本文主要介绍如何在Java中实现查询Elasticsearch索引中指定条件下的全部数据,通过设置滚动查询参数(scrol... 目录需求背景通常情况Java 实现查询 Elasticsearch 全部数据写在最后需求背景通常情况下

查询Oracle数据库表是否被锁的实现方式

《查询Oracle数据库表是否被锁的实现方式》本文介绍了查询Oracle数据库表是否被锁的方法,包括查询锁表的会话、人员信息,根据object_id查询表名,以及根据会话ID查询和停止本地进程,同时,... 目录查询oracle数据库表是否被锁1、查询锁表的会话、人员等信息2、根据 object_id查询被

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI