题目:奇数位丢弃
题目描述:
对于一个由0…n的所有数按升序组成的序列,我们要进行一些筛选,每次我们取当前所有数字中从小到大的第奇数位个的数,并将其丢弃。重复这一过程直到最后剩下一个数。请求出最后剩下的数字。
输入:500
输出:255
解题思路:
第一轮:0(0),1(01),2(10),3(11)…发现末尾为0的数都会被丢弃剩下1(01),3(11),5(101),7(111) (括号内为二进制数)
第二轮:此时原来的数都落在number>>1的位置1(0),3(1),5(10),7(11)……此时发现末尾为0的依然被丢弃,则可以发现0~n中1最多的数最后会被留下,所以此题变成在[0,n]区间内找二进制1最多的数
…
public int getLast(int n) {
int count= 1;//二进制含有最少1的数
while(count< n+1) {
count= (count<<1)+1 ;//位移保证了运算速度,同时要注意运算顺序
}
return count>>1;//循环结束后count肯定大于n因此要右移一位
}
题目:
有一个二维数组(n*n),写程序实现从右上角到左下角沿主对角线方向打印。给定一个二位数组arr及题目中的参数n,请返回结果数组。
输入:
{{1,2,3,4},
{5,6,7,8},
{9,10,11,12},
{13,14,15,16}}
输出:
{4,3,8,2,7,12,1,6,11,16,5,10,15,9,14,13}
解题思路:
打印起点移动:水平4->1,垂直:1->13
本质就是边界检测,超出边界停止
public void sort(int[][] arr) {
int[] res = new int[arr.length*arr.length];//记录返回结果数组
int index = 0;//结果数组序号
int startRow = 0;//起始位置的行号
int startCol = arr.length-1;//起始位置的列号
while(startRow<arr.length) {
int row = startRow;//记录当前位置行号
int col = startCol;//记录当前位置列号
while(row<arr.length&&col<arr.length) {
//arr[row][col]为基础打印斜对角线,超出边界则退出
res[index++] = arr[row++][col++];
}
if(startCol > 0) {//移动要打印的对角线的位置
//startCol大于0说明打印起始位置还在第一行,需要沿4->1方向移动
startCol--;
}else {
//startCol小于0说明打印起始位置移动到第一列,此时需要沿1->13方向
startRow++;
}
}
}