链接:https://www.nowcoder.com/questionTerminal/e953b0dc87bb43f29cb042c7a9f31598?toCommentId=3228116
来源:牛客网
小爱有一个奇怪的计数器。在第一个时刻计数器显示数字3,在接下来的每一个时刻,屏幕上的数字都会减1,直到减到1为止。
接下来,计数器会重置为上一个计数周期初始值的两倍,然后再每一个时刻减1。具体过程如下图所示:
找出规律,并打印出t时刻计数器的值。
输入描述:
输入为时刻t,一个整形数字。0<t<1e12
输出描述:
计数器显示的值。
示例1
输入
4
输出
6
题目分析:这个题我看到有人用碰的方式去做的,也就是我第一次减去3,第二次减去6,每次减去的都变成原来的2倍(位运算最擅长解决这种问题了),直到有一次小于0了,这时候就是所在的那一列了,然后再求具体数字,这种办法挺好的!贴个别人的代码:
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
long time = scanner.nextLong();
long value = 3;
while(time - value > 0){
time -= (value);
value <<= 1;
}
value -= time - 1;
System.out.println(value);
}
}
这个代码的时间复杂度为O(lgn);
然后还有一种就是利用数学公式,这个题很明显是一个等比数列前n项和这个一个东西,然后列出来公式,反推就好!
代码如下:
import java.util.;
public class Main {
public static void main(String[] args){
Scanner sc= new Scanner(System.in);
long m=sc.nextLong();
double n= Math.ceil(Math.log(m/3.0+1)/Math.log(2));
System.out.println((long)(3Math.pow(2, n-1)-m+3*(Math.pow(2, n-1)-1)+1));//里面可以合并
}
}
里面公式可以合并,合并后为:
import java.util.*;
public class Main {
public static void main(String[] args){
Scanner sc= new Scanner(System.in);
long m=sc.nextLong();
double n= Math.ceil(Math.log(m/3.0+1)/Math.log(2));
System.out.println((long)(6*Math.pow(2, n-1)-m-2));
}
}
这个方法就是直接去准确的找了,时间复杂度是O(1);