美团笔试题:(一)(java代码)

题目:奇数位丢弃

题目描述:

对于一个由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++;
          }
      }
   }
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值