如何有效地写算法题

2024-06-21 22:18
文章标签 算法 有效 地写

本文主要是介绍如何有效地写算法题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、目的

持续做算法题的目的仍然是自身能力提升。可以继续细化成三点:

  • 保持思维敏捷。非常重要,状态好才能保持对编程的热情。
  • 对基础的数据结构、查找和排序保持熟练。能解决日常开发中的性能相关问题。
  • 积累对问题域的探索。只有对问题域有足够的探索,才可能举一反三,迸发灵感。

二、方法

为了更有效地实现上面的目标。推荐用下面的方式来做题:

(1)严格使用番茄时钟进行规划

在刷题的过程中非常最容易产生挫败感,无法坚持。原因是,长时间的思考导致疲倦,多次积累的疲倦使得自己产生了 抵触记忆。以至于会下意识觉得做题就是 刻苦。推荐大家在开始之前看看《意志力》。里面指出 喜好 是会被记忆操控的,如果每次做一件事最后留下的印象都是轻松愉快的,那么人就会越来越喜欢做此事,反之厌恶。所以为了能保持做题的兴趣,务必每次要主动给自己留下好的记忆。番茄时钟能够很好地保障不会出现 长时间的思考,同时也能保障不容易疲倦。如果你已经能很熟练的使用番茄时钟,请跳过。如果你对番茄时钟的印象仍然只是20分钟休息一次。那么请继续阅读。番茄时钟有两个重点,一是通过长期的训练,让大脑习惯在一段时间内保持高效。二是通过要求每次在开始前有规划和每次结束后有总结,保障产出。当把这两点应用到做算法的过程中时,应该采取以下的方式:

[1]用一个番茄时钟对题目进行彻底的分析

目前 leetcode 上的题大致可分为两种类型:

  • 对某种复杂规则的彻底解析,很有可能要构造状态机,充分考虑边界情况。
  • 对某种数据结构及算法的应用。
  • 对数学概念、遍历、动态规划等的综合应用。

在这个分析过程中首先要大致判断出属于哪一类。在掌握了基本的数据结构和算法后,应该能很好的判断是不是属于前两类。如果判断不出说明需要回头先重新复习基本数据结构。推荐《算法》一书。不要强行刷题。算法书的每种数据结构及算法的大概思路、解决的问题以及相应的时间和空间复杂度了解之后可以再回来。

第一种情况

例子:https://leetcode.com/problems/valid-number/description/

这个番茄时钟内的目标是:

  • 理清题目背后解法要用的技术
  • 充分收集可能涉及到的边界

完成后应该有的总结是:

  • 是否理清楚要用的技术
  • 是否有不确定的地方
  • 收集到的边界是否能覆盖所有情况

如果发现在要用的技术中有不熟悉的地方,应该立即中断,开启另一个番茄时钟进行学习。切忌盲目尝试。当发现有不确定的地方时,重新开启一个番茄时钟,按照当前思路把不确定地方当成一个单独的算法问题进行解决。

[2]第二种情况

例子:https://leetcode.com/problems/reverse-pairs/

这一类题目通常采取遍历的方法一定都能找到解法。重点是找到最优解,因此需要提前有足够的数据结构的知识。数据结构可大致分为链(数组、栈、队列)、树、图。在这三类数据中要分别掌握排序和查找算法。特别是相应的时间复杂度。这类题目很好判断,通常题目中会描述了几个数据或者状态的关联的关系,然后需要你找出符合条件的某些数据。那么将题目中的关联关系转换成相应的数据结构,再使用对应算法就够了。要对数据结构的足够熟悉,才能知道如何转化。
这种情况下番茄时钟的目标是:

  • 将问题转化为对相应数据结构的问题。

总结是:

  • 需不需要分情况讨论,需要一种数据结构还是多种
  • 相应数据结构是否能完全覆盖题目问题中的所有情况

[3]第三种情况

例子:https://leetcode.com/problems/minimum-window-substring/

这一类情况最好用排除法,发现不是第一种或者第二种,那么再往这种情况下考虑。这类题的特点是通常是发散性质,刚看到题目容易有思路,但不太容易找到最优解。这种情况下,也要先判断题目子类型。

  • 如果发现题目能从遍历的角度解决问题,那么可以往遍历的优化上去想。例如是否在遍历的时候能够排除掉一些情况。或者通过排序等手段之后,能实现遍历时排除某些情况。
  • 如果发现题目中存在多种约束关系,然后求某个值,那么可以往数学方程组上去想。
  • 如果发现问题可以被递归解决,并且能够将递归方式转化成顺序方式,可以往动态规划上去想。

在这种情况下,番茄时钟的目标:

  • 判断出题目类型。

总结:

  • 是否有其他类型更适合。
  • 是否需要多种手段结合

(2)执行时的番茄时钟

当分析完之后,建议不要开始写代码,一定要休息片刻。执行阶段是对我们平时写代码状态的一种锻炼,应该非常珍惜。如果一个番茄时钟执行不完,应该拆分成多个。在这段时间中,设定的番茄时钟目标应该是:

  • 高效地验证分析阶段的思路

要实现执行高效,最重要的是养成良好的编码习惯,不要犯小错误。要始终朝着只要想清楚了,一次写好,不要调试的状态要求自己。这里常见的小错误有:

  • 拼写错误。变量命名要足够清楚,不要用单个字母或者语意不明的单词。
  • 数组边界未考虑。
  • 空值未考虑。
  • 用 Math.ceil 之类函数时未考虑清楚上下界。

调试超过写代码时间 30% 时说明状态非常有问题。在这个阶段的总结是:

  • 是否完成了对分析的验证
  • 编码过程是否足够高效

如果中间发现了分析阶段的错误或者疏漏。应该立即结束编码,休息。并且重新开启分析阶段的时钟。切忌边写边改方案。如果发现编码过程状态不够好,应该加长休息时间,或者干脆结束掉。不要给自己留下低效的映像。将任务留到第二天其实也可以检验自己第一天的思路是否足够系统化,如果是,那么第二天应该能很快的重新找回思路。

(3)任一番茄时钟结束时

一定要做好总结,特别是当没有解出题来,没有思路的时候,一定要通过结束阶段的总结来反思犯了什么错误。解出来了也一定要总结题目的特点,题目中哪些要素是解出该题的关键。不做总结的话,花掉的时间所得到的收获通常只有 50% 左右。
在题目完成后,要特别注意总结此题最后是归纳到哪种类型中,它在这种类型中的独特之处是什么。经过总结,这样题目才会变成你在此问题域中的积累。
        做好总结,让每道题都有最大的收获。一个月之后自己的状态应该会有很大变化。

(4)如何分享(写博客)

在这个仓库中进行解题分享时,建议大家就把自己番茄时钟的执行记录进行分享。最后标准的解法以及思路其实在 discussion 中都有。对他人有用的分享不是结果,而是:

  • 你在番茄时钟中是如何规划的,也就是番茄时钟的目标。
  • 你是如何分析,也就是思路。
  • 你的结论是什么,或者是你在执行时除了什么问题。
  • 你所总结出的题目的关键部分。也就是对问题域进行探索的经验。

这篇关于如何有效地写算法题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1082486

相关文章

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

Java时间轮调度算法的代码实现

《Java时间轮调度算法的代码实现》时间轮是一种高效的定时调度算法,主要用于管理延时任务或周期性任务,它通过一个环形数组(时间轮)和指针来实现,将大量定时任务分摊到固定的时间槽中,极大地降低了时间复杂... 目录1、简述2、时间轮的原理3. 时间轮的实现步骤3.1 定义时间槽3.2 定义时间轮3.3 使用时

一文教你PyCharm如何有效地添加源与库

《一文教你PyCharm如何有效地添加源与库》在使用PyCharm进行Python开发的时候,很多时候我们需要添加库或者设置源,下面我们就来和大家详细介绍一下如何在PyCharm中添加源和库吧... 在使用PyCharm进行python开发的时候,很多时候我们需要添加库或者设置源。这些操作可以帮助我们更方便

OpenManus本地部署实战亲测有效完全免费(最新推荐)

《OpenManus本地部署实战亲测有效完全免费(最新推荐)》文章介绍了如何在本地部署OpenManus大语言模型,包括环境搭建、LLM编程接口配置和测试步骤,本文给大家讲解的非常详细,感兴趣的朋友一... 目录1.概况2.环境搭建2.1安装miniconda或者anaconda2.2 LLM编程接口配置2

Linux虚拟机不显示IP地址的解决方法(亲测有效)

《Linux虚拟机不显示IP地址的解决方法(亲测有效)》本文主要介绍了通过VMware新装的Linux系统没有IP地址的解决方法,主要步骤包括:关闭虚拟机、打开VM虚拟网络编辑器、还原VMnet8或修... 目录前言步骤0.问题情况1.关闭虚拟机2.China编程打开VM虚拟网络编辑器3.1 方法一:点击还原VM

如何通过Golang的container/list实现LRU缓存算法

《如何通过Golang的container/list实现LRU缓存算法》文章介绍了Go语言中container/list包实现的双向链表,并探讨了如何使用链表实现LRU缓存,LRU缓存通过维护一个双向... 目录力扣:146. LRU 缓存主要结构 List 和 Element常用方法1. 初始化链表2.

golang字符串匹配算法解读

《golang字符串匹配算法解读》文章介绍了字符串匹配算法的原理,特别是Knuth-Morris-Pratt(KMP)算法,该算法通过构建模式串的前缀表来减少匹配时的不必要的字符比较,从而提高效率,在... 目录简介KMP实现代码总结简介字符串匹配算法主要用于在一个较长的文本串中查找一个较短的字符串(称为

通俗易懂的Java常见限流算法具体实现

《通俗易懂的Java常见限流算法具体实现》:本文主要介绍Java常见限流算法具体实现的相关资料,包括漏桶算法、令牌桶算法、Nginx限流和Redis+Lua限流的实现原理和具体步骤,并比较了它们的... 目录一、漏桶算法1.漏桶算法的思想和原理2.具体实现二、令牌桶算法1.令牌桶算法流程:2.具体实现2.1

查询SQL Server数据库服务器IP地址的多种有效方法

《查询SQLServer数据库服务器IP地址的多种有效方法》作为数据库管理员或开发人员,了解如何查询SQLServer数据库服务器的IP地址是一项重要技能,本文将介绍几种简单而有效的方法,帮助你轻松... 目录使用T-SQL查询方法1:使用系统函数方法2:使用系统视图使用SQL Server Configu

Python中的随机森林算法与实战

《Python中的随机森林算法与实战》本文详细介绍了随机森林算法,包括其原理、实现步骤、分类和回归案例,并讨论了其优点和缺点,通过面向对象编程实现了一个简单的随机森林模型,并应用于鸢尾花分类和波士顿房... 目录1、随机森林算法概述2、随机森林的原理3、实现步骤4、分类案例:使用随机森林预测鸢尾花品种4.1