P1145 约瑟夫

P1145 约瑟夫

题目描述

n个人站成一圈,从某个人开始数数,每次数到m的人就被杀掉,然后下一个人重新开始数,直到最后只剩一个人。现在有一圈人,k个好人站在一起,k个坏人站在一起。从第一个好人开始数数。你要确定一个最小的m,使得在第一个好人被杀死前,k个坏人先被杀死。

感谢yh大神指出样例数据的错误。

输入输出格式

输入格式:

一个k,0<k<14

输出格式:

一个m

输入输出样例

输入样例#1: 复制

3

输出样例#1: 复制

5

输入样例#2: 复制

4

输出样例#2: 复制

30

说明

0<k<14

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
int n,k;
int vis[29];
bool dfs(int kk){
    int sum=0,s=0;
    memset(vis,0,sizeof(vis));
    for(int i=1;i<=n;i++){
        if(sum==k)    break;
        if(!vis[i])    s++;
        if(s==kk){
            s=0;
            if(i<=k&&sum<k)    return    false;
            else{
                if(i>k)    sum++;
                vis[i]=1;
            }
        }
        if(i==n)    i=0;
    }
    return true;
}
int main(){
    scanf("%d",&k);n=k*2;
    for(int i=1;i;i++)
        if(dfs(i)){
            cout<<i;
            break;
        }
}

暴力代码

思路:剪枝后的dfs

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
int n,k,s;
int vis[300];
bool judge(int p,int m){
    int ans=(s+m)%p;
    if(ans>=k){ s=ans;return 0; }
    else return 1;
}
bool dfs(int p){
    s=0;
    for(int i=0;i<k;i++)
        if(judge(2*k-i,p))
            return false;
    return true;
}
int main(){
    scanf("%d",&k);n=k*2;
    for(int i=1;i;i++)
        if(dfs(i)){
            cout<<i+1;
            break;
        }
}
时间: 2024-10-10 08:47:18

P1145 约瑟夫的相关文章

洛谷P1145 约瑟夫 数学

洛谷P1145 约瑟夫   数学 在做这题之前最好先做一下普通的约瑟夫问题 普通的约瑟夫问题  有一种递推的做法,比如说  12345 五个数,删掉3  之后,那你就把4编号改成3   5改成4,然后继续做就行了,但是后来这样求出的编号并不是其真实的编号,而是虚的编号 然后这道题如果前k个一直没被删,那么被删除的编号一定是大于 k 的, 这样做下去一直判断下去就行了,因为要mod 然后重新编号有些麻烦,所以还是坐标还是从0开始方便安全 1 #include<cstdio> 2 using na

洛谷P1145 约瑟夫

题目描述 n个人站成一圈,从某个人开始数数,每次数到m的人就被杀掉,然后下一个人重新开始数,直到最后只剩一个人.现在有一圈人,k个好人站在一起,k个坏人站在一起.从第一个好人开始数数.你要确定一个最小的m,使得在第一个好人被杀死前,k个坏人先被杀死. 感谢yh大神指出样例数据的错误. 输入输出格式 输入格式: 一个k,0<k<14 输出格式: 一个m 输入输出样例 输入样例#1: 3 输出样例#1: 5 输入样例#2: 4 输出样例#2: 30 说明 0<k<14 分析:正解就是暴

luogu P1145 约瑟夫

题目描述 n个人站成一圈,从某个人开始数数,每次数到m的人就被杀掉,然后下一个人重新开始数,直到最后只剩一个人.现在有一圈人,k个好人站在一起,k个坏人站在一起.从第一个好人开始数数.你要确定一个最小的m,使得在第一个好人被杀死前,k个坏人先被杀死. 感谢yh大神指出样例数据的错误. 输入输出格式 输入格式: 一个k,0<k<14 输出格式: 一个m 输入输出样例 输入样例#1: 3 输出样例#1: 5 输入样例#2: 4 输出样例#2: 30 说明 0<k<14 枚举答案——检验

洛谷 P1145 约瑟夫

题目描述 n个人站成一圈,从某个人开始数数,每次数到m的人就被杀掉,然后下一个人重新开始数,直到最后只剩一个人.现在有一圈人,k个好人站在一起,k个坏人站在一起.从第一个好人开始数数.你要确定一个最小的m,使得在第一个好人被杀死前,k个坏人先被杀死. 感谢yh大神指出样例数据的错误. 输入输出格式 输入格式: 一个k,0<k<14 输出格式: 一个m 输入输出样例 输入样例#1: 3 输出样例#1: 5 输入样例#2: 4 输出样例#2: 30 说明 0<k<14 屠龙宝刀点击就送

一个不简洁的约瑟夫环解法

约瑟夫环类似模型:已知有n个人,每次间隔k个人剔除一个,求最后一个剩余的. 此解法为变种,k最初为k-2,之后每次都加1. 例:n=5,k=3.从1开始,第一次间隔k-2=1,将3剔除,第二次间隔k-1=2,将1剔除.依此类推,直至剩余最后一个元素. 核心思路:将原列表复制多份横向展开,每次根据间隔获取被剔除的元素,同时将此元素存入一个剔除列表中.若被剔除元素不存在于剔除列表,则将其加入,若已存在,则顺势后移至从未加入剔除列表的元素,并将其加入.如此重复n-1次.面试遇到的题,当时只写了思路,没

【c语言】数据结构(约瑟夫生者死者游戏的问题)

约瑟夫生者死者游戏:30个旅客同乘一条船,因为严重超载,加上风高浪大,危险万分:因此船长告诉大家,只有将全船一半的旅客投入海中,其余人才能幸免遇难.无奈,大家只得同意这种办法,并议定30个人围成一圈,由第一个人开始,依次报数,数到第9个人,就把他投入大海中,然后从他的下一个人开始从1数起,数到第9个人,再将她投入大海,如此循环,直到剩下15个人乘客为止.问哪些位置是将被扔到大海的位置. 解法有许多种,可以用数组,应为涉及到删除操作,数组(顺序线性表)比较麻烦,但不必要删除,只需要给跳船的人(元素

算法系列:约瑟夫斯问题

约瑟夫斯问题(有时也称为约瑟夫斯置换),是一个出现在计算机科学和数学中的问题.在计算机编程的算法中,类似问题又称为约瑟夫环. 有{\displaystyle n}个囚犯站成一个圆圈,准备处决.首先从一个人开始,越过{\displaystyle k-2}个人(因为第一个人已经被越过),并杀掉第k个人.接着,再越过{\displaystyle k-1}个人,并杀掉第k个人.这个过程沿着圆圈一直进行,直到最终只剩下一个人留下,这个人就可以继续活着. 问题是,给定了{\displaystyle n}和{

ytu 1067: 顺序排号(约瑟夫环)

1067: 顺序排号Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 31  Solved: 16[Submit][Status][Web Board] Description 有n人围成一圈,顺序排号.从第1个人开始报数(从1到3报数),凡报到3的人退出圈子,问最后留下的是原来的第几号的那位. Input 初始人数n Output 最后一人的初始编号 Sample Input 3 Sample Output 2 HINT Source freepro

约瑟夫环 C语言 单循环链表

/*---------约瑟夫环---------*/ /*---------问题描述---------*/ /*编号为1,2,-,n的n个人围坐一圈,每人持一个密码(正整数). 一开始任选一个正整数作为报数上限值m, 从第一个人开始自1开始顺序报数,报到m时停止. 报m的人出列,将他的密码作为新的m值,从他的下一个人开始重新从1报数, 如此下去,直至所有人全部出列为止.试设计一个程序求出列顺序.*/ /*---------问题分析---------*/ /*n个人围坐一圈,且不断有人出列,即频繁