蓝桥杯备赛:Day3-P1918 保龄球

📚 算法笔记:P1918 保龄球 (STL map 离散映射)

1. 题目简述

P1918 保龄球 - 洛谷

给定 N N N 堆保龄球,每堆有 a i a_i ai 个球,位置编号为 1 ∼ N 1 \sim N 1∼N。有 Q Q Q 次询问,每次给出一个数字 M M M,问哪一堆的球数恰好是 M M M。

  • 数据范围 : N , Q ≤ 10 5 N, Q \le 10^5 N,Q≤105, a i ≤ 10 9 a_i \le 10^9 ai≤109。
  • 输出要求 :存在则输出原位置编号,不存在输出 0 0 0。

2. 核心代码 (C++ 实现)

c++ 复制代码
#include<bits/stdc++.h>
using namespace std;

typedef long long ll;
ll N;             // 堆数
map<ll, ll> Ai;   // 核心:建立从 "球的数量" 到 "位置编号" 的映射
ll Q;             // 询问次数
ll M;             // 询问的球数

void solve()
{
    // 1. 读入数据并建立映射
    if(!(cin >> N)) return;
    for(int i = 1; i <= N; i++)
    {
        ll a;
        cin >> a;
        // Key: 球数 (10^9级别), Value: 位置 (10^5级别)
        Ai[a] = i; 
    }
    
    // 2. 处理多次查询
    cin >> Q;
    for (int i = 1; i <= Q; i++)
    {
        cin >> M;
        // 使用 count 检查是否存在,时间复杂度 O(log N)
        if(Ai.count(M) == 1)
            cout << Ai[M] << "\n"; // 使用 "\n" 避免 endl 频繁刷新缓冲区
        else
            cout << 0 << "\n";
    }
}

int main()
{
    // 蓝桥杯必备:加速大量数据的输入输出
    ios::sync_with_stdio(0);
    cin.tie(0), cout.tie(0);
    
    solve();
    return 0;
}

3. 核心考点与注意事项

🔍 核心考点
  1. 反向索引映射 :通常我们习惯用"位置"查"数值",本题要求"数值"查"位置"。在数值范围巨大( 10 9 10^9 109)而堆数有限( 10 5 10^5 105)的情况下,使用 std::map 是最优选择。

    这里的键值对的插入方式和前面学籍管理的数据插入方式很不一样,注意区分。

  2. 查找效率map 底层是红黑树,count()operator[] 的复杂度均为 O ( log ⁡ N ) O(\log N) O(logN)。总复杂度为 O ( ( N + Q ) log ⁡ N ) O((N+Q) \log N) O((N+Q)logN),能稳过 10 5 10^5 105 级别的数据。

  3. I/O 优化 :处理十万级别的输入输出时,endl 会强制清空缓冲区导致速度剧降。"\n" 是提速的关键。

⚠️ 注意事项
  • 空间换时间map 会带来额外的内存开销。虽然本题内存限制宽裕(125MB),但在更严苛的题目中,若只需计数或存在性判断且内存吃紧,可考虑 unordered_map(平均 O ( 1 ) O(1) O(1),但最坏 O ( N ) O(N) O(N))。
  • Long Long 的必要性 :球数 10 9 10^9 109 虽然在 int 范围内,但为了防止计算溢出或题目潜在升级,统一使用 ll 是更稳健的习惯。

4. 易错点回顾 (My Mistakes)

  1. 映射方向倒置 :最初尝试以位置为键。纠正: 明确"已知量"作为 Key,"待求量"作为 Value。
  2. 输入作用域错误 :曾将查询目标 M 的输入放在了循环外。纠正: 查询题目必须在每次循环内部实时读入新的目标。
  3. 语法误区 :误将成员函数 counterase 当作数组访问。纠正: 函数调用需使用 ()
  4. 不必要的删除 :曾尝试在查询后使用 erase纠正: 同一个球数可能被多次询问,不可随意删除已建立的索引。
相关推荐
wuminyu4 小时前
Java锁机制之park和unpark源码剖析
java·linux·c语言·jvm·c++
玖玥拾5 小时前
C/C++ 基础笔记(十一)类的进阶
c语言·c++·设计模式·
-森屿安年-5 小时前
1137. 第 N 个泰波那契数
c++·动态规划
程序员老舅6 小时前
从内核视角,看Linux文件读写过程
linux·服务器·c++·内核·linux内核·vfs·linux内存
Soari6 小时前
llama.cpp更新(b9553):LLM inference in C/C++,本地和云端实现高性能大模型推理
c语言·c++·llama
2601_961194026 小时前
考研资料电子版|去哪找|网盘
java·c语言·c++·python·考研·php
Peter·Pan爱编程6 小时前
23. 算法库:用算法代替手写循环
c++·人工智能·算法
大白话_NOI7 小时前
【洛谷 P1303】A*B Problem + 详细分析
c++
小欣加油7 小时前
leetcode2161 根据给定数字划分数组
数据结构·c++·算法·leetcode·职场和发展
吃着火锅x唱着歌7 小时前
深度探索C++对象模型 学习笔记 第五章 构造、解构、拷贝语意学(2)
c++·笔记·学习