问题:
有n盏灯,编号为1~n。第1个人把灯都打开,第2个人按下所有编号为2的倍数的灯的开关,第3个人按下所有编号为3的倍数的灯的开关。一共有k个人,问最后有哪些灯开着?(k<=n<=1000)
方法1(书上方法)
1. 首先确定灯和人的数量,并声明一个布尔变量代表灯的开关状态。
2. 嵌套循环,外循环为人的序号。内循环为灯的序号。
3. 人和灯都从1开始增加,先确定人的序数,再不断增加灯的序号,并判断灯的序号是否是人的序号的倍数,如果是,那么灯泡开关状态改变。
大致代码如下
int a=1000;
int b=800;
boolean []c=new boolean[a];
for(int i=1;i<=b;i++){
for(int j=1;j<=a;j++){
if(j%i==0){
c[j-1]=!c[j-1];
}
}
}
for(int i=0;i<c.length;i++){
if(c[i]){
System.out.println(i+1);
}
}
第二种方法
第二种方法不是通过取余判断是否是倍数而是通过加法,因为计算机计算除法要比加法复杂的多,所以用加法会快很多,亲测近30倍差距。
代码如下
long start=System.currentTimeMillis();
int light=1000;
int k=800;
boolean []Light=new boolean [light];
for(int j=1;j<=k;j++)
for(int m=j;m<=light;m+=j){
Light[m-1]=!Light[m-1];
}
}
for(int i=0;i<Light.length;i++){
if(Light[i]){
System.out.println(i+1);
}
}
如果各位有更好的算法,还望指教