利用简洁的C语言代码解决跳台阶问题与约瑟夫环问题

 更新时间:2016年02月07日 17:10:09   作者:Zhang_H  
这篇文章主要介绍了利用简洁的C语言代码解决跳台阶问题与约瑟夫环问题的方法,跳台阶问题与约瑟夫环问题是常见的基础算法题目,需要的朋友可以参考下

跳台阶问题

题目:

一个台阶总共有 n 级,如果一次可以跳 1 级,也可以跳 2 级。

求总共有多少总跳法,并分析算法的时间复杂度。

分析:

也是比较基础的题目,通过递归可以方便的求解

代码实现如下(GCC编译通过):

#include "stdio.h"
#include "stdlib.h"
 
int function(int n);
 
int main(void)
{
  int tmp;
   
  tmp = function(5);
  printf("%3d\n",tmp);
 
  return 0;
}
 
int function(int n)
{
  if(n == 1)
    return 1;
  else if(n == 2)
    return 2;
  else  
    return function(n-1) + function(n-2);
}


约瑟夫环问题
题目:

n个数字(0,1,…,n-1)形成一个圆圈,从数字0开始,每次从这个圆圈中删除第m个数字(第一个为当前数字本身,第二个为当前数字的下一个数字)。当一个数字删除后,从被删除数字的下一个继续删除第m个数字。求处在这个圆圈中剩下的最后一个数字。

(其实说了这么多就是约瑟夫环问题)

分析:

以前学习链表的时候也见过约瑟夫环问题,当时是拿循环链表模拟整个过程来解决的,今天在网上看到一种分析。记录下来:

    题目要求最后剩下的一个数(用last表示),也就是这个数是第几个,在(0,1,…,n-1)的位置是多少。明确了题目中的信息,所以我们要对这个数进行归纳。假设知道这个数在剩下的k个数中的位置,怎么来求得它在剩余K+1个数中的位置,这样一步一步推导出它在有n个数中的位置,即为所求。为什么能这样归纳,因为这个最后剩下的数在所有删除过程中有幸存活下来,只不过每次删除了一个数,它的位置就变了,知道最后,它的位置为0(只剩一个数了)。

现在来分析删除第一个数后,last这个数的位置已之前有什么样的关系。在这n个数字中,第一个被删除的数字是(m-1)%n,为简单起见记为k。那么删除k之后的剩下n-1的数字为0,1,…,k-1,k+1,…,n-1,并且下一个开始计数的数字是k+1。相当于在剩下的序列中,k+1排到最前面,从而形成序列k+1,…,n-1,0,…k-1。

k+1    ->    0
k+2    ->    1

n-1    ->    n-k-2
0       ->    n-k-1

k-1   ->   n-2

现在我们知道了有n-1个数时last的位置,记为f(n-1,m),那么如何来求得f(n,m)关于f(n-1,m)之间的关系?用X,Y来表示,如下:

Y              X

k+1    ->    0
k+2    ->    1

n-1    ->    n-k-2
0       ->     n-k-1

k-1    ->    n-2

y=( x+k+1) %n

k = (m-1)%n

所以y=(x+m)%n,最终关系如下:

                0                              n=1
f(n,m)={
                [f(n-1,m)+m]%n     n>1

根据关系可以很方便的得到代码

代码实现如下:

int LastRemaining(int n, int m)
{
  if(n < 1 || m < 1)
    return -1;
 
  int last = 0;
  for (int i = 2; i <= n; i ++) 
    last = (last + m) % i;
 
  return last;
}

相关文章

  • linux之sed命令的用法

    linux之sed命令的用法

    sed是一个很好的文件处理工具,本身是一个管道命令,主要是以行为单位进行处理,可以将数据行进行替换、删除、新增、选取等特定工作,下面先了解一下sed的用法
    2013-10-10
  • C++保存txt文件实现方法代码实例

    C++保存txt文件实现方法代码实例

    这篇文章主要介绍了C++保存txt文件实现方法代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-11-11
  • C++动态规划计算最大子数组

    C++动态规划计算最大子数组

    所谓最大子数组就是连续的若干数组元素,如果其和是最大的,那么这个子数组就称为该数组的最大子数组
    2022-06-06
  • C语言实现为无声avi视频添加wave音乐

    C语言实现为无声avi视频添加wave音乐

    这篇文章主要为大家详细介绍了C语言如何实现为无声avi视频添加wave音乐,文中的示例代码讲解详细,具有一定的参考价值,感兴趣的小伙伴可以了解一下
    2023-11-11
  • C语言中的abs()函数和exp()函数的用法

    C语言中的abs()函数和exp()函数的用法

    这篇文章主要介绍了C语言中的abs()函数和exp()函数的用法,是C语言入门学习中的基础知识,需要的朋友可以参考下
    2015-08-08
  • C++实现消消乐游戏

    C++实现消消乐游戏

    这篇文章主要为大家详细介绍了C++实现消消乐游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-05-05
  • C++深入详解单例模式与特殊类设计的实现

    C++深入详解单例模式与特殊类设计的实现

    这篇文章主要为大家详细介绍了C++单例模式和特殊类的设计,单例模式这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-06-06
  • 浅谈c++的编译和运行

    浅谈c++的编译和运行

    下面小编就为大家带来一篇浅谈c++的编译和运行。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-05-05
  • C++使用链表存储实现通讯录功能管理

    C++使用链表存储实现通讯录功能管理

    这篇文章主要为大家详细介绍了C++使用链表存储实现通讯录功能管理,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-06-06
  • c++primer类详解

    c++primer类详解

    今天小编就为大家分享一篇关于C++Primer中变量和基本类型的文章,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2021-09-09

最新评论