IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 【2021.10.10CCPC复赛】1006 Nun Heh Heh Aaaaaaaaaaa -> 正文阅读

[数据结构与算法]【2021.10.10CCPC复赛】1006 Nun Heh Heh Aaaaaaaaaaa

题目链接:Nun Heh Heh Aaaaaaaaaaa - HDU 7131 - Virtual Judge (vjudge.net) ?

题目描述:

?


解题思路:

快速幂+动态规划+排列组合+思维?

把主串s分成前缀数组和后缀数组两部分,后缀数组记录a的个数,前缀数组保存每个位置前出现nunhehheh的个数(dp),那么答案就是:假设某个nunhehheh后面有n个a,这一组的个数就是2^(n-1)-1,所有情况个数就是算出每种可能然后求和。


快速幂:

直接上代码

//快速幂
ll powerMod(ll a,ll b)
{
	ll ans=1;
	a=a%mod;
	while(b>0)
	{
		if(b%2==1)  ans=(ans*a)%mod;
		b=b/2;
		a=(a*a)%mod;
	}
	return ans;
}

动态规划:

首先是后缀数组(求每个位置之后的a有多少个),简单dp,不做说明。

void get_a(int len)
{
    for(int i = len-1;i>=0;i--){
        s[i] = s[i+1];
        if(s[i] == 'a') s[i] += 1;
    }
}

然后是前缀数组,这里有两种算法,一种空间复杂度O(mn),一种空间复杂度O(n),时间复杂度都是O(mn)。前缀数组这儿卡了好久,其实和LCS有点像,可以先去学LCS。


第一种,空间复杂度O(mn):

s串为用户输入,t串为nunhehheh。

二维动态规划,定义一个dp[10] [MAX_N], dp[j] [i]的意思是s的前i-1个字符中有多少个子串和t中的1~j-1字符串相同。

动态规划:如果s的前i-1个字符中有x个子串与t中的1~j-1字符串相同,那么s的前i个字符中一定有x个子串与t中的1~j-1字符串相同。如果s[i] == t[j-1],那么s的前i个字符中还会多出 s的前i-2个字符的子串与t中1~j-2子符串相同时个数的个数。(绕的要死,建议看公式)

换成公式就是:

dp[j][i] = dp[j][i-1]%mod;
            if(s[i] == t[j])
                dp[j][i] += dp[j-1][i-1]%mod;

当找出完整字符串时,直接计算一次答案。

if(j == len_t){
                dp[j][i] = (dp[j-1][i-1]%mod)*(s[i] == t[j]);
                //如果是完整串,单次计算,只需要上一位的所有可能就行了
                if(dp[j][i]){
                    //计算
                    ans  = ans%mod + dp[j][i]*(powerMod(2,a[i+1])-1)%mod;
                }
            }

该模块代码:

void Dp(int len_s,int len_t)
{
    for(int i = 0;i<=len_s;i++) dp[0][i] = 1;//s的第i个字符之前(包括自己)与 t一个都不匹配的数量是1

    for(int i = 1;i<=len_s;i++){
        for(int j = 1;j<=len_t;j++){
            dp[j][i] = dp[j][i-1]%mod;
            if(s[i] == t[j])
                dp[j][i] += dp[j-1][i-1]%mod;//如果该位相同,会多出上一位匹配的总可能数。
            if(j == len_t){
                dp[j][i] = (dp[j-1][i-1]%mod)*(s[i] == t[j]);
                //如果是完整串,单次计算,只需要上一位的所有可能就行了
                if(dp[j][i]){
                    //计算
                    ans  = ans%mod + dp[j][i]*(powerMod(2,a[i+1])-1)%mod;
                }
            }
        }
    }
}


AC代码:

#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<queue>
#include<cmath>
#include<string>
using namespace std;
typedef long long int ll;

const int MAX_N = 1e5+10;
const ll mod = 998244353;
ll a[MAX_N];
ll dp[10][MAX_N];
ll ans = 0;
string s;
string t= "$nunhehheh";

inline void init()
{
    ans = 0;
    memset(a,0,sizeof(a));
    memset(dp,0,sizeof(dp));
}

void get_a(int len)
{
    for(int i = len;i>0;i--){
        a[i] = a[i+1];
        if(s[i] == 'a') a[i] += 1;
    }
}

//快速幂
ll powerMod(ll a,ll b)
{
	ll ans=1;
	a=a%mod;
	while(b>0)
	{
		if(b%2==1)  ans=(ans*a)%mod;
		b=b/2;
		a=(a*a)%mod;
	}
	return ans;
}


void Dp(int len_s,int len_t)
{
    for(int i = 0;i<=len_s;i++) dp[0][i] = 1;//s的第i个字符之前(包括自己)与 t一个都不匹配的数量是1

    for(int i = 1;i<=len_s;i++){
        for(int j = 1;j<=len_t;j++){
            dp[j][i] = dp[j][i-1]%mod;
            if(s[i] == t[j])
                dp[j][i] += dp[j-1][i-1]%mod;//如果该位相同,会多出上一位匹配的总可能数。
            if(j == len_t){
                dp[j][i] = (dp[j-1][i-1]%mod)*(s[i] == t[j]);
                //如果是完整串,单次计算,只需要上一位的所有可能就行了
                if(dp[j][i]){
                    //计算
                    ans  = ans%mod + dp[j][i]*(powerMod(2,a[i+1])-1)%mod;
                }
            }
        }
    }
}

int main()
{
    int T;
    cin>>T;
    
    while(T--){
        init();
        cin>>s;
        s = "$"+s;
        int len_s = s.length()-1;
        int len_t = t.length()-1;
        get_a(len_s);
        Dp(len_s,len_t);
        cout<<ans%mod<<endl;
    }
    return 0;
}

第二种之后再写思路,先贴个AC代码(朋友写的)

空间复杂度:O(n)

AC代码:?

#include<iostream>
#include<cstring>
using namespace std;
#define ll long long
ll num[100050];
ll numa[100050];
const ll mod = 998244353;

inline ll PowerMod(ll a, ll b, ll c){
	ll ans = 1;
	a = a % c;
	while(b>0){
		if(b % 2 == 1)
			ans = ((ans % c) * (a % c)) % c;
		b = b/2;
		a = (a * a) % c;
	}
	return ans;
}

int main(void){
    string aim = "nunhehheh";
    int len = aim.length();
    int t;
    cin>>t;
    while(t--){
        ll ans = 0;
        string str;
        cin>>str;
        int l = str.length();
        memset(num, 0, sizeof(num));
        memset(numa, 0, sizeof(numa));
        for(int i = l-1;i >= 0;i--){
            if(str[i] == 'a') numa[i] = numa[i + 1] + 1;
            else numa[i] = numa[i + 1];
        }
        num[0] = 1;
        for(int i = 0;i < l;i++){
            for(int j = len;j>0;j--){
                if(str[i] == aim[j-1]){
                    if(j == len)  ans = (ans + (((PowerMod(2, numa[i], mod) - 1) % mod) * num[len-1]) % mod) % mod;
                    else{
                        num[j] = (num[j] + num[j-1]) % mod;
                    }
                }
            }
        }
        cout<<ans % mod<<endl;
    }
    return 0;
}

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-10-12 23:43:35  更:2021-10-12 23:44:13 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/6 17:12:42-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码