1.4状态机模型

news2024/11/17 10:02:32

状态机简介:
在这里插入图片描述

1.大盗阿福

阿福是一名经验丰富的大盗。趁着月黑风高,阿福打算今晚洗劫一条街上的店铺。

这条街上一共有 N N N家店铺,每家店中都有一些现金。

阿福事先调查得知,只有当他同时洗劫了两家相邻的店铺时,街上的报警系统才会启动,然后警察就会蜂拥而至。

作为一向谨慎作案的大盗,阿福不愿意冒着被警察追捕的风险行窃。

他想知道,在不惊动警察的情况下,他今晚最多可以得到多少现金?

输入格式
输入的第一行是一个整数 T T T,表示一共有 T T T组数据。

接下来的每组数据,第一行是一个整数 N N N,表示一共有 N N N家店铺。

第二行是 N N N个被空格分开的正整数,表示每一家店铺中的现金数量。

每家店铺中的现金数量均不超过 1000 1000 1000

输出格式
对于每组数据,输出一行。

该行包含一个整数,表示阿福在不惊动警察的情况下可以得到的现金数量。

数据范围
1 ≤ T ≤ 50 , 1≤T≤50, 1T50,
1 ≤ N ≤ 1 0 5 1≤N≤10^{5} 1N105

输入样例:

2
3
1 8 2
4
10 7 6 14

输出样例:

8
24

样例解释
对于第一组样例,阿福选择第2家店铺行窃,获得的现金数量为 8 8 8

对于第二组样例,阿福选择第1和4家店铺行窃,获得的现金数量为 10 + 14 = 24 10+14=24 10+14=24

1.1题解

在这里插入图片描述

在这里插入图片描述
现在考虑只从上一层状态,而不是从上两层状态

分解状态
在这里插入图片描述
状态机模型
在这里插入图片描述
利用DP分析法+状态机

在这里插入图片描述
利用状态机的形式把不好表示的状态分离开

思路1
我们可以定义一个数组,为 f [ ] f[] f[]

f [ i ] f[i] f[i]表示抢劫前i家能得到的最多现金数量。

那么我们前i家的抢劫结果就有两种情况:

第一种情况:不偷第i家店铺
那么 f [ i ] = f [ i − 1 ] f[i]=f[i−1] f[i]=f[i1];

第二种情况:偷第i家店铺
那么 f [ i ] = f [ i − 1 ] + w [ i ] . f[i]=f[i−1]+w[i]. f[i]=f[i1]+w[i].

w [ i ] w[i] w[i]表示第i家店铺总共的现金)

思路1出现的问题:
如果第 i − 1 i−1 i1家店已经被抢了,那么如果抢了第 i i i家,那是不符合题目要求的。

那怎么办呢?

正确方法(思路2):

我们把f数组定为二维的,即 f [ ] [ ] f[][] f[][]

我们用数组储存两种情况:偷与不偷。

f [ i ] [ 0 ] f[i][0] f[i][0]代表的是不偷第i家店铺能得到的最多现金数量;
f [ i ] [ 1 ] f[i][1] f[i][1]代表的是偷第i家店铺能得到的最多现金数量。

则就会出现三种情况:
在这里插入图片描述
解释:

图中红色的线是可行方案,你可以不抢第 i − 1 i−1 i1家,也不抢第 i i i家;
你可以不抢第 i − 1 i−1 i1家,但抢第 i i i家。
你可以抢第 i − 1 i−1 i1家,但不抢第 i i i家;

那么我们就可以得出状态转移方程了:

f[i][0] = max(f[i - 1][0], f[i - 1][1]);
f[i][1] = f[i - 1][0] + w[i];
1.2代码实现
#include<iostream>
#include<cstring>
#include<algorithm>

using namespace std;

const int N = 100010,INF = 0x3f3f3f3f;
int n;
int w[N],f[N][2];

int main()
{
    int T;
    scanf("%d",&T);
    while(T--)
    {
        scanf("%d",&n);
        for(int i = 1;i <= n;i++)   scanf("%d",&w[i]);
        f[0][0] = 0,f[0][1] = -INF;
        for(int i = 1;i <= n;i++)
        {
            f[i][0] = max(f[i - 1][0],f[i - 1][1]);
            f[i][1] = f[i - 1][0] + w[i];
        }
        printf("%d\n",max(f[n][0],f[n][1]));
    }
    
    return 0;
}

2.股票买卖 IV

给定一个长度为 N N N的数组,数组中的第 i i i个数字表示一个给定股票在第 i i i天的价格。

设计一个算法来计算你所能获取的最大利润,你最多可以完成 k k k笔交易。

注意:你不能同时参与多笔交易(你必须在再次购买前出售掉之前的股票)。一次买入卖出合为一笔交易。

输入格式
第一行包含整数 N N N k k k,表示数组的长度以及你可以完成的最大交易笔数。

第二行包含 N N N个不超过 10000 10000 10000的正整数,表示完整的数组。

输出格式
输出一个整数,表示最大利润。

数据范围
1 ≤ N ≤ 1 0 5 , 1≤N≤10^{5}, 1N105,
1 ≤ k ≤ 100 1≤k≤100 1k100
输入样例1:

3 2
2 4 1

输出样例1:

2

输入样例2:

6 2
3 2 6 5 0 3

输出样例2:

7

样例解释
样例1:在第 1 天 (股票价格 = 2) 的时候买入,在第 2 天 (股票价格 = 4) 的时候卖出,这笔交易所能获得利润 = 4-2 = 2 。

样例2:在第 2 天 (股票价格 = 2) 的时候买入,在第 3 天 (股票价格 = 6) 的时候卖出, 这笔交易所能获得利润 = 6-2 = 4 。随后,在第 5 天 (股票价格 = 0) 的时候买入,在第 6 天 (股票价格 = 3) 的时候卖出, 这笔交易所能获得利润 = 3-0 = 3 。共计利润 4+3 = 7.

2.1题解

在这里插入图片描述
初始化:f[0][j][0] = 0, 其余-INF,因为在第0个股票一定是无货的,必定从这个位置开始转移才有效

注意:
这里状态机的过程,对于每个股票,要么就买,要么就卖,不能说是买了然后在同一个点直接卖掉,这样是不符合状态机模型的,因此对于上述转移方程可以会有人提出疑问。
为什么状态转移方程不能是下面代码,即卖的时候才算做了一次交易,原代码是买的时候才算一次交易

f[i][j][0] = max(f[i - 1][j - 1][1] + w[i], f[i - 1][j][0]);
f[i][j][1] = max(f[i - 1][j][0] - w[i], f[i - 1][j][1]);

终究要回归到状态转移的起点,第一支股票只有买,和不买这两个操作,一定不可能是卖和不卖的这两个操作,因此第一支股票如果买入时,必须按照一次交易处理。否则如果第一次股票如果买入时,不按一次交易处理,也就代表着第一支股票卖出才算一次交易,也就代表着在第一支股票卖出之前还买了一支股票,显然是矛盾的。

2.2代码实现

#include<iostream>
#include<cstring>
#include<algorithm>

using namespace std;

const int N = 100010,M = 110,INF = 0x3f3f3f3f;

int n,m;
int w[N];
int f[N][M][2];//状态

int main()
{
    scanf("%d%d",&n,&m);
    for(int i = 1;i <= n;i++)   scanf("%d",&w[i]);   
    
    memset(f,-0x3f,sizeof f);
    for(int i = 0;i <= n;i++)    f[i][0][0] = 0;
   
    for(int i = 1;i <= n;i++)
        for(int j = 1;j <= m;j++)
        {
            f[i][j][0] = max(f[i - 1][j][0],f[i - 1][j][1] + w[i]);
            f[i][j][1] = max(f[i - 1][j][1],f[i - 1][j - 1][0] - w[i]);
        }
    
    //枚举一下进行多少次交易
    int res = 0;
    //最后手中一定没货
    for(int i = 0;i <= m;i++)   res = max(res,f[n][i][0]);
    printf("%d",res);
    return 0;
}

3.股票买卖 V

给定一个长度为 N N N的数组,数组中的第 i i i个数字表示一个给定股票在第 i i i天的价格。

设计一个算法计算出最大利润。在满足以下约束条件下,你可以尽可能地完成更多的交易(多次买卖一支股票):

  • 你不能同时参与多笔交易(你必须在再次购买前出售掉之前的股票)。
  • 卖出股票后,你无法在第二天买入股票 (即冷冻期为 1天)。

输入格式
第一行包含整数 N N N,表示数组长度。

第二行包含 N N N个不超过 10000 10000 10000的正整数,表示完整的数组。

输出格式
输出一个整数,表示最大利润。

数据范围
1 ≤ N ≤ 1 0 5 1≤N≤10^{5} 1N105
输入样例:

5
1 2 3 0 2

输出样例:

3

样例解释
对应的交易状态为: [买入, 卖出, 冷冻期, 买入, 卖出],第一笔交易可得利润 2 − 1 = 1 2-1 = 1 21=1,第二笔交易可得利润 2 − 0 = 2 2-0 = 2 20=2,共得利润 1 + 2 = 3 1+2 = 3 1+2=3

3.1题解

在这里插入图片描述
初始化:f[0][2] = f[0][1] = 0,f[0][0] = -INF, 因为在第0个股票一定是无货的,必定从这个位置开始转移才有效

3.2代码实现

#include<iostream>
#include<cstring>
#include<algorithm>

using namespace std;

const int N = 100010,INF = 0x3f3f3f3f;

int n;
int w[N];
int f[N][3];

int main()
{
    scanf("%d",&n);
    for(int i = 1;i <= n;i++)   scanf("%d",&w[i]);
    f[0][0]  = - INF;
    f[0][2] = f[0][1] = 0;
    
    for(int i = 1;i <= n;i++)
    {
        f[i][0] = max(f[i - 1][0],f[i - 1][2] - w[i]);
        f[i][1] = f[i - 1][0] + w[i];
        f[i][2] = max(f[i - 1][1],f[i - 1][2]);
    }
    
    cout << max(f[n][1],f[n][2]) <<endl;
    
    return 0;
}

4.设计密码

你现在需要设计一个密码 S S S S S S需要满足:

S S S的长度是 N N N
S S S只包含小写英文字母;
S S S不包含子串 T T T
例如: a b c abc abc a b c d e abcde abcde a b c d e abcde abcde的子串, a b d abd abd不是 a b c d e abcde abcde的子串。

请问共有多少种不同的密码满足要求?

由于答案会非常大,请输出答案模 1 0 9 + 7 10^{9}+7 109+7的余数。

输入格式
第一行输入整数 N N N,表示密码的长度。

第二行输入字符串 T T T T T T中只包含小写字母。

输出格式
输出一个正整数,表示总方案数模 1 0 9 + 7 10^{9}+7 109+7后的结果。

数据范围
1 ≤ N ≤ 50 , 1≤N≤50, 1N50,
1 ≤ ∣ T ∣ ≤ N , ∣ T ∣ 1≤|T|≤N,|T| 1TNT T T T的长度。

输入样例1:

2
a

输出样例1:

625

输入样例2:

4
cbc

输出样例2:

456924

4.1题解(此题较为麻烦,融合了自动机和KMP)

在这里插入图片描述
在这里插入图片描述

为什么这样的状态表示是可行的呢?
因为 S S S数组中的第 n n n位有 26 26 26个小写字母,匹配在 T T T中的位置一定存在(因为不匹配,匹配到的位置是 0 0 0),
所以把所有 f [ n ] [ 0   m − 1 ] f[n][0~m-1] f[n][0 m1]加起来即为总方案数

4.2代码实现

#include<iostream>
#include<algorithm>
#include<cstring>

using namespace std;

const int N=55,mod=1e9+7;

int f[N][N],ne[N];
char str[N];//子串

int main()
{
    int n,m;
    cin>>n>>str+1;
    m=strlen(str+1);

    for(int i=2,j=0;i<=m;i++)//求出ne数组(kmp模板)
    {
        while(j&&str[j+1]!=str[i]) j=ne[j];
        if(str[j+1]==str[i]) j++;
        ne[i]=j;
    }

    f[0][0]=1;//已经匹配了0位,且匹配的子串的位置是0时的方案数为1;(初始化)
    for(int i=0;i<n;i++)//枚举密码位
     for(int j=0;j<m;j++)//把第i位密码匹配到的子串位置都枚举一遍
     //j表示第i位密码匹配到的位置,因为不能包含子串,所以不能匹配到m这个位置
      for(char k='a';k<='z';k++)//把第i+1所有可能的字母都枚举一遍
       {
           //匹配过程:寻找当第i+1的位置是k时,并且密码已经生成了第i位,匹配的子串的位置是j时,能跳到哪个位置
           int u=j;
           while(u&&str[u+1]!=k) u=ne[u];
           if(str[u+1]==k) u++;

           if(u<m) f[i+1][u]=(f[i+1][u]+f[i][j])%mod;
           //因为是从f[i][j](i+1的位置为k)跳到f[i+1][u]这个位置,所以f[i+1][u]=f[i+1][u]+f[i][j];
           /*
           注:可能存在重边,因为j不同但ne[j]是相同的,并且k是相同的,所以此时
           f[i][j1]和f[i][j2]跳到的位置是一样的(k相同,ne[j1]=ne[j2])
           */
       }

    int res=0;
    for(int i=0;i<m;i++) res=(res+f[n][i])%mod;
    //将所有的方案数加起来即为总方案数
    printf("%d",res);

    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/963118.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

js对中文进行base64编码和解码操作,解决中文乱码问题

我使用github api的接口获取文件内容&#xff0c;然后使用atob进行解码&#xff0c;但是发现&#xff1a;乱码.......糟心啊 所以就有了我封装的方法&#xff1a; export const encode64 (str) > {// 首先&#xff0c;我们使用 encodeURIComponent 来获得百分比编码的UTF…

m1芯片macOS系统卡顿问题解决方法

m1芯片的MacBook在使用过程中会出现“假死”的情况。主要表现为鼠标转圈圈&#xff0c;很多操作都不能实现&#xff0c;不能输入文本&#xff0c;系统ui也响应十分慢&#xff0c;而资源监视却看不到很高的占用。一般出现此类情况只能关机或重启。这其中的"罪魁祸首"便…

管理类联考——逻辑——形式逻辑——汇总篇——知识点突破——形式逻辑——联言选言假言——矛盾

角度 角度——汇总——持续优化 性质 &#xff08;1&#xff09; 所有的 S 是 P 所有的S是P 所有的S是P 与 有的 S 不是 P 有的S不是P 有的S不是P矛盾 &#xff08;2&#xff09; 所有的 S 不是 P 所有的S不是P 所有的S不是P 与 有的 S 是 P 有的S是P 有的S是P 矛盾 &#xf…

Spring源码解析(十):spring整合mybatis源码

Spring源码系列文章 Spring源码解析(一)&#xff1a;环境搭建 Spring源码解析(二)&#xff1a;bean容器的创建、默认后置处理器、扫描包路径bean Spring源码解析(三)&#xff1a;bean容器的刷新 Spring源码解析(四)&#xff1a;单例bean的创建流程 Spring源码解析(五)&…

计算机毕设之基于数据可视化的智慧社区内网平台python+django+mysql(含开题+源码+部署教程)

系统阐述的是一款基于数据可视化的智慧社区内网平台的设计与实现&#xff0c;对于Python、B/S结构、MySql进行了较为深入的学习与应用。主要针对系统的设计&#xff0c;描述&#xff0c;实现和分析与测试方面来表明开发的过程。开发中使用了 django框架和MySql数据库技术搭建系…

el表达式和标签库的简单使用!!!

备工作1&#xff1a;tomcat10.1.13版本&#xff0c;可去官网下载&#xff0c;不会下载的童靴看过来&#xff1a;如何正确下载tomcat&#xff1f;&#xff1f;&#xff1f;_明天更新的博客-CSDN博客 准备工作2&#xff1a;添加架包&#xff08;需要三个架包&#xff09; jstl架包…

老师们快看过来,这里有使用ChatGPT当助教的方法

最近OpenAI官方博客发布了一篇文章How teachers are using ChatGPT&#xff08;老师们如何使用ChatGPT&#xff09;&#xff0c;讲的是老师们如何在教学中使用ChatGPT&#xff0c;其中有几个例子挺好的&#xff0c;我转述一下&#xff0c;希望对你有用。 制定教案 第一个例子…

[git]分支操作

Checkout 相当于切换到该分支&#xff0c;但是因为不能直接操作远程分支&#xff0c;会在本地同步一个完全一样的分支。 注意&#xff1a;切换分支前本地先进行提交&#xff08;addcommit&#xff09;&#xff0c;否则有可能代码会丢失。 New Branch from Selected... 创建一…

【小沐学Unity3d】3ds Max 多维子材质编辑(Multi/Sub-object)

文章目录 1、简介2、精简材质编辑器2.1 先创建多维子材质&#xff0c;后指定它2.2 先指定标准材质&#xff0c;后自动创建多维子材质 3、Slate材质编辑器3.1 编辑器简介3.2 编辑器使用 结语 1、简介 多维子材质&#xff08;Multi/Sub-object&#xff09;是为一个模形&#xff0…

信息安全检测和应用信息系统安全测试

安全测试 信息安全检测&#xff0c;为软件/信息系统出具的软件检测报告&#xff08;或第三方检测报告、软件安全测试报告&#xff09;&#xff0c;是信息系统/软件上线前都需要的测试报告。 信息安全检测的标准&#xff1a; 信息安全检测依据DB31/T272-2008《计算机信息系统…

【ICer的脚本练习】“精通各种语言的hello world!“

系列的目录说明请见&#xff1a;ICer的脚本练习专栏介绍与全流程目录_尼德兰的喵的博客-CSDN博客 前言 这一节呢主要是检查一下Linux和win环境是不是能正常的支持咱们的脚本学习&#xff0c;所以来答应各种语言的hello world!&#xff0c;毕竟打印了就是学会了٩(๑❛ᴗ❛๑)۶…

Nex.js Web 应用程序 SSG 与 SSR——选择正确的渲染方法

Next.js&#xff0c;一个流行的React框架&#xff0c;改变了开发人员构建现代Web应用程序的方式。它提供了强大的功能&#xff0c;例如服务器端渲染 &#xff08;SSR&#xff09; 和静态站点生成 &#xff08;SSG&#xff09;&#xff0c;可优化应用程序的性能和用户体验。在这…

03. 程序在内存中被CPU执行

1. 程序是什么&#xff1f; 程序是由指令和数据组成的。 当我们使用计算机运行一个程序时&#xff0c;计算机会读取程序中的指令一步步执行&#xff0c;直到达到程序结束的地方。 程序的指令&#xff1a;就像一份菜谱&#xff0c;告诉计算机按照哪些步骤来做事情。 程序的数…

【原创】H3C三层交换机的路由模式

网络拓扑图 将三层交换机当路由器使用 交换机配置 <H3C>dis stp briefMST ID Port Role STP State Protection0 GigabitEthernet1/0/1 DESI LEARNING NONE0 GigabitEthernet1/0/2 …

docker常用中间件安装

文章目录 1、前言2、中间件安装2.1、mysql2.2、gitlab容器2.3、nacos2.4、redis2.5、xxljob2.6、zipkin2.7、sentinel2.8、seata2.8.1、获取镜像2.8.2、运行容器并获取配置 2.9、rockerMQ2.9.1、rockerMQ-namesrv2.9.2、rockerMQ-broker2.9.3、rockerMQ-console 2.10、jenkins2…

java企业工程管理系统源码之提高工程项目管理软件的效率

高效的工程项目管理软件不仅能够提高效率还应可以帮你节省成本提升利润 在工程行业中&#xff0c;管理不畅以及不良的项目执行&#xff0c;往往会导致项目延期、成本上升、回款拖后&#xff0c;最终导致项目整体盈利下降。企企管理云业财一体化的项目管理系统&#xff0c;确保…

文件分片上传

概要 在日常开发中上传文件是常见的功能&#xff0c;像使用 SpringBoot 作为服务端接收上传的文件是很方便的&#xff0c;但是默认情况下 SpringBoot 为我限定了单次上传文件的大小&#xff0c;默认是1MB&#xff0c;当我们单次上传的大小超过1MB的时候就会报错&#xff0c;这…

Vmware虚拟机操作系统和本地操作系统互Ping要求、解决方式讲解

Vmware虚拟机操作系统和本地操作系统互Ping讲解 在虚拟化环境中&#xff0c;如VMware&#xff0c;虚拟机&#xff08;Virtual Machine&#xff0c;简称VM&#xff09;和本地操作系统之间进行Ping测试是一项常见的任务。Ping测试可用于检查虚拟机是否能够与本地操作系统或其他网…

wap2app 隐藏系统状态栏

一、首先创建wap2App项目 1、文件》新建》项目 2、选择Wap2App项目&#xff1a;输入项目名称、网站首页地址&#xff08;如果是本地localhost的话改为你的IP地址即可&#xff09;&#xff0c;点击创建 二、创建完wap2App项目后 隐藏系统状态栏只要修改1、2选项即可 1、找到根…

智慧工地源码 智慧大屏、手机APP、SaaS模式

一、智慧工地可以通过安全八要素来提升安全保障&#xff0c;具体措施包括&#xff1a; 1.安全管理制度&#xff1a;建立科学完善的安全管理制度&#xff0c;包括安全标准规范、安全生产手册等&#xff0c;明确各项安全管理职责和要求。 2.安全培训教育&#xff1a;对工地人…