【位运算,递推】不用加减乘除做加法,构建乘积数组

面试题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;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值