动态规化 (Dynamic Programming) 是个计算机领域里很重要的算法,我在高中参加过三次信息学奥林匹克竞赛 (ACM),每年必有一题用动归(DP)来解答. 动态规化其实就是 把问题分解成子问题+记忆子问题求解的一个过程.
你如何教你的孩子DP是什么呢?
比如:你给你的孩子5根火柴,你的孩子数了数然后说有5根.然后你再给你的孩子1根火柴然后问一共有多少根,这时候你的孩子会马上说出6根,因为他知道已经有5根了,再加上1根就是6根.
原理就是:把问题分析成更小的问题,并分而求之,子问题的解会被保存下来这样在求解更大的问题的时候就不需要再重新把中间结果再算一遍了.
动态规化的解法经常是较优的一种解法,我们来看这么一道面试题:
给定一个正整数,将它拆分成至少两个正整数,求出这些正整数的最大乘积.比如 整数2,可以拆分成1+1, 乘积是1,当输入是10,需要分解成3+3+4,这样所得的最大乘积是36.
你会怎么解?暴力搜索?这种解法不好写,而且时间复杂度也大.可以用回溯+剪枝,但时间复杂度也相对较大,特别是当N较大的时候也会时间太久Time Exceeded.
动规解答这题就较为简单.这题并没有让你详细写出怎么拆分的方案,只需要解出最大的乘积即可.所以我们有以下的方案:
f(n) = max(f(n), max(i – j, f(i – j)) * j)) for 1 <= j < i <= n
1 2 3 4 5 6 7 8 9 10 11 12 13 14 | class Solution { public: int integerBreak(int n) { vector<int> f(n + 1, 0); f[1] = 0; f[2] = 1; for (int i = 2; i < n + 1; i ++) { for (int j = 1; j < i; j ++) { f[i] = max(f[i], max(i - j, f[i - j]) * j); } } return f[n]; } }; |
class Solution { public: int integerBreak(int n) { vector<int> f(n + 1, 0); f[1] = 0; f[2] = 1; for (int i = 2; i < n + 1; i ++) { for (int j = 1; j < i; j ++) { f[i] = max(f[i], max(i - j, f[i - j]) * j); } } return f[n]; } };
我们用 f(n) 来表示整数N拆分后的最大乘积,那么我们当我们把整数 i 分解成两部分 i – j 和 j, 那么 最大乘积是 f(i – j) * j 和 (i – j) * j 的较大值.我们需要O(N^2)循环来寻求最优拆分法.
动规在计算 f(n) 的时候会需要取决于 f(i – j)的值,这时候每个f(m) 的值就会被保存起来以供之后迭代使用.这也是一个记忆的过程.以下实现就相对简单好理解.空间复杂度也是 O(N^2)
英文: Dynamic Programming – Integer Break
PS: 据说现在互联网大厂已经不爱面动态规划的题了, 一是因为有点难, 二是感觉工作中用不上.
面试经历
- 写了十几年代码, 谷歌/Google认为我还不够Senior
- Jane Street第一轮一小时面试体验卡(伦敦软件工程师)
- Meta/Facebook四次面试经历
- 三次冲击谷歌软件工程师: 我的面试起伏录 (谷歌面试是不是一生只有三次机会?)
- 记两次伦敦抖音面试经历(Tiktok)
- 我的面试谷哥GOOGLE伦敦SRE的经验和教训
- 记Facebook的第一轮技术面试(伦敦脸书)
- 记微软Principal SE的第一轮面试
- 我的AMAZON面试经历与经验之谈(亚麻伦敦面经)
- 离伦敦脸书最近的一次 - 记FACEBOOK伦敦终面经历
面试题
- 软件工程师面试: TCP/IP协议是什么?
- 软件工程师经典面试题: 当你在浏览器的地址栏敲入google.com并按回车后发生了什么?
- 谷歌面试题: 迷宫随机生成算法
- 软件工程师数据库面试技巧之 SQL中的第二名记录
- 软件工程师面试技巧之 动态规化 - 整数拆分
- 软件工程师面试技巧之 如何检查数独的有效性
- 去年 Google 的面试题 - 打印消息
- 软件工程师面试技巧之 使用哈希表降复杂度
- 微软面试题: 三角形的面积是多少?
- 英国 IT公司 电话面试的一些技巧 (程序员)
- C/C++ 中的内存管理器(堆与栈)
- C++的 map 当键(Key)不存在的时候会发生什么?
- 随机数独游戏的算法设计 (Sudoku)
- 经典二叉树的镜像的递归算法
- 谷歌的扔鸡蛋问题
- 面经: Python 的 List 和 Dictionary 有啥区别?
- 逻辑测试系列 - 一种只有4种语句的编程语言 - (1)
- 逻辑测试系列之二 - DECR
- 逻辑测试系列之三 - SUBT
面试技巧
面试其它
- 产品设计和系统设计面的区别(Product Design vs System Design)
- 45 分钟模拟面试(编程、系统设计)+职业发展建议
- 英国和美国IT公司面试的主要区别
- 拒了甲骨文(Oracle)的 Offer
loading...
上一篇: 软件工程师面试技巧之 如何检查数独的有效性
下一篇: 从A到B再到C有多少种方法?
