算法学习笔记之:找出所缺的整数

题目:数组A中包含n-1[0,n-1]内的不同整数,n个数中只有一个数不在所给的数组中。设计一个找到这个数的O(n)时间的算法。除了数组A自身以外,只能利用O(1)的额外空间。

与之相似的另一个题目见《算法导论》思考题4-2:问题同上,但在这里,不能由一个单一操作来访问A中的一个完整整数,因为A中整数是以二进制表示的。我们所能用的唯一操作就是“取A[i]的第j位”,这个操作所花时间为常数。题目要求:证明如果仅用此操作,仍能在O(n)时间内找出所缺整数。

第一个题目可以想出以下几种方法:

解法一:[0,n-1]这个区间中所有整数的和不变,为n*(n-1)/2,对数组A中的所有元素求和,设为s,则丢失的整数就是n*(n-1)/2 – s

解法二:异或运算。异或是个非常神奇的运算。设所缺的整数为k[0,n-1]区间中所有n个数的异或结果为s(n),异或运算满足交换率和结合率,所以s(n)可以被看作[0,n-1]中去掉k外的另外n-1个数的异或结果s(n-1)k的异或。也即:s(n)=s(n-1)^k,我们给等式两边同时异或s(n-1),等式变成了:s(n-1)^s(n)=s(n-1)^s(n-1)^k=k。而且,很明显s(n-1)其实就是数组A中所有元素的异或。所以,解法二就是:求出[0,n-1]内所有整数的异或结果s,再求出数组A中所有元素的异或结果t,所缺的整数就是s异或t

解法三:因为A自身也有n-1个位置。可以把A作为一个散列表,这样做虽然能够得到结果,但是破坏了数组A

至于《算法导论》思考题4-2的解法,在《算法艺术与信息学竞赛》上有解答。思路简述:自然数顺序的二进制表示最低位总是01交替出现,所以,首先读取数组A中所有元素的最低二进制位,如果得到的01的个数一样多,则说明所缺整数的最低二进制位为0,否则,哪个少,所缺整数的最低二进制位就是哪个。比如,我们得到所缺整数的最低二进制位为0,那么,说明数组A中最低二进制位为1的那些整数已经与此题无干,只需要在那些最低位为0的整数中找所缺整数。所以,时间复杂度是:T(n)=T(n/2)+n,计算:

n的上取整或下取整不影响时间复杂度。即T(n)=O(n)

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
题目描述 有 $n$ 种不同面额的硬币,每种硬币的数量无限。假设硬币面额为 $a_1,a_2,...,a_n$,现在要用这些硬币来找零 $m$ 元,求最少需要的硬币个数。 样例 输入:5,[1,2,5] 输出:2 算法1 (动态规划) $O(nm)$ 很明显,这是一道动态规划的问题。 设 $f[i][j]$ 为只考虑前 $i$ 种硬币,总面值为 $j$ 元时所需的最少硬币数。 显然,对于 $f[i][j]$,我们可以选择不取第 $i$ 种硬币,此时 $f[i][j]=f[i-1][j]$。也可以选择取第 $i$ 种硬币,此时 $f[i][j]=f[i][j-a_i]+1$。 于是,状态转移方程为: $$f[i][j]=\min(f[i-1][j],f[i][j-a_i]+1)$$ 注意,当 $j<a_i$ 时,显然不能取第 $i$ 种硬币,因此 $f[i][j]=f[i-1][j]$。 最终答案即为 $f[n][m]$。 时间复杂度 状态数为 $nm$,转移复杂度为 $O(1)$,故总时间复杂度为 $O(nm)$。 参考文献 无 C++ 代码 class Solution { public: int coinChange(vector<int>& coins, int amount) { int n = coins.size(); vector<vector<int>> f(n+1,vector<int>(amount+1,0x3f3f3f3f)); for(int i=0;i<=n;++i) f[i][0] = 0; for(int i=1;i<=n;++i){ for(int j=1;j<=amount;++j){ if(j<coins[i-1]) f[i][j] = f[i-1][j]; else f[i][j] = min(f[i-1][j],f[i][j-coins[i-1]]+1); } } return f[n][amount]>=0x3f3f3f3f?-1:f[n][amount]; } }; Java 代码 class Solution { public int coinChange(int[] coins, int amount) { int n = coins.length; int[][] f = new int[n+1][amount+1]; for(int i=0;i<=n;++i) f[i][0] = 0; for(int i=1;i<=n;++i){ for(int j=1;j<=amount;++j){ if(j<coins[i-1]) f[i][j] = f[i-1][j]; else f[i][j] = Math.min(f[i-1][j],f[i][j-coins[i-1]]+1); } } return f[n][amount]>=0x3f3f3f3f?-1:f[n][amount]; } }

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值