【洛谷刷题 | 第五天】

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

【洛谷刷题 | 第五天】

    • 今日题目:
      • [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 s[30][10009] 是二维字符串数组,第一维对应 26 个字母分组,第二维对应组内单词的位置。

s[1][10009] 就表示 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);
}
知识点:递归

什么是递归:

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

总结:

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

相关推荐
王老师青少年编程3 小时前
csp信奥赛C++高频考点专项训练之贪心算法 --【哈夫曼贪心】:合并果子
c++·算法·贪心·csp·信奥赛·哈夫曼贪心·合并果子
叼烟扛炮4 小时前
C++第二讲:类和对象(上)
数据结构·c++·算法·类和对象·struct·实例化
天疆说4 小时前
【哈密顿力学】深入解读航天器交会最优控制中的Hamilton函数
人工智能·算法·机器学习
wuweijianlove4 小时前
关于算法设计中的代价函数优化与约束求解的技术7
算法
leoufung5 小时前
LeetCode 149: Max Points on a Line - 解题思路详解
算法·leetcode·职场和发展
样例过了就是过了5 小时前
LeetCode热题100 最长公共子序列
c++·算法·leetcode·动态规划
HXDGCL5 小时前
矩形环形导轨:自动化循环线的核心运动单元解析
运维·算法·自动化
谭欣辰5 小时前
C++ 排列组合完整指南
开发语言·c++·算法
代码中介商6 小时前
银行管理系统的业务血肉 —— 流程、状态机、输入校验与持久化(下篇)
c语言·算法
foundbug9996 小时前
自适应滤除直达波干扰的MATLAB实现
开发语言·算法·matlab