Algorithm之路二十九:Divide Two Integers

题目:

给出两个整数,计算二者相除的结果,计算过程中不能用乘除和取模操作。如果得到的数字溢出,则返回MAX_INT。

思路一:

将除数的2倍,4倍,8倍……n倍全都存储到一个数组或者列表中,n * 除数 <= 被除数,然后在列表或者数组中从前往后找元素,使得元素<= 被除数,被除数 -= 元素 后,继续往后找,直到被除数小于除数的1倍为止,在期间如果发现被除数变为了0,则停止迭代。

代码一:

import java.util.ArrayList;
import java.util.List;

public class Divide_Two_Integers {

	public static int divide(int dividend, int divisor)
	{
		if(divisor == 0)
			return Integer.MAX_VALUE;
		if(divisor == - 1 && dividend == Integer.MIN_VALUE)
			return Integer.MAX_VALUE;
		if(divisor == 1)
			return dividend;
		if(divisor == -1)
			return -dividend;
		boolean flag = (dividend < 0 && divisor < 0) || (dividend > 0 && divisor > 0); 
		int divid = (dividend < 0)?dividend:-dividend;
		int divis = (divisor < 0)?divisor:-divisor;
		if(divid > divis) return 0;
		int result = 0;
		int times = 1;
		List<Integer> list = new ArrayList<>();
		System.out.println(divid);
		System.out.println(divis);
		while(divid - divis<= divis )
		{
			times <<= 1;
			divis <<= 1;
			if(divis <= Integer.MIN_VALUE >> 1)break;
		}
		int times2 = times;
		while(times2 != 1)
		{
			list.add(divis);
			divis >>= 1;
			times2 >>= 1;
		}
		list.add(divis);
		int i = 0;
		while(i < list.size())
		{
			if(divid < list.get(i))
			{
				divid -= list.get(i);
				result += times;
			}
			if(divid == list.get(i))
			{
				result += times;
				break;
			}
			times >>= 1;
			i++;
		}
		return flag?result:-result;
    }
	public static void main(String[] args)
	{
		System.out.println(divide(-2147483648,2));
	}

}

时间复杂度:

假设被除数>= n*除数,被除数 < (n+1)*除数,则在链表中存储的元素个数为(int)log(n) + 1(还有元素本身),那么要经过多少次操作才能计算出结果,就要看被除数-n*除数的大小了,最坏情况下,把列表遍历了一遍。所以最坏情况下的时间复杂度,O((int)log(n) + 1)。其实最好情况下,计算n的时候,也要计算(int)log(n) 步。

空间复杂度:

O((int)log(n) + 1)。

思路二:

相当于思路一的简化,并且节省了空间。但是执行步数要多。

代码二:

public class Divide_Two_Integers {

	public static int divide(int dividend, int divisor)
	{
		if(divisor == 0)return Integer.MAX_VALUE;
		if(divisor == - 1 && dividend == Integer.MIN_VALUE)return Integer.MAX_VALUE;
		if(divisor == 1)return dividend;
		if(divisor == -1)return -dividend;
		boolean flag = (dividend < 0 && divisor < 0) || (dividend > 0 && divisor > 0); 
		int result = 0;
		long  divid = dividend;
		divid = Math.abs(divid);
		long  divis = divisor;
		divis = Math.abs(divis);
		if(divid < divis) return 0;
		while(divid >= divis)
		{
			long temp = divis;
			int times = 1;
			while(divid - temp >= temp )
			{
				temp <<= 1;
				times <<= 1;
			}
			divid -= temp;
			result += times;
		}
		return flag?result:-result;
    }
	public static void main(String[] args)
	{
		System.out.println(divide(-2147483648,2));
	}

}


时间复杂度:

被除数 = n1 * 除数 + n2 * 除数 + …… nm * 除数 + constant,n1,n2,……nm,均为2的幂。如果nm为1的话,则constant可能不为0。整的计算过程的步骤:logn1+logn2+……lognm,所以时间复杂度就由除数和被除数的关系决定了,由于本人太菜,就不再往下计算了。

空间复杂度:

O(1)。

Python网络爬虫与推荐算法新闻推荐平台:网络爬虫:通过Python实现新浪新闻的爬取,可爬取新闻页面上的标题、文本、图片、视频链接(保留排版) 推荐算法:权重衰减+标签推荐+区域推荐+热点推荐.zip项目工程资源经过严格测试可直接运行成功且功能正常的情况才上传,可轻松复刻,拿到资料包后可轻松复现出一样的项目,本人系统开发经验充足(全领域),有任何使用问题欢迎随时与我联系,我会及时为您解惑,提供帮助。 【资源内容】:包含完整源码+工程文件+说明(如有)等。答辩评审平均分达到96分,放心下载使用!可轻松复现,设计报告也可借鉴此项目,该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的。 【提供帮助】:有任何使用问题欢迎随时与我联系,我会及时解答解惑,提供帮助 【附带帮助】:若还需要相关开发工具、学习资料等,我会提供帮助,提供资料,鼓励学习进步 【项目价值】:可用在相关项目设计中,皆可应用在项目、毕业设计、课程设计、期末/期中/大作业、工程实训、大创等学科竞赛比赛、初期项目立项、学习/练手等方面,可借鉴此优质项目实现复刻,设计报告也可借鉴此项目,也可基于此项目来扩展开发出更多功能 下载后请首先打开README文件(如有),项目工程可直接复现复刻,如果基础还行,也可在此程序基础上进行修改,以实现其它功能。供开源学习/技术交流/学习参考,勿用于商业用途。质量优质,放心下载使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值