面试题65:不用加减乘除做加法
写一个函数,求两个整数之和,要求在函数体内不得使用+、-、×、÷四则运算符号。
加法本质是先各位相加不考虑进位,再把进位和前一步结果各位相加,如此反复直到不产生进位。
第①步各位相加不考虑进位在二进制情形下和异或一样;第二步进位仅当二进制是1和1时向高位产生一个进位,故可以作与运算后左移一位,如此反复直到不产生进位。
#include<bits/stdc++.h>
using namespace std;
//不用四则运算做加法
int Add(int num1, int num2) {
int sum, carry;//分别表示:不考虑进位的加和,进位
do {
sum = num1 ^ num2;//由前面分析直接用二进制异或
carry = (num1 & num2) << 1;//与运算后再左移一位
//分别存到num1和num2上,因为下次循环做的就是sum和carry的相加
num1 = sum;
num2 = carry;
} while(num2 != 0);//直到num2也就是carry进位是0为止
return num1;//这时结果已经计算好,在num1即sum里了
}
int main() {
cout<<Add(130,1888)<<endl;//2018
return 0;
}
面试题66:构建乘积数组
给定一个数组A[0, 1, …, n-1],请构建一个数组B[0, 1, …, n-1],其中B中的元素B[i] =A[0]×A[1]×… ×A[i-1]×A[i+1]×…×A[n-1]。不能使用除法。
相当于i位置抠掉,可以分成左右两个子数组C和D,而这两个子数组自上而下有递推关系。
#include<bits/stdc++.h>
using namespace std;
//构建乘积数组,A=input,B=output
void BuildProductionArray(const vector<double>& input, vector<double>& output) {
//A和B的长度
int length1 = input.size();
int length2 = output.size();
//合法性检查:A和B的长度需要一样,而且都大于1
if(length1 == length2 && length2 > 1) {
//自上而下计算C[i]数组,保存在output数组里
output[0] = 1;//C[0]=1
for(int i = 1; i < length1; ++i) {//自上而下
output[i] = output[i - 1] * input[i - 1];//C[i]=C[i-1]*A[i-1]
}
//自下而上计算D[i]数组,顺便和C[i]相乘即为结果保存在output数组里
double temp = 1;//D[length-1]=1
for(int i = length1 - 2; i >= 0; --i) {//自下而上
temp *= input[i + 1];//D[i]=D[i+1]*A[i+1]
output[i] *= temp;//乘到C[i]上
}
}
}
int main() {
double input[] = { 1, 2, 3, 4, 5 };
double output[] = { 0, 0, 0, 0, 0 };
vector<double> A(input,input+sizeof(input)/sizeof(double));
vector<double> B(output,output+sizeof(output)/sizeof(double));
BuildProductionArray(A,B);
for(vector<double>::const_iterator cit=B.cbegin();cit!=B.cend();cit++){
cout<<*cit<<" ";
}//120 60 40 30 24
return 0;
}