【算法学习笔记】27.动态规划 解题报告 SJTU OJ 1254 传手绢

2023-11-01 08:40

本文主要是介绍【算法学习笔记】27.动态规划 解题报告 SJTU OJ 1254 传手绢,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Description

活动的时候,老师经常带着同学们一起做游戏。这次,老师带着同学们一起传手绢。

游戏规则是这样的:n个同学站成一个圆圈,其中的一个同学手里拿着手绢,当老师吹哨子时开始传,每个同学可以把手绢传给自己左右的两个同学中的一个(左右任意),当老师在此吹哨子时,游戏停止,此时,拿着手绢的那个同学要给大家表演一个节目。

abc提出一个有趣的问题:有多少种不同的传手绢方法可以使得从abc手里开始传的手绢,传了m次以后,又回到abc手里。两种传手绢方法被视作不同的方法,当且仅当这两种方法中,接到球的同学按接手绢顺序组成的序列是不同的。比如有三个同学1号、2号、3号,并假设abc为1号,手绢传了3次回到abc手里的方式有1->2->3->1和1->3->2->1,共2种。

Input Format

共一行,有两个用空格隔开的整数n,m(3<=n<=30,1<=m<=30)。

Output Foramt

共一行,有一个整数,表示符合题意的方法数。

Sample Input

3 3 

Sample Output

2

Hint

40%的数据满足:3<=n<=30,1<=m<=20 100%的数据满足:3<=n<=30,1<=m<=30

 

此问题很容易让人联想到邻接矩阵乘法含义。wiki了一下,这类问题叫做”传球问题“  

证明在这里  矩阵乘法确实好理解 但是实际应用还是稍微复杂,

这里先用一个递归方案来把这个问题进行解析。

注意到一个人传球,只能传左一位和右一位,根据这个突破点就可以写成递归函数。

/*分治法
返回 一共n个人 还剩m次传球机会 最终传到第k个人手里的路线数
k是当前拿球的人的编号 1号是最开始拿球的人
*/
int passGame(int k,int n, int m){if(m==0)//若还剩0次传球机会 整好到了原处 那就返回1 表示原来的路线是对的 否则是0return (k==1) ? 1 : 0;m--;//开始传球了 所以要减少一次传球机会//向+1传的话int ans1 = passGame((k==n ? 1 : k+1 ),n,m);//向-1传的话int ans2 = passGame((k==1 ? n : k-1 ),n,m);return ans1+ans2;}    
分治法

然后调用passGame(1,n,m)即可

会超时,这种分治法的解决方案是从顶至底的。它会有很多重复的计算,所以我们就会想到和分治法路线相反的动态规划。

动态规划的一个核心思想就是将重复子问题提前计算,从而减少计算量。这也就意味着它必须从底向上计算。

在这个背景下,我们可能直觉想到的是设置一个dp[n][m] 其中d[n][m]表示n个人 传m次 回到原点的方案数,

然后想办法建立起 d[n][m]和d[n][m-1]的关系 或者和 d[n-1][m]的关系 或者和d[n-1][m-1]的关系,从而实现把这个问题分成若干个子问题来进行DP。

但是发现找到他们之间的关系,非常困难。

我们试着换一个思路,由分治法的思维,我们知道了一个事情就是 还剩s次传球机会时,球传到第k个人手里的路线数 = 还剩s-1次传球机会时,球传到第k+1个人手里的路线数 + 还剩s-1次传球机会,球传到第k-1个人手里的路线数

PS s-1的原因是最后还要耗费一次传球机会来把球传到k手里 k+1 k-1 只是简单的表示k的左右

那么我们就可以考虑构造一个dp[][]来存储还剩m次机会 传到第k个人手里的路线数目 这个dp方案就是把分治法给颠倒了而已 思路是完全一模一样的

 

//动态规划算法
int dp[35][35]={0};/*dp[i][j]表示 传i次球 传到第j个人的路线的个数
起始点和终止点都是为1号人
*/
int dp_game(int n,int m){dp[0][1]=1;//传0次 传到1号人手中 路线只有一个//开始找状态转移方程 /*其实想法很简单 就是想d[i][j]是从哪些状态过来的 d[i][j]应该是d[i-1][j-1]和d[i-1][j+1]来的 因为j只能收到j-1和j+1的球而每次传球恰好i+1传0次球 只有1号人才能接到球 其他人都是0 不用算*/for (int i = 1; i <= m; ++i){for (int j=1; j <= n; ++j){dp[i][j] = dp[i-1][j==1 ? n: j-1 ] + dp[i-1][j==n? 1:j+1];}}return dp[m][1];
}
动态规划

 

转载于:https://www.cnblogs.com/yuchenlin/p/sjtu_oj_1254.html

这篇关于【算法学习笔记】27.动态规划 解题报告 SJTU OJ 1254 传手绢的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/321826

相关文章

Go学习记录之runtime包深入解析

《Go学习记录之runtime包深入解析》Go语言runtime包管理运行时环境,涵盖goroutine调度、内存分配、垃圾回收、类型信息等核心功能,:本文主要介绍Go学习记录之runtime包的... 目录前言:一、runtime包内容学习1、作用:① Goroutine和并发控制:② 垃圾回收:③ 栈和

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET

MyBatis编写嵌套子查询的动态SQL实践详解

《MyBatis编写嵌套子查询的动态SQL实践详解》在Java生态中,MyBatis作为一款优秀的ORM框架,广泛应用于数据库操作,本文将深入探讨如何在MyBatis中编写嵌套子查询的动态SQL,并结... 目录一、Myhttp://www.chinasem.cnBATis动态SQL的核心优势1. 灵活性与可

Android学习总结之Java和kotlin区别超详细分析

《Android学习总结之Java和kotlin区别超详细分析》Java和Kotlin都是用于Android开发的编程语言,它们各自具有独特的特点和优势,:本文主要介绍Android学习总结之Ja... 目录一、空安全机制真题 1:Kotlin 如何解决 Java 的 NullPointerExceptio

Mybatis嵌套子查询动态SQL编写实践

《Mybatis嵌套子查询动态SQL编写实践》:本文主要介绍Mybatis嵌套子查询动态SQL编写方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言一、实体类1、主类2、子类二、Mapper三、XML四、详解总结前言MyBATis的xml文件编写动态SQL

SpringBoot实现Kafka动态反序列化的完整代码

《SpringBoot实现Kafka动态反序列化的完整代码》在分布式系统中,Kafka作为高吞吐量的消息队列,常常需要处理来自不同主题(Topic)的异构数据,不同的业务场景可能要求对同一消费者组内的... 目录引言一、问题背景1.1 动态反序列化的需求1.2 常见问题二、动态反序列化的核心方案2.1 ht

golang实现动态路由的项目实践

《golang实现动态路由的项目实践》本文主要介绍了golang实现动态路由项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习... 目录一、动态路由1.结构体(数据库的定义)2.预加载preload3.添加关联的方法一、动态路由1

使用雪花算法产生id导致前端精度缺失问题解决方案

《使用雪花算法产生id导致前端精度缺失问题解决方案》雪花算法由Twitter提出,设计目的是生成唯一的、递增的ID,下面:本文主要介绍使用雪花算法产生id导致前端精度缺失问题的解决方案,文中通过代... 目录一、问题根源二、解决方案1. 全局配置Jackson序列化规则2. 实体类必须使用Long封装类3.

重新对Java的类加载器的学习方式

《重新对Java的类加载器的学习方式》:本文主要介绍重新对Java的类加载器的学习方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、介绍1.1、简介1.2、符号引用和直接引用1、符号引用2、直接引用3、符号转直接的过程2、加载流程3、类加载的分类3.1、显示

Python Selenium动态渲染页面和抓取的使用指南

《PythonSelenium动态渲染页面和抓取的使用指南》在Web数据采集领域,动态渲染页面已成为现代网站的主流形式,本文将从技术原理,环境配置,核心功能系统讲解Selenium在Python动态... 目录一、Selenium技术架构解析二、环境搭建与基础配置1. 组件安装2. 驱动配置3. 基础操作模