抱歉,未找到你想要查询的结果
  • 前端开发

    JavaScript

    JavaScript 入门教程

    本教程带你从初级到高级全面掌握 Javascript 的使用方法

    TypeScript 入门教程

    这是一个很好的简单课程,只需2小时你就可以学习TypeScript基础知识。

    Vue 入门教程

    本教程带您从零开始学习 Vue 框架的使用,让您轻松应对 Vue 项目的开发。

    Ajax 入门教程

    本教程涵盖Ajax的实现原理,及Ajax封装,最后是框架实现方法。

    ES6-10 入门教程

    对比 ES5 进行学习 ES6+,理解 ES6+ 语法背后的思想

    Yarn 入门教程

    Yarn得相关基础知识和高级进阶

    ECharts 入门教程

    从零开始学习 ECharts ,掌握 ECharts 核心内容

    HTML & CSS

    CSS3 入门教程

    本课程从盒模型、文字、颜色、过渡、动画、布局、伪类等方面介绍 CSS3 的使用。

    雪碧图入门教程

    本文详细介绍了雪碧图的由来历史以及各种使用方式

    移动端布局教程

    由于移动互联网的兴起,移动端项目占据了很大一部分比重,本章将详细讲解几种常见布局

    Html5 入门教程

    最新一代的HTML标准,增加了许多实用的特性

    Sass 入门教程

    前端项目中 Sass 的快速入门教程

    HTML 入门教程

    从零讲解 HTML,掌握基础 HTML 知识内容

    canvas 入门教程

    本教程带你从初级到高级全面掌握canvas的使用方法

    uni-app 入门教程

    从零开始学习 uni-app 框架,轻松上手应用开发

  • 服务端相关

    服务器

    Nginx 入门教程

    本教程使您掌握 Nginx 安装、配置、核心模块的详解、实际使用的能力。

    HTTP 入门教程

    从协议原理开始到 Web 服务器以及 Web 安全一网打尽

    Docker 入门教程

    从 Docker 的基础概念开始,从实际问题入手带你学习 Docker

    Shell 入门教程

    本教程由浅入深,系统性的讲解Linux Shell脚本编程。

    Linux 入门教程

    本教程从安装 Linux 开始,囊括 Linux 基础命令操作以及进阶系统管理

    开发工具

    Gradle 入门教程

    本教程使您掌握实际使用gradle进行项目构建、测试、打包、发布的能力。

    Vim 编辑器教程

    课程主要讲解Vim的安装配置,四种模式、基本操作,以及包管理工具和寄存器等内容。

    RESTful 规范教程

    本教程从什么是 REST 开始带你领略 Web 开发中无处不在的规范

    Dreamweaver 教程

    DW 是一款同时具有网页制作和网页管理功能的网站开发工具,可以快速进行网站建设

    Markdown 入门教程

    本课程涵盖 Markdown 的基本及扩展语法。

    Maven 入门教程

    从最基础的安装 Maven 开始到 Maven 在开发中的实际应用

    Eclipse 编辑器教程

    本教程从Eclipse安装开始带你轻松掌握Eclipse常用开发技巧

    GitHub 入门教程

    本教程带你轻松掌握最实用的 GitHub 知识

    Android Studio 编辑器教程

    Android Studio 编程技巧一网打尽

    PyCharm 编辑器教程

    工作经常用到的 PyCharm 编辑器使用技巧一网打尽

    Sublime Text 使用教程

    花里胡哨展示sublime编辑器的各种功能

    Postman 教程

    Postman 由Google 开发用来做接口请求测试,前后端开发人员都可以使用

    Git入门教程

    从入门到精通。

    热门服务端语言

    C 语言入门教程

    本教程从语法基础、进阶知识等各方面详解 C 语言。

    Go 入门教程

    本教程从 Go 语言的基本语法掌握到进阶编程实践

    Kotlin 教程

    从 Kotlin 的基础语法到高级特性一网打尽

    Ruby 入门教程

    本教程从 Ruby 的各种对象开始学习到 Ruby 的实际使用

    ThinkPHP 入门教程

    本教程主要讲解 ThinkPHP 框架如何上手开发应用

  • Java

    基础应用

    Java 入门教程

    深入浅出讲解 Java 语言基础知识,带你入门 Java 语言

    Android 入门教程

    为你解析最实用的 Android 技术,让你平滑上手,顺利进阶,为开发保驾护航

    算法入门教程

    分析讲解常见算法的思想及使用

    数据结构入门教程

    通俗易懂的带你了解 Java 数据结构

    Lambda 表达式教程

    本教程展现了Lambda表达式的基础语法以及在程序中的应用

    Java 并发原理入门教程

    本教程为Java并发原理入门教程,在Java程序开发中占据着举足轻重的地位

    设计模式入门教程

    带你分析最常见的九个设计模式

    Java并发工具

    本课程简洁明了展示最基本的并发工具类相关概念及应用方法。

    JVM 入门教程

    JVM 入门教程,对JVM结构进行分模块讲解,简单易懂。

    RabbitMQ 入门教程

    超系统的RabbitMQ基础知识课程,你还在等什么?

    网络编程入门教程

    Java 网络编程核心要点详解

    后端通用面试教程

    带你系统梳理后端高频面试题,轻松丰富你的校招&社招阶段

    框架应用

    Spring Boot 入门教程

    循序渐进讲解 Spring Boot 企业级应用开发

    Spring 入门教程

    通俗易懂 渐进式讲解 Spring 企业级开发应用

    Hibernate 入门教程

    由浅入深讲解 Hibernate 企业级 JDBC 应用框架

    MyBatis 入门教程

    本教程整理出"百分之二十"的知识,帮你办到"百分之八十"事情

    Spring MVC 入门教程

    通俗易懂讲解 Spring MVC 框架应用

    Swagger 入门教程

    本课程以图文并茂的方式带你学习 Swagger 核心知识和应用剖析

    Zookeeper 入门教程

    由浅入深的 学习 ZooKeeper 的基本使用以及高级使用

    Netty 教程

    由浅入深的讲解 Netty 的核心知识体系,快速上手使用和理解 Netty

    Spring Security

    本课程涵盖了 Spring Security 框架的基本原理和集成方法

    微服务

    Spring Cloud Hystrix

    系统介绍 Hystrix 支持特性与实际应用场景实战

  • Python

    基础应用

    Python 入门语法教程

    本教程带你从 Python 的基础语法开始学习 Python。

    Python 原生爬虫教程

    本教程从爬虫基础知识到进阶技巧到实际应用。

    Python 进阶应用教程

    本教程涵盖 Python 的面向对象、标准库解析、异常处理直至最后的领域应用

    Python 算法入门教程

    用 Python 代码实现常用算法并汲取算法核心思想。

    进阶方向应用

    Django 入门教程

    从 Web 基础到 Django 框架的实际开发应用

    Flask 框架教程

    Flask 框架快速入门实现一个 TodoList 功能

    NumPy 入门教程

    本教程从基础的数据类型开始到 NumPy 的高级应用一网打尽

    Scrapy 入门教程

    从爬虫基础开始到使用 Scrapy 框架抓取各大网站数据

    TensorFlow 入门教程

    通过本教程对 TensorFlow 框架快速入门

    Python 办公自动化教程

    本教程带你使用Python快速操作Excel、Word、PPT,处理各种文件

    Pandas 入门教程

    本教程从基础的数据类型开始到 Pandas 的高级应用一-网打尽

  • 数据库

    MySQL

    MySQL 入门教程

    本教程主要讲解 MySQL 增删改查等基础操作

    SQL 入门教程

    本教程讲解使用 SQL 访问和处理数据系统中的数据的方法。

    MySQL 进阶教程

    那些你还不理解的 MySQL 高阶特性一网打尽

贪心算法介绍

接下来的三个小节内容我们将介绍基础算法中的一个经典算法:贪心算法。贪心算法比较直观,有时候能很好的解决问题,但有时候求出的结果非最优解。所以在使用贪心算法时一定要对其应用场景以及相应的问题掌握清楚,方能在算法题解中游刃有余。

1. 贪心算法介绍

所谓贪心算法是指在对问题求解时,总是选择在当前看来是最好的方法。即从局部考察最优而非整体。贪心算法没有固定的算法框架,算法设计的关键是贪心策略的选择。必须注意的是:贪心算法不是对所有问题都能得到整体最优解,选择的贪心策略必须具备无后效性 (即某个状态以后的过程不会影响以前的状态,只与当前状态有关)。 所以,对所采用的贪心策略一定要仔细分析其是否满足无后效性。

2. 贪心算法思路介绍

对于贪心算法,其实现思路一般如下:

  • 建立数学模型来描述问题;
  • 把求解的问题分成若干个子问题;
  • 对每个子问题求解,得到子问题的局部最优解;
  • 把子问题的解局部最优解合成原来问题的一个解。

3. 要不要使用贪心算法?

使用贪心算法在很多问题上不一定能求得最优解,因此我们再使用贪心算法时需要严格思考使用贪心的策略能否达到我们想要的结果。贪心策略适用的前提是:局部最优策略能导致产生全局最优解。实际上,贪心算法适用的情况很少,一般对一个问题分析是否适用于贪心算法,可以先选择该问题下的几个实际数据进行分析,就可以做出判断。

贪心选择性质是指所求问题的整体最优解可以通过一系列局部最优的选择,即当考虑做何种选择的时候,我们只考虑对当前问题最佳的选择而不考虑子问题的结果。这是贪心算法可行的第一个基本要素。贪心算法以迭代的方式作出相继的贪心选择,每作一次贪心选择就将所求问题简化为规模更小的子问题。对于一个具体问题,要确定它是否具有贪心选择性质,必须证明每一步所作的贪心选择最终导致问题的整体最优解。 当一个问题的最优解包含其子问题的最优解时,称此问题具有最优子结构性质。问题的最优子结构性质是该问题可用贪心算法求解的关键特征。

在介绍完贪心相关的概念、实现思路、缺陷以及相关适用场景后,我们介绍一个关于贪心算法的典型应用场景,帮助我们更好的理解贪心的实现,便于后面完成相应的题解。

4. 贪心算法的典型应用

这是一个非常经典的编程题,题名为:洞中取宝,题目的描述如下:

假设山洞中有 n 种宝物,每种宝物有一定重量 w 和相应的价值 v,毛驴运载能力有限,只能运走 m 重量的宝物,一种宝物只能拿一样,宝物可以分割。那么怎么才能使毛驴运走宝物的价值最大呢?

注意:由于宝物有重量和价值两种特征,我们的贪心是应该取价值还是重量呢?还是二者兼顾?这样会产生三种贪心策略:

  • 每次挑选价值最大的宝物,直到无法运载下一个宝物为止;
  • 每次挑选重量最轻的宝物,直到无法运载下一个宝物为止;
  • 每次挑选价值/重量 最大的宝物,直到无法运载下一个宝物为止;

很明显,我们贪心的策略应该选择第三种,这样能兼顾宝物价值和重量,找到满足要求的最优解:

  • $ts[i]\:表示第 i 个宝物的价值 i=1...ni=1...n;
  • ws[i]ws[i]:表示第 i 个宝物的重量 i=1...ni=1...n;
  • tt:表示毛驴最大运载能力。

此时我们需要计算宝物的单位价值:
pre_value[i]=ts[i]ws[i],i=1,,n pre\_value[i]=\frac{ts[i]}{ws[i]},i=1,\cdots,n

5. 贪心算法的 Python 代码实现

有了宝物的单位价值之后,接下来我们需要对宝物按照单位价值进行排序,使用 python 自带的 sort 方法即可。注意要保持宝物的价值和总量一起变化,所以需要将宝物的重量、价值以及单位价值作为一个整体,然后进行排序:

pre_value = []
for i in range(len(ts)):
 pre_value.append([ts[i], ws[i], ts[i] / ws[i]])
# 按照单位价值从大到小进行排序
pre_value.sort(key=lambda d: d[2], reverse=True)

接下来就是按照贪心算法从大到小选择,直到宝物价值超出剩余的运载价值,然后将宝物切割带走:

for i in range(len(pre_value)):
 if pre_value[i][1] <= t:
 # 从大到小开始装
 w += pre_value[i][0]
 # 每装一个运载能力减去宝物重量
 t -= pre_value[i][1]
 else:
 # 可以取走宝物的一部分,达到最大装载
 w += pre_value[i][2] * t
 break

整体给出洞中取宝问题的贪心实现,如下:

def get_treasures(ts, ws, t):
 """ 
 ts: 宝物价值
 ws: 宝物重量
 t:最大运载能力
 """
 
 w = 0
 # 计算宝物的性价比
 pre_value = []
 for i in range(len(ts)):
 pre_value.append([ts[i], ws[i], ts[i] / ws[i]])
 # 将宝物按单位价值从大到小进行排序
 pre_value.sort(key=lambda d: d[2], reverse=True)
 for i in range(len(pre_value)):
 if pre_value[i][1] <= t:
 w += pre_value[i][0]
 t -= pre_value[i][1]
 else:
 # 可以取走宝物的一部分,达到最大装载
 w += pre_value[i][2] * t
 break
 return w
if __name__ == '__main__':
 ts = [4, 2, 9, 5, 5, 8, 5, 4, 5, 5]
 ws = [3, 8, 18, 6, 8, 20, 5, 6, 7, 15]
 t = 30
 print("运走宝物的最大价值为:{}".format(get_treasures(ts, ws, t)))

还有许多类似的问题,如背包问题、最优装载问题、教室调度问题等等,都可以使用贪心算法解决。但是我们一定要找对贪心的值,这样才能确保得到最优的解。

6. 小结

本小节我们主要介绍了贪心算法以及相应的解题方式。接着介绍了贪心算法的一个典型应用,同时也指出了贪心算法在一些问题上的问题,比如无法得到最优解等。本小节主要是理解贪心算法以及掌握解题思路,接下来的两小节将进入实战训练,彻底理解和掌握贪心算法。

  • 划线
  • 写笔记
  • 复制

0/1000

· 最近更新于 请填写更新时间
使用手机查看
最近更新
向你推荐
更多
索引目录

AltStyle によって変換されたページ (->オリジナル) /