【洛谷刷题 | 第五天】

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

【洛谷刷题 | 第五天】

    • 今日题目:
      • [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);
}
知识点:递归

什么是递归:

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

总结:

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

相关推荐
Navigator_Z2 小时前
LeetCode //C - 990. Satisfiability of Equality Equations
c语言·算法·leetcode
bbbb3652 小时前
图算法的最优路径搜索与边界约束建模的技术7
算法
東雪木2 小时前
编程算法学习——栈与队列算法
学习·算法·排序算法
CSDN_Colinw2 小时前
C++中的工厂方法模式
开发语言·c++·算法
liulilittle2 小时前
范围随机算法实现
开发语言·c++·算法·lua·c·js
2401_857918293 小时前
C++中的访问者模式实战
开发语言·c++·算法
elseif1233 小时前
CSP-S提高级大纲
开发语言·数据结构·c++·笔记·算法·大纲·考纲
熊猫_豆豆3 小时前
Python 基于Dlib和OpenCV实现人脸融合算法+代码
图像处理·python·算法·人脸融合
Book思议-3 小时前
【数据结构实战】双向链表:在指定位置插入数据
c语言·数据结构·算法·链表