倍增优化dp,P10976 统计重复个数

目录

一、题目

1、题目描述

2、输入输出

2.1输入

2.2输出

3、原题链接

二、解题报告

1、思路分析

2、复杂度

3、代码详解


一、题目

1、题目描述

2、输入输出

2.1输入
2.2输出

3、原题链接

https://www.luogu.com.cn/problem/P10976


二、解题报告

1、思路分析

注意到 \[s2, n2, m] = s2, m \* n2,所以如果我们可以计算出 s1, n1 最多可以得到多少个s2,不妨设为 ans 个,那么最大的 m 就是 floor(ans / m)

如何计算 ans?

如果我们定义 fi, j 为 从 i 开始,得到 j 个 s2 所需最少字符数的话,fi, j = f(i + \|s1\|) % \|s1\|, j - 1 + fi, 1

很经典的阶段性递推,我们考虑倍增加速这个过程:

定义 fi, j 为 从 i 开始,得到 2^j 个 s2 所需的最少字符数,那么 fi, j = fi, j - 1 + f(i + f\[i, j - 1) % |s1|, j - 1]

预处理完f\[\] 后,我们从 0 出发(从开头出发一定是最优的,因为从中间出发,计算所需字符数还是要加上跳过的字符),从高位到低位贪心的累加f\[\],即可得到ans

2、复杂度

时间复杂度: O(|s1||s2| + n1 log(n1 |s1| / |s2|))空间复杂度:O(|s1| + |s2| + n1 log(n1 |s1| / |s2|)

3、代码详解

复制代码
 ​
cpp 复制代码
#include <bits/stdc++.h>
namespace ranges = std::ranges;
namespace views = std::views;
using i64 = long long;
using u32 = unsigned;
using u64 = unsigned long long;

constexpr int B = 30;
constexpr int N = 100;

i64 dp[N][B];

int main() {
    std::ios::sync_with_stdio(false);
    std::cin.tie(nullptr);

    std::string s1, s2;
    int n1, n2;
    while (std::cin >> s2 >> n2 >> s1 >> n1) {
        /*
        [[s2, n2], m] = [s2, m * n2]        
        */ 
        bool ok = true;
        for (int i = 0; i < s1.size(); ++i) {
            dp[i][0] = 0;
            int cur = i;
            for (int j = 0; j < s2.size(); ++j) {
                int c = 0;
                while (s1[cur] != s2[j]) {
                    if (++cur == s1.size()) cur = 0;
                    if (++c > s1.size()) {
                        ok = false;
                        break;
                    }
                }   
                if (!ok) break;
                if (++cur == s1.size()) cur = 0;
                dp[i][0] += c + 1;
            }
            if (!ok) break;
        }
        if (!ok) {
            std::cout << 0 << '\n';
            continue;
        }
        for (int j = 0; j + 1 < B; ++j) {
            for (int i = 0; i < s1.size(); ++i) {
                dp[i][j + 1] = dp[i][j] + dp[(i + dp[i][j]) % s1.size()][j];
            }
        }
        i64 ans = 0, tot = 0;
        for (int j = B - 1; j >= 0; --j) {
            if (tot + dp[tot % s1.size()][j] <= n1 * s1.size()) {
                tot += dp[tot % s1.size()][j];
                ans |= 1 << j;
            }
        }

        std::cout << ans / n2 << '\n';
    }

    return 0;
}
相关推荐
AI Dog18 小时前
MathHub数学建模交流社区-V2
人工智能·机器学习·数学建模·阿里云
南境十里·墨染春水18 小时前
讲讲移动语义
算法
西凉的悲伤18 小时前
Guava类库——Range连续区间
java·算法·guava
菜菜的顾清寒18 小时前
力扣HOT(100)54多维动态规划-最长公共子序列
算法·leetcode·动态规划
随意起个昵称18 小时前
线性dp-LIS题目3(合唱队形)
算法
小六学编程18 小时前
二分查找详解:从普通二分到左右边界
算法·c/c++
wayz1118 小时前
Volume:PVO(百分比成交量震荡指标)技术指标详解
算法·金融·数据分析·量化交易·特征工程
毕竟是shy哥19 小时前
PromptHash:基于亲和提示协同学习的自适应哈希检索跨模态算法
学习·算法·哈希算法
甄心爱学习19 小时前
【项目实训(个人12)】
人工智能·python·算法
团象科技19 小时前
走访近百支出海技术团队后的海外云计算资源选型实操观察
大数据·人工智能·算法