P3612 秘密奶牛码题解

题目传送门

思路:

递归

题意:给定一个序列s,用它能够生成一个无限长的序列
第一个序列:s1=s
第二个序列:s2=s1+s1
第三个序列:s3=s2+s2
...
其中si表示将si中的最后一个元素放到最前面形成的新序列。

把一个序列看成前半部分和后半部分,看位置x在前半部分还是后半部分:
在前半部分 => 子问题:前半部分的x位置
在后半部分 => 子问题:通过xmid位置想办法找到对应的前半部分的位置 。

xmid==1?mid:xmid1

举个栗子秒懂:
COW>COW  WCO>COWWCO  OCOWWC>....
x=8,就是想知道上面第8个位置,就是红色位置的字符是什么,上面的用例就是C

在任何一个普通的情况下,我们都试图找出当前位置与前一个字符串的位置对应关系,如果关系确定了,那么问题就可通过递归一层层返回到原始串了,也就解决了问题。

那么如何确定与前一个字符串的字符位置对应关系呢?
比如上面的栗子:
x=7,目标值是O,它其实是上一个字符串的最后一个字符转过来的。就是prex=6

x=8,目标值是C,它其实是上一个字符串的第一个字符转过来的。就是prex=1

x=9,目标值是O,它其实是上一个字符串的第二个字符转过来的。就是prex=2

x=10,目标值是W,它其实是上一个字符串的第三个字符转过来的。就是prex=3

...

结论:除了交接位置的那个是它的前一个以外,其余的都是x一半的字符串长度1,写成数学表达式就是:

xmid==1?mid:xmid1

C++代码

#include <bits/stdc++.h>

using namespace std;
typedef long long LL;

//原始字符串
string s;

//预求的位置值
LL x;

//在n个长度的字符串中,查找位置x的字符
char dfs(LL n, LL x) {
    //如果回到原始串,那么直接返回指定位置的字符
    if (n == s.size()) return s[x - 1];//之所以x-1,是因为字符串下标是从0开始的

    //字符串的中间点
    LL mid = n >> 1;

    //如果在后半部分(转化为前半部分的位置)
    if (x > mid)
        return dfs(mid, (x == mid + 1 ? mid : x - mid - 1));
        //如果在前半部分
    else
        return dfs(mid, x);
}

int main() {
    cin >> s >> x;
    //原始长度
    LL n = s.size();

    //不断乘2,翻倍,找出能够容纳x这个数字的最短长度
    while (n < x) n <<= 1;

    //递归
    cout << dfs(n, x);
}
posted @   糖豆爸爸  阅读(186)  评论(0编辑  收藏  举报
编辑推荐:
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
阅读排行:
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 【自荐】一款简洁、开源的在线白板工具 Drawnix
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
· Docker 太简单,K8s 太复杂?w7panel 让容器管理更轻松!
历史上的今天:
2013-07-19 java连接mysql批量写入数据
Live2D
点击右上角即可分享
微信分享提示