笔试算法题(14):整数二进制表示中的1 & 判定栈的push和pop序列是否对应
出题:输入一个整数,要求计算此整数的二进制表示中1的个数
分析:
- 如果整数表示为k,当其是负数的时候,使用1<<i分别检测k的每一位;当其位整数的时候,则k/2表示将其二进制表示右移一位,k%2 ==0表示其是否是偶数,如果不是则说明当前二进制表示的最右边一位为1,当k==0成立的时候移位结束;
- 另外还可以使用‘消1’的方法,如果二进制表示A为'****1000',则A-1为'****0111',也就是我们仅关注二进制表示最右边的第一个 1,这样的话A&(A-1)的结果就可以将最右边的第一个1开始往右边的所有位都清除为0,'****0000';所以没进行一次处理就消除一个 1,直到整个数字为0,则A&(A-1)进行的次数就是1的个数;
- 另外可以使用空间换时间的策略,将int所有取值所对应的1的个数存储为一个数组,则直接取数组值就可以得到;
解题:
1 int count1Binary(int k) { 2 int count=0; 3 if(k<0) { 4 /** 5 * 针对负数的情况,使用1<<i可以每次检测k的一个 6 * 位上是否为1,移动次数为sizeof(int),也可以 7 * 使用while(i){………, i=i<<1}这样当1溢出的时候 8 * 就是0,也就是结束的时候 9 * */ 10 int limit=sizeof(int)*8; 11 printf("the size of int is: %d",limit); 12 for(int i=0;i<limit;i++) { 13 if((k & (1 << i)) != 0) count++; 14 } 15 } 16 while(k>0) { 17 /** 18 * 此判断条件可以替换为 k&1 == 1 19 * */ 20 if(k%2 == 1) { 21 count++; 22 } 23 /** 24 * 移位操作可以替换为 k>>1 25 * 可以获得更高运算效率 26 * */ 27 k/=2; 28 } 29 30 return count; 31 } 32 33 int count1Binary2(int k) { 34 int count=0; 35 while(k) { 36 k=k&(k-1); 37 count++; 38 } 39 return count; 40 } 41 42 int main() { 43 printf("\n%d\n", count1Binary(-23)); 44 printf("\n%d\n", count1Binary2(-23)); 45 return 0; 46 }
出题:判断stack的push和pop序列是否对应,push和pop可能交替发生
分析:例如:push序列"1,2,3,4,5",pop序列"4,5,3,2,1"
解题:
1 /** 2 * 由于当前pop出去的元素肯定位于堆栈的栈顶,所以可以根据 3 * 这个性质,用push中的元素重建堆栈,创建tempStack,则 4 * 当前pop出去的元素要么已经位于tempStack的栈顶,要么还 5 * 在push序列中,如果是后者则需要将对应元素之前的元素都压入 6 * tempStack。所以如果上述两种情况都没有发生,则失败;如果 7 * 最终的pop序列和push序列都遍历完全,tempStack非空,则返 8 * 回true,否则返回失败。 9 * */ 10 bool judgePushPopSeq(int *push, int pushL, int *pop, int popL) { 11 MyStack *tempStack=new MyStack(); 12 int pushI=0, popI=0; 13 int temp;bool isFound; 14 while(true) { 15 isFound=false; 16 if(tempStack->peek(&temp) && temp==pop[popI]) { 17 isFound=true; 18 tempStack->pop(NULL); 19 popI++; 20 } else { 21 for(int i=pushI;i<pushL;i++) { 22 if(push[i] == pop[popI]) { 23 for(int j=pushI;j<i;j++) { 24 tempStack->push(push[j]); 25 } 26 pushI=i++; 27 popI++; 28 isFound=true; 29 break; 30 } 31 } 32 } 33 if(!isFound) { 34 delete tempStack; 35 return false; 36 } 37 if(popI==popL && pushI==pushL && tempStack->isEmpty()) { 38 delete tempStack; 39 return true; 40 } else { 41 delete tempStack; 42 return false; 43 } 44 } 45 }