利用简洁的C语言代码解决跳台阶问题与约瑟夫环问题_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;
}

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索c语言
, 算法
, 约瑟夫环
跳台阶
约瑟夫环 c语言、约瑟夫环问题 c语言、约瑟夫环 c语言链表、约瑟夫环 c语言 数组、c语言实现约瑟夫环,以便于您获取更多的相关知识。

时间: 2025-01-31 06:05:00

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

c语言-代码位置不同为什么会报错?【C语言基础问题】

问题描述 代码位置不同为什么会报错?[C语言基础问题] gets()语句放在第8行会报错,第10行就不会报错,菜鸟不明白是怎么回事了,求解. 代码如下: # include <stdio.h> # include <string.h> int main() { char string[100]; char c; gets(string); int i, num = 0, word = 0; //gets(string); for(i = 0; (c = string[i]) !=

c语言中数组名a和&amp;amp;a详细介绍_C 语言

最近又把学习c语言提上日程上来了~~~先把我打算看的书都写下来吧,<C语言深度剖析>,<c和指针>系类,<c语言陷阱和缺陷> 先说说a和&a的区别(有三点,三个方向):1.是a和&a的本质,都是什么类型的.2.从2维数组的角度看.3.从指针运算的角度看. 声明:虽然数组名不是指针,但是用的很像指针,我们暂且把它叫做一个指针吧. 第一个问题:int a[10];  a ,&a和&a[0] 都是分别是什么?先说明a ,&a和&

C语言中结构体(struct)的几种初始化方法_C 语言

本文给大家总结的struct数据有3种初始化方法      1.顺序      2.C风格的乱序      3.C++风格的乱序 下面通过示例代码详细介绍这三种初始化方法. 1)顺序 这种方法很常见,在一般的介绍C的书中都有介绍.顺序初始化的特点是: 按照成员定义的顺序,从前到后逐个初始化:允许只初始化部分成员:在被初始化的成员之前,不能有未初始化的成员. 示例: struct User oneUser = {10, "Lucy", "/home/Lucy"}; 2

C语言中操作进程信号的相关函数使用详解_C 语言

C语言signal()函数:设置信号处理方式头文件: #include <signal.h> 定义函数: void (*signal(int signum, void(* handler)(int)))(int); 函数说明:signal()会依参数signum 指定的信号编号来设置该信号的处理函数. 当指定的信号到达时就会跳转到参数handler 指定的函数执行. 如果参数handler 不是函数指针, 则必须是下列两个常数之一: 1.SIG_IGN 忽略参数signum 指定的信号. 2.

C语言安全编码之数组索引位的合法范围_C 语言

C语言中的数组索引必须保证位于合法的范围内! 示例代码如下: enum {TABLESIZE = 100}; int *table = NULL; int insert_in_table(int pos, int value) { if(!table) { table = (int *)malloc(sizeof(int) *TABLESIZE); } if(pos >= TABLESIZE) { return -1; } table[pos] = value; return 0; } 其中:p

Linux下C语言的fork()子进程函数用法及相关问题解析_C 语言

forkfork()函数是linux下的一个系统调用,它的作用是产生一个子进程,子进程是当前进程的一个副本,它跟父进程有一样的虚存内容,但也有一些不同点. 但是,值得注意的是,父进程调用fork()后,fork()返回的是生成的子进程(如果能顺利生成的话)的ID.子进程执行的起点也是代码中fork的位置,不同的是下面这段C语言代码展示了fork()函数的使用方法: // myfork.c #include <unistd.h> #include <stdio.h> int main

C语言求两个字符串的最长公共子串_C 语言

本文实例讲述了C语言求两个字符串的最长公共子串的方法.分享给大家供大家参考.具体实现方法如下: #include "stdio.h" #include "string.h" #include "stdlib.h" void getCommon(char str1[],char str2[],char * str3); int stringLength(char * str); void main(){ char str1[50]; char st

约瑟夫环的C语言数组解决程序

编号为1,2,...,n的n个人按顺时针方向坐在圆桌边, 每人持有一密码(正整数),从第s人开始报数, 报到第m人出列, 再从下一人开始报,  直至所有的人都出列为止 下面是用数组的方法编的C语言源程序!仅供参考,欢迎多提宝贵意见! #include <stdio.h>main(){  int n,s,password;  int a[100];  int i,j,counter1=1,counter2=0;  printf("please input the value of n:

解决C++中重定义的方法总结_C 语言

C++由于头文件重复包含了所定义的变量或者常量,编译器就会报重复定义的错误.如果你碰见这样的问题可以考虑重下面几个方面去解决: 1.在出现重定义错误的头文件加上:#ifndef FileName_H_#define FileName_H_ ....(头文件内容)#endif注意如果FileName_H_这个名字已经被使用,将会出现未定义问题(这里不讨论),这是你保证FileName_H_唯一就可以. 2.在出现重定义错误的头文件加上这一句:#pragma once 就可以解决(VS建立的类都会默