每日c/c++题 备战蓝桥杯(洛谷P1481 魔族密码 题解)

06-01 1597阅读

洛谷P1481 魔族密码 题解——动态规划破解最长前缀链问题

题目大意

给定N个蜜蜂族密码(字符串),要求找出最长的密码链。链中每个密码必须是前一个密码的前缀,且每个密码只能出现一次。例如密码"a"可以接"ab"、"abc"等,但"ba"不满足条件。

解题思路

本题本质是求字符串数组中的最长前缀链长度,核心思路是动态规划(DP)。通过维护一个dp数组,其中dp[i]表示以第i个字符串结尾的最长链长度。

状态定义

int dp[2005] = {0}; // dp[i]表示以ss[i]结尾的最长链长度

状态转移

对于每个字符串ss[i],遍历它之前的所有字符串ss[j]:

  1. 当ss[j]是ss[i]的前缀时,说明可以将ss[i]接在ss[j]之后
  2. 此时状态转移方程为:dp[i] = max(dp[i], dp[j] + 1)

关键优化

使用字符串的find方法判断前缀关系:

if(ss[i].find(ss[j]) == 0) // 精确判断前缀

该方法的时间复杂度为O(L),其中L是字符串长度,比逐字符比较更高效。

代码解析

#include
using namespace std;
int N;
string ss[2005] = {};
int dp[2005] = {0};
int main() {
    ios::sync_with_stdio(false); // 加速输入输出
    cin.tie(0); cout.tie(0);
    
    cin >> N;
    int ans = 0;
    
    for(int i = 1; i 
        cin > ss[i];
        dp[i] = 1; // 每个字符串自身至少成链
        
        for(int j = 1; j 
免责声明:我们致力于保护作者版权,注重分享,被刊用文章因无法核实真实出处,未能及时与作者取得联系,或有版权异议的,请联系管理员,我们会立即处理! 部分文章是来自自研大数据AI进行生成,内容摘自(百度百科,百度知道,头条百科,中国民法典,刑法,牛津词典,新华词典,汉语词典,国家院校,科普平台)等数据,内容仅供学习参考,不准确地方联系删除处理! 图片声明:本站部分配图来自人工智能系统AI生成,觅知网授权图片,PxHere摄影无版权图库和百度,360,搜狗等多加搜索引擎自动关键词搜索配图,如有侵权的图片,请第一时间联系我们。

目录[+]

取消
微信二维码
微信二维码
支付宝二维码