本教程带你从初级到高级全面掌握 Javascript 的使用方法
这是一个很好的简单课程,只需2小时你就可以学习TypeScript基础知识。
本教程带您从零开始学习 Vue 框架的使用,让您轻松应对 Vue 项目的开发。
本教程涵盖Ajax的实现原理,及Ajax封装,最后是框架实现方法。
对比 ES5 进行学习 ES6+,理解 ES6+ 语法背后的思想
Yarn得相关基础知识和高级进阶
从零开始学习 ECharts ,掌握 ECharts 核心内容
本课程从盒模型、文字、颜色、过渡、动画、布局、伪类等方面介绍 CSS3 的使用。
本文详细介绍了雪碧图的由来历史以及各种使用方式
由于移动互联网的兴起,移动端项目占据了很大一部分比重,本章将详细讲解几种常见布局
最新一代的HTML标准,增加了许多实用的特性
前端项目中 Sass 的快速入门教程
从零讲解 HTML,掌握基础 HTML 知识内容
本教程带你从初级到高级全面掌握canvas的使用方法
从零开始学习 uni-app 框架,轻松上手应用开发
本教程使您掌握 Nginx 安装、配置、核心模块的详解、实际使用的能力。
从协议原理开始到 Web 服务器以及 Web 安全一网打尽
从 Docker 的基础概念开始,从实际问题入手带你学习 Docker
本教程由浅入深,系统性的讲解Linux Shell脚本编程。
本教程从安装 Linux 开始,囊括 Linux 基础命令操作以及进阶系统管理
本教程使您掌握实际使用gradle进行项目构建、测试、打包、发布的能力。
课程主要讲解Vim的安装配置,四种模式、基本操作,以及包管理工具和寄存器等内容。
本教程从什么是 REST 开始带你领略 Web 开发中无处不在的规范
DW 是一款同时具有网页制作和网页管理功能的网站开发工具,可以快速进行网站建设
本课程涵盖 Markdown 的基本及扩展语法。
从最基础的安装 Maven 开始到 Maven 在开发中的实际应用
本教程从Eclipse安装开始带你轻松掌握Eclipse常用开发技巧
本教程带你轻松掌握最实用的 GitHub 知识
Android Studio 编程技巧一网打尽
工作经常用到的 PyCharm 编辑器使用技巧一网打尽
花里胡哨展示sublime编辑器的各种功能
Postman 由Google 开发用来做接口请求测试,前后端开发人员都可以使用
从入门到精通。
本教程从语法基础、进阶知识等各方面详解 C 语言。
本教程从 Go 语言的基本语法掌握到进阶编程实践
从 Kotlin 的基础语法到高级特性一网打尽
本教程从 Ruby 的各种对象开始学习到 Ruby 的实际使用
本教程主要讲解 ThinkPHP 框架如何上手开发应用
深入浅出讲解 Java 语言基础知识,带你入门 Java 语言
为你解析最实用的 Android 技术,让你平滑上手,顺利进阶,为开发保驾护航
分析讲解常见算法的思想及使用
通俗易懂的带你了解 Java 数据结构
本教程展现了Lambda表达式的基础语法以及在程序中的应用
本教程为Java并发原理入门教程,在Java程序开发中占据着举足轻重的地位
带你分析最常见的九个设计模式
本课程简洁明了展示最基本的并发工具类相关概念及应用方法。
JVM 入门教程,对JVM结构进行分模块讲解,简单易懂。
超系统的RabbitMQ基础知识课程,你还在等什么?
Java 网络编程核心要点详解
带你系统梳理后端高频面试题,轻松丰富你的校招&社招阶段
循序渐进讲解 Spring Boot 企业级应用开发
通俗易懂 渐进式讲解 Spring 企业级开发应用
由浅入深讲解 Hibernate 企业级 JDBC 应用框架
本教程整理出"百分之二十"的知识,帮你办到"百分之八十"事情
通俗易懂讲解 Spring MVC 框架应用
本课程以图文并茂的方式带你学习 Swagger 核心知识和应用剖析
由浅入深的 学习 ZooKeeper 的基本使用以及高级使用
由浅入深的讲解 Netty 的核心知识体系,快速上手使用和理解 Netty
本课程涵盖了 Spring Security 框架的基本原理和集成方法
系统介绍 Hystrix 支持特性与实际应用场景实战
本教程带你从 Python 的基础语法开始学习 Python。
本教程从爬虫基础知识到进阶技巧到实际应用。
本教程涵盖 Python 的面向对象、标准库解析、异常处理直至最后的领域应用
用 Python 代码实现常用算法并汲取算法核心思想。
从 Web 基础到 Django 框架的实际开发应用
Flask 框架快速入门实现一个 TodoList 功能
本教程从基础的数据类型开始到 NumPy 的高级应用一网打尽
从爬虫基础开始到使用 Scrapy 框架抓取各大网站数据
通过本教程对 TensorFlow 框架快速入门
本教程带你使用Python快速操作Excel、Word、PPT,处理各种文件
本教程从基础的数据类型开始到 Pandas 的高级应用一-网打尽
本教程主要讲解 MySQL 增删改查等基础操作
本教程讲解使用 SQL 访问和处理数据系统中的数据的方法。
那些你还不理解的 MySQL 高阶特性一网打尽
本小节会以 leetcode 上的 4 道编程题来进行动态规划算法的实战,以帮助大家更好的理解和掌握动态规划算法。
给定一个数组,它的第 i 个元素是一支给定股票第 i 天的价格。
如果你最多只允许完成一笔交易(即买入和卖出一支股票一次),设计一个算法来计算你所能获取的最大利润。
注意:你不能在买入股票前卖出股票。
示例 1:
输入: [7,1,5,3,6,4]
输出: 5
解释: 在第2天(股票价格=1)的时候买入,在第5天(股票价格=6)的时候卖出,最大利润=6-1=5 。注意利润不能是 7-1 = 6, 因为卖出价格需要大于买入价格;同时,你不能在买入前卖出股票。
示例 2:
输入: [7,6,4,3,1]
输出: 0
解释: 在这种情况下, 没有交易完成, 所以最大利润为 0。
这样的典型一维 DP 问题,我们会先思考下如何定义问题的状态,如何能得到递推关系?假设我定义这样一个和股票数据相同大小的列表 F[x],表示以位置 x 结尾时能获得的最大利润,将这个作为问题的状态。容易分析其状态转移方程如下,其中 minVal 等于 nums[0:x-1] 中的最小值:
示意图如下:
图片描述
初始的状态条件自然是 F[0] = 0,接下来我们根据上面的 DP 过程完成相应的代码,如下:
class Solution:
def maxProfit(self, prices: List[int]) -> int:
n = len(prices)
if n == 0: return 0
dp = [0] * n
minVal = prices[0]
for i in range(1, n):
minVal = min(minVal, prices[i])
dp[i] = max(dp[i - 1], prices[i] - minVal)
return dp[-1]
你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。
给定一个代表每个房屋存放金额的非负整数数组,计算你不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。
示例 1:
输入: [1,2,3,1]
输出: 4
解释: 偷窃 1 号房屋 (金额 = 1) ,然后偷窃 3 号房屋 (金额 = 3)。
偷窃到的最高金额 = 1 + 3 = 4 。
示例2:
输入: [2,7,9,3,1]
输出: 12
解释: 偷窃 1 号房屋 (金额 = 2), 偷窃 3 号房屋 (金额 = 9),接着偷窃 5 号房屋 (金额 = 1)。
偷窃到的最高金额 = 2 + 9 + 1 = 12 。
这个同样也是一维 DP 问题,我们这样思考下,定义两个状态数组 F 和 M,其含义分别如下:
那这样可以得到这样的递推方程:
最后我们的结果为:
这样可以得到如下的 DP 代码:
class Solution:
def rob(self, nums):
if not nums:
return 0
# 初始条件
F = [0] * len(nums)
M = [0] * len(nums)
# 初始条件
F[0] = nums[0]
M[0] = 0
for i in range(1, len(nums)):
# 得到状态转移方程
F[i] = M[i - 1] + nums[i]
M[i] = max(F[i - 1], M[i - 1])
# 返回最后结果
return max(F[-1], M[-1])
一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为"Start" )。机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为"Finish")。问总共有多少条不同的路径?
这题和前面递归中讲解的爬楼梯问题非常相似,我们也可以和之前一样使用递归算法完成解答。但是爬楼梯以及该问题中使用递归存在大量的重复计算,我们要考虑是用 DP 方法来解决该问题。这是一个二维的网格,因此对于 DP 问题而言,我们要考虑二维 DP。定义如下二维 DP 数组 ,表示到位置的不同路径数。
有了这样一个状态的定义,来看相应的递归方程。注意到机器人只能向下或者向右移动一步,因此有:
初始状态也非常容易,有:
于是可以立马写出相应的 DP 答案:
class Solution:
def uniquePaths(self, m, n):
if not m or not n:
return 0
P = []
# 保证 P[i][0]=1,P[0][j]=1,即可。其余位置将会递推得到
for i in range(m):
P.append([1] * n)
# 动态转移方程
for i in range(1, m):
for j in range(1, n):
P[i][j] = P[i-1][j] + P[i][j-1]
return P[-1][-1]
给定两个字符串 text1 和 text2,返回这两个字符串的最长公共子序列的长度。
一个字符串的子序列是指这样一个新的字符串:它是由原字符串在不改变字符的相对顺序的情况下删除某些字符(也可以不删除任何字符)后组成的新字符串。
例如,"ace" 是 "abcde" 的子序列,但 "aec" 不是 "abcde" 的子序列。两个字符串的「公共子序列」是这两个字符串所共同拥有的子序列。
若这两个字符串没有公共子序列,则返回 0。
示例1:
输入:text1 = "abcde", text2 = "ace"
输出:3
解释:最长公共子序列是 "ace",它的长度为 3。
示例2:
输入:text1 = "abc", text2 = "abc"
输出:3
解释:最长公共子序列是 "abc",它的长度为 3。
示例3:
输入:text1 = "abc", text2 = "def"
输出:0
解释:两个字符串没有公共子序列,返回 0。
这个也是非常经典的一道面试题,它的解法也是典型的二维动态规划。我们继续按照上面的步骤来定义问题的状态数组以及推导相应的状态转移方程。
我们和前面一样用一个二维的状态矩阵来递推该问题,假设二维状态矩阵如下:
现在来看图说明该二维状态矩阵以及从中推导出状态转移方程:
图片描述
容易递推出状态转移方程:
这样我们很容易就能写出相应的 DP 代码,如下:
class Solution:
def longestCommonSubsequence(self, text1: str, text2: str) -> int:
if not text1 or not text2:
return 0
# 初始状态矩阵
P = []
for _ in range(len(text1)):
P.append([0] * len(text2))
# 编写初始状态,以下两个for循环主要是处理一些边界问题
for j in range(len(text2)):
if j == 0:
P[0][j] = 1 if text2[j] == text1[0] else 0
else:
P[0][j] = max(P[0][j-1], 1 if text2[j] == text1[0] else 0)
for i in range(len(text1)):
if i == 0:
P[i][0] = 1 if text1[i] == text2[0] else 0
else:
P[i][0] = max(P[i-1][0], 1 if text1[i] == text2[0] else 0)
# 核心部分,状态转移方程
for i in range(1, len(text1)):
for j in range(1, len(text2)):
# 上面图片中推到的状态转移方程
if text1[i] == text2[j]:
P[i][j] = P[i-1][j-1] + 1
else:
P[i][j] = max(P[i-1][j], P[i][j-1])
return P[-1][-1]
本小节中我们以 4 道代表性题目给大家展示了如何求解动态规划问题,在 leetcode 上还有许许多多更为复杂和为烧脑的动态规划题目,大家要多多练习,达到熟能生巧的地步。
0/1000