【洛谷刷题 | 第五天】

本系列文章我将总结我在刷算法题所用到的知识,如果你也在刷算法并且是新手,我相信这系列文章会很适合你。

【洛谷刷题 | 第五天】

    • 今日题目:
      • [1.COCI 2017/2018 #2 ​​ZigZag(字符串,排序)](#2] ZigZag(字符串,排序))
      • [2. 国王的魔镜(模拟,字符串,搜索,递归)](#2. 国王的魔镜(模拟,字符串,搜索,递归))

今日题目:

1.COCI 2017/2018 #2 ​​ZigZag(字符串,排序)

链接:P4432 COCI 2017/2018 #2 ​​ZigZag

给你一堆单词,再给你一串首字母查询。每次查一个字母,你要在以这个字母开头的单词里,选出用得次数最少的那个;如果次数一样,就选字典序最小的,选完把这个单词的计数 + 1,最后输出每次选的单词。

案例:

cpp 复制代码
输入                                输出
4 5                                zadar
zagreb                             sisak
split                              split
zadar                              zagreb
sisak                               zadar
z
s
s
z
z

这道题主要用了按首字母分组 + 字典序排序 + 轮询选取的思想。最开始我想着是把所有字符串都按字典序排序,但后面才了解掌握到把每个按首字母分组分别排序的方法;排完序后可以为每个首字母分组维护一个指针,每次查询对应首字母时,指针向后移动(若超出该组单词数量则重置为 1),直接输出指针指向的单词即可。

题解:

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
string s[30][10009],S;
int p[30],num[30];
char ch;
int main()
{
    int k,n;
    cin>>k>>n;
    for(int i=0;i<k;i++){
        cin>>S;
        int t = S[0]-'a'+1;
        s[t][++num[t]] = S;//注意是先++而不是后++
    }
    for(int i=1;i<=26;i++)
    {
        sort(s[i]+1,s[i]+1+num[i]);
    }
    while(n--){
        ch=getchar();
        while(ch<'a'||ch>'z') ch = getchar();
        int x = ch-'a'+1;
        p[x]++;
        if(p[x]>num[x]) p[x]=1;
        cout<<s[x][p[x]]<<'\n';
    }
}
知识点:字符串

1.字典序排序规则

从单词的第一个字符开始比,字符小的单词整体更小(比如 apple < banana,因为 a 在字母表里比 b 靠前);

如果第一个字符相同,就比第二个字符(比如 apple < apply,前 4 个字符都一样,第 5 个 e < y);

以此类推,直到比出第一个不同的字符;

如果一个单词是另一个的前缀(比如 app 和 apple),则短的单词更小。

2.如何排序:

跟数组一样用sort就可以从小到大将字符串按照字典序排序。

像这到题string s3010009 是二维字符串数组,第一维对应 26 个字母分组,第二维对应组内单词的位置。

s110009 就表示 a 开头的第 10009 个单词

cpp 复制代码
 for(int i=1;i<=26;i++)
    {
        sort(s[i]+1,s[i]+1+num[i]);
    }

这个代码的作用就是:把每个首字母(a-z)对应的所有单词,单独整理成 "字典序从小到大" 的顺序。

3.getchar

getchar() 是 C/C++ 里的一个输入函数,核心作用是:从输入读取一个字符(包括空格、换行、Tab 这些看不见的字符),相比 cin,getchar() 更底层、更灵活,适合需要精准控制输入的场景。

2. 国王的魔镜(模拟,字符串,搜索,递归)

链接:P2799 国王的魔镜

给一个由大写字母组成的最终项链字符串(是通过 "每次把一端接触魔镜,让项链长度翻倍且新增部分反转" 得到的),求最初项链的最小可能长度。

案例:

cpp 复制代码
输入                               输出
ABBAABBA                           2

从最终的字符串不断往前倒推:每次检查当前字符串是不是前后对称,如果对称,就说明它是由前半段用魔镜变出来的,那就只保留前半段继续检查;如果不对称或者长度是奇数,就不能再缩小了,此时的长度就是最初项链的最小长度。

题解:

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
string s;
int p(int x){
    if((x+1)%2==1) return x+1;
    if(x==0) return 1;
    int le = 0;int re = x;
    while(le<re){
        if(s[le]!=s[re])
        		  return x+1;
              le++;re--;
                }
     return p(re);
}
int main()
{
    cin>>s;
    int n = s.size();
    cout<<p(n-1);
}
知识点:递归

什么是递归:

递归就是函数自己调用自己,核心逻辑是 "把复杂问题拆成和原问题相似但规模更小的子问题",直到子问题简单到能直接解决,再从子问题的答案推导出原问题的答案。

总结:

如果我的内容对你有帮助,请点赞,评论,收藏。创作不易,大家的支持就是我坚持下去的动力!

相关推荐
JieE21221 小时前
LeetCode 101. 对称二叉树|JS 递归 + 迭代双解法,彻底搞懂镜像判断
javascript·算法
JieE2122 天前
LeetCode 56. 合并区间|超清晰 JS 图解思路,面试高频区间题
javascript·算法·面试
Jack202 天前
HarmonyOS开发中错误处理策略:网络异常统一处理
算法
小小杨树2 天前
读懂色彩:拍照调色不再难
算法·计算机视觉·配色
JieE2123 天前
LeetCode 226. 翻转二叉树|JS 递归超详细拆解,二叉树入门经典题
javascript·算法
JieE2123 天前
LeetCode 104. 二叉树的最大深度|递归思路超详细拆解
javascript·算法
vivo互联网技术3 天前
CVPR 2026 | 全新强化学习框架 BeautyGRPO:重塑真实人像
算法·大模型·cvpr·影像
Darling噜啦啦3 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
用户497863050733 天前
(一)小红的数组操作
算法·编程语言