狱史问题

问题描述:

某王国对囚犯进行大赦,让一狱吏n次通过一排锁着的n间牢房,每通过一次按所定规则转动门锁,每转动一次,原来锁着的被打开,原来打开的被锁上;通过n次后,门锁开着的,牢房中的犯人放出,否则犯人不得获释.转动门锁的规则是这样的,第一次通过牢房,要转动每一把门锁,即把全部锁打开;第二次通过牢房时,从第二间开始转动,每隔一间转动一次;第k次通过牢房,从第k间开始转动,每隔k-1 间转动一次;问通过n次后,那些牢房的锁仍然是打开的?

package nunber2;
//狱史问题
import java.util.Iterator;
import java.util.Scanner;
public class Demo1 {
	public static void main(String[] args) {
		Scanner sc=new Scanner(System.in);
		System.out.println("请输入n的值:");
		int n=sc.nextInt();
		int[] a=new int[n];//这里有n扇门
		
		for (int i = 1; i <= a.length; i++) {
			a[i-1]=1;//第一个人出来的时候把门都开着,1为真,0为假
		}
		
		for (int j = 2; j < a.length-1; j++) {//第n个人
			for (int i = 1; i < a.length; i++) {//n的倍数都能转动们
				if(i*j<=a.length)
					a[j*i]=1-a[j*i];
				else
					break;
			}
		}
		
		for (int i = 1; i < a.length; i++) {
			if(a[i]==1)
				System.out.println(i+",");//输出打印
		}
		
	}//
}//

代码的优化
其实这个问题就是一个数学问题:一开始的时候,所有的门都是开着的,那么只要在n个人走过之后,被转动奇数次门,那么这个门就一样保持开着的状态。
如何才能知道这个门被转动了奇数次还是偶数次:我们来分析知道,其实这个问题就是求一个数的因数,因数的个数就是转动门的次数。再来说说,一个数的因数一般都是成对存在的。只有当两个因数重复的时候,这两个数才不是成对的,而是成单的。
所以这个问题其实就转化为求一个数以内的平方数有多少个,分别是什么了。
以上,代码就可以优化为:

package nunber2;
//狱史问题
import java.util.Iterator;
import java.util.Scanner;
public class Demo1 {
	public static void main(String[] args) {
		Scanner sc=new Scanner(System.in);
		System.out.println("请输入n的值:");
		int n=sc.nextInt();
		int[] a=new int[n];//这里有n扇门
		
		//求一个数以内的平方数
		for (int i = 1; i <= Math.sqrt(n); i++) {
			System.out.println(i*i+",");
		}
		
	}//
}//

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值