华为OD 最小循环子数组

1. 题意

给定一个由若干整数组成的数组 nums,请检查数组是否是由某个子数组重复循环拼接而成,请输出这个最小的子数组。

2. 题解

利用 k m p kmp kmp中的 n e x t next next数组性质,我们可以求出 n u m s nums nums中的最长公共

前缀后缀。如果要是由一个子数组循环组成的话,那么必然有 n e x t [ s z ] % ( n − n e x t [ s z ] ) = 0 next[sz] \% (n-next[sz]) =0 next[sz]%(n−next[sz])=0。

为什么满足这个性质就能说明是由子数组循环若干次组成的呢?

我们假设上面的条件成立,且将字符串每 k = n − n e x t [ n ] k = n-next[n] k=n−next[n]个划为一组。那么字符串可以表示为 b 1 ⋯ b n / k b_1\cdots\ b_{n/k} b1⋯ bn/k。

由于 k = n − n e x t [ n ] k=n-next[n] k=n−next[n], 因此串 b 0 b 1 ⋯ b n / k − 1 = b 2 ⋯ b n / k b_0b_1\cdots b_{n/k-1}=b_2\cdots b_{n/k} b0b1⋯bn/k−1=b2⋯bn/k(最长公共前后缀)。进而
b 1 = b 2 b 2 = b 3 ⋯ b n / k − 1 = b n / k b_1=b_2\\ b_2=b_3\\ \cdots\\ b_{n/k-1}=b_{n/k} b1=b2b2=b3⋯bn/k−1=bn/k

因此
b 1 = b 2 = ⋯ = b n / k b_1=b_2=\cdots=b_{n/k} b1=b2=⋯=bn/k

串为循环串。

例如

复制代码
s     :    a  b c a b c a b c
next  :    -1 0 0 0 1 2 3 4 5 6
n = 9
next[n] = 6
n -next[n] = 3
6 = 2 x 3
  • 代码
cpp 复制代码
#include <iostream>
#include <string>
#include <vector>

int main()
{
    int n;
    std::cin >> n;

    std::vector<int> a(n, 0);

    for ( int i = 0; i < n; ++i) {
        std::cin >> a[i];
    }

    std::vector<int> next( n + 1, 0);
    int j = -1;
    int i =  0;
    next[0] = -1;

    while ( i < n) {
        if ( j == -1 || a[i] == a[j]) {
            ++i;
            ++j;
            next[i] = j;
        }
        else {
            j = next[j];
        }
    }

    int k = n;
    if ( (next[n] >= (n - next[n])) && (next[n] % (n - next[n]) == 0)) {
        k = n - next[n];
    }

    for (int i = 0; i < k; ++i) {
        if ( i )
            std::cout << " ";
        std::cout << a[i];
    }


    return 0;
}
相关推荐
_OP_CHEN5 小时前
【算法基础篇】(五十七)线性代数之矩阵乘法从入门到实战:手撕模板 + 真题详解
线性代数·算法·矩阵·蓝桥杯·c/c++·矩阵乘法·acm/icpc
天天爱吃肉82185 小时前
【跨界封神|周杰伦×王传福(陶晶莹主持):音乐创作与新能源NVH测试,底层逻辑竟完全同源!(新人必看入行指南)】
python·嵌入式硬件·算法·汽车
im_AMBER5 小时前
Leetcode 114 链表中的下一个更大节点 | 删除排序链表中的重复元素 II
算法·leetcode
xhbaitxl5 小时前
算法学习day38-动态规划
学习·算法·动态规划
多恩Stone5 小时前
【3D AICG 系列-6】OmniPart 训练流程梳理
人工智能·pytorch·算法·3d·aigc
历程里程碑6 小时前
普通数组----轮转数组
java·数据结构·c++·算法·spring·leetcode·eclipse
pp起床6 小时前
贪心算法 | part02
算法·leetcode·贪心算法
sin_hielo6 小时前
leetcode 1653
数据结构·算法·leetcode
2501_901147836 小时前
面试必看:优势洗牌
笔记·学习·算法·面试·职场和发展
YuTaoShao6 小时前
【LeetCode 每日一题】3634. 使数组平衡的最少移除数目——(解法二)排序 + 二分查找
数据结构·算法·leetcode