约瑟夫环(丢手绢问题)

文章目录

  • 问题描述
  • 思路
  • 代码实现


问题描述

1~N 个数字,从 1~m 依次报数,数到 m 的数字要被删掉,求最后剩下的数字是?


思路

第一次报数第二次报数
1n-m+1
2n-m+2
m-2n-2
m-1n-1
m被删掉了
m+11
m+22
n-1n-1-m
nn-m

通过上面的表格,我们可以发现这样的规律:

将某数字第一次报数设为 first ,第二次报数设为 second 。那么存在这样的关系:first=(second+m−1)%n+1first = (second + m - 1) \% n + 1first=(second+m1)%n+1(公式一)

为什么不是 first=(second+m)%nfirst = (second + m) \% nfirst=(second+m)%n (公式二)呢?其实一开始我确实总结出来的是公式二,但是发现有个漏洞,数字编号是从 1 开始的,而公式二的编号是从 0 开始的,具体来说,就是当 second = n-m 时,first = 0 。可以看到并不符合实际,first=n 才对。换言之,也就是如果我们从 0 开始计数,那么公式二是可用的,如何从 0 开始计数呢? 答案就是把数字序列存到数组里嘛~

因此,将公式二可以进化为 first=(second+m)%n+1first = (second + m) \% n + 1first=(second+m)%n+1(公式三),但是简单的为公式二的结果 +1 ,就导致在公式二中,本来只有 second = n-m 结果不符合实际,而在公式三中,变成了只有 second = n-m 的结果符合实际,原本没问题的都变得有问题了……这是因为我们没有做到加减均衡,只有 +1,而没有 -1 。因此公式一应运而生~

那么我们可以得到这样的规律:

当n>1时,f(n)=(f(n−1)+m−1)%n+1当 n>1 时,f(n) = (f(n-1) + m - 1) \% n + 1n>1f(n)=(f(n1)+m1)%n+1
当n=1时,f(1)=1当 n=1 时,f(1) = 1n=1f(1)=1

解释一下就是,剩最后一个数的时候直接返回最后一次报数为 1 的数,反之则需要继续删除一个数。

而当 n-1=1 时,f(n) 也就意味着,最后一次报数为 1 的数,倒数第二次报数的时候它报的是几。

那么对于 f(n-1) 的调用也就意味着,本次报数为 n 的数,下一次报数为几。

说到这里已经很清楚了,显而易见的递归思想。


代码实现

int m;
int yuesefu(int n){if(n == 1) return 1; // 最后一次报数为 1,开始回溯return (yuesefu(n-1) + m - 1) % n + 1; // f(n-1)==1的时候开始回溯,求最后一次报数为 1 的数,第一次报数为几?
}// 还可以再简化
int m;
int yuesefu(int n){return n == 1 ? 1 : (yuesefu(n-1) + m - 1) % n + 1;
}// 如果将数字序列存入数组中,则可用公式二
int m;
int yuesefu(int n){return n == 0 ? 0 : (yuesefu(n-1) + m) % n;
}

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

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

相关文章

MySQL 锁的相关知识 | lock与latch、锁的类型、简谈MVCC、锁算法、死锁、锁升级

文章目录lock与latch锁的类型MVCC一致性非锁定读(快照读)一致性锁定读(当前读)锁算法死锁锁升级lock与latch 在了解数据库锁之前,首先就要区分开 lock 和 latch。在数据库中,lock 和 latch 虽然都是锁&…

Hibernate使用原生SQL适应复杂数据查询

HQL尽管容易使用,但是在一些复杂的数据操作上功能有限。特别是在实现复杂的报表统计与计算,以及多表连接查询上往往无能为力,这时可以使用SQL(Native SQL)实现HQL无法完成的任务。 1、使用SQL查询 使用SQL查询可以通过…

MySQL 存储引擎 | MyISAM 与 InnoDB

文章目录概念innodb引擎的4大特性索引结构InnoDBMyISAM区别表级锁和行级锁概念 MyISAM 是 MySQL 的默认数据库引擎(5.5版之前),但因为不支持事务处理而被 InnoDB 替代。 然而事物都是有两面性的,InnoDB 支持事务处理也会带来一些…

MySQL 事务 | ACID、四种隔离级别、并发带来的隔离问题、事务的使用与实现

文章目录事务ACID并发带来的隔离问题幻读(虚读)不可重复读脏读丢失更新隔离级别Read Uncommitted (读未提交)Read Committed (读已提交)Repeatable Read (可重复读)Serializable (可串行化)事务的使用事务的实现Redoundo事务 事务指逻辑上的一组操作。 …

MySQL 备份与主从复制

文章目录备份主从复制主从复制的作用备份 根据备份方法的不同,备份可划分为以下几种类型: 热备(Hot Backup) : 热备指的是在数据库运行的时候直接备份,并且对正在运行的数据库毫无影响,这种方法在 MySQL 官方手册中又…

C++ 流的操作 | 初识IO类、文件流、string流的使用

文章目录前言IO头文件iostreamfstreamsstream流的使用不能拷贝或对 IO对象 赋值条件状态与 iostate 类型输出缓冲区文件流fstream类型文件模式文件光标函数tellg() / tellp()seekg() / seekp()向文件存储内容/读取文件内容string流istringstreamostringstream前言 我们在使用 …

Hibernate 更新部分更改的字段 hibernate update

Hibernate 中如果直接使用 Session.update(Object o);或则是Session.updateOrUpdate(Object o); 会把这个表中的所有字段更新一遍。 如: ExperClass4k e new ExperClass4k(); e.setTime(time); e.setQ_num(q_num); e.setK(k); if (str "finch_fix")…

C++ 类的行为 | 行为像值的类、行为像指针的类、swap函数处理自赋值

文章目录概念行为像值的类行为像指针的类概念引用计数动态内存实现计数器类的swap概念swap实现自赋值概念 行为像值的类和行为像指针的类这两种说法其实蛮拗口的,这也算是 《CPrimer》 翻译的缺点之一吧。。。 其实两者的意思分别是: 行为像值的类&am…

C++ 右值引用 | 左值、右值、move、移动语义、引用限定符

文章目录C11为什么引入右值?区分左值引用、右值引用move移动语义移动构造函数移动赋值运算符合成的移动操作小结引用限定符规定this是左值or右值引用限定符与重载C11为什么引入右值? C11引入了一个扩展内存的方法——移动而非拷贝,移动较之拷…

且谈关于最近软件测试的面试

前段时间有新的产品需要招人,安排和参加了好几次面试,下面就谈谈具体的面试问题,在面试他人的同时也面试自己。 面试问题是参与面试同事各自设计的,我也不清楚其他同事的题目,就谈谈自己设计的其中2道题。 过去面试总是…

C++ 多态 | 虚函数、抽象类、虚函数表

文章目录多态虚函数重写重定义(参数不同)协变(返回值不同)析构函数重写(函数名不同)final和override重载、重写、重定义抽象类多态的原理虚函数常见问题解析虚函数表多态 一种事物,多种形态。换…

C++ 运算符重载(一) | 输入/输出,相等/不等,复合赋值,下标,自增/自减,成员访问运算符

文章目录输出运算符<<输入运算符>>相等/不等运算符复合赋值运算符下标运算符自增/自减运算符成员访问运算符输出运算符<< 通常情况下&#xff0c;输出运算符的第一个形参是一个 非常量ostream对象的引用 。之所以 ostream 是非常量是因为向流写入内容会改变…

C++ 重载函数调用运算符 | 再探lambda,函数对象,可调用对象

文章目录重载函数调用运算符lambdalambda等价于函数对象lambda等价于类标准库函数对象可调用对象与function可调用对象function函数重载与function重载函数调用运算符 函数调用运算符必须是成员函数。 一个类可以定义多个不同版本的调用运算符&#xff0c;互相之间应该在参数数…

C++ 运算符重载(二) | 类型转换运算符,二义性问题

文章目录类型转换运算符概念避免过度使用类型转换函数解决上述问题的方法转换为 bool显式的类型转换运算符类型转换二义性重载函数与类型转换结合导致的二义性重载运算符与类型转换结合导致的二义性类型转换运算符 概念 类型转换运算符&#xff08;conversion operator&#…

Tomcat中JVM内存溢出及合理配置

Tomcat本身不能直接在计算机上运行&#xff0c;需要依赖于硬件基础之上的操作系统和一个Java虚拟机。Tomcat的内存溢出本质就是JVM内存溢出&#xff0c;所以在本文开始时&#xff0c;应该先对Java JVM有关内存方面的知识进行详细介绍。 一、Java JVM内存介绍 JVM管理两种类型的…

俄罗斯农民乘法 | 快速乘

文章目录概念概念 俄罗斯农民乘法经常被用于两数相乘取模的场景&#xff0c;如果两数相乘已经超过数据范围&#xff0c;但取模后不会超过&#xff0c;我们就可以利用这个方法来拆位取模计算贡献&#xff0c;保证每次运算都在数据范围内。 A 和 B 两数相乘的时候我们如何利用加…

Linux网络编程 | socket选项设定 及 网络信息API

文章目录读取和设置 socket 选项SO_REUSEADDRSO_RCVBUF 和 SO_SNDBUFSO_RCVLOWAT 和 SO_SNDLOWATSO_LINGER 选项网络信息APIgethostbyname 和 gethostbyaddrgetservbyname 和 getservbyportgetaddrinfogetnameinfo读取和设置 socket 选项 正如 fcntl 系统调用是控制文件描述符…

Linux | 高级I/O函数

文章目录创建文件描述符的函数pipe函数dup函数、dup2函数读取或写入数据readv函数、writev函数零拷贝sendfile函数splice函数tee函数进程间通信——共享内存mmap函数 和 munmap函数控制文件描述符fcntl函数创建文件描述符的函数 pipe函数 不再赘述&#xff0c;详情见我的另一…

分布式理论:CAP、BASE | 分布式存储与一致性哈希

文章目录分布式理论CAP定理BASE理论分布式存储与一致性哈希简单哈希一致性哈希虚拟节点分布式理论 CAP定理 一致性&#xff08;Consistency&#xff09;&#xff1a; 在分布式系统中的所有数据副本&#xff0c;在同一时刻是否一致&#xff08;所有节点访问同一份最新的数据副…

Tomcat服务器性能优化

一、概述 本文档主要介绍了Tomcat的性能调优的原理和方法。可作为公司技术人员为客户Tomcat系统调优的技术指南&#xff0c;也可以提供给客户的技术人员作为他们性能调优的指导手册。 二、调优分类 由于Tomcat的运行依赖于JVM&#xff0c;从虚拟机的角度我们把Tomcat的调整分为…