数学建模社区-数学中国

标题: 算法越学越扎心,有没啥破解之法? [打印本页]

作者: 杨利霞    时间: 2020-4-24 17:58
标题: 算法越学越扎心,有没啥破解之法?

7 @, }: m1 H! _! S; b算法越学越扎心,有没啥破解之法?
( d6 Y* l  X/ I2 o  ~算法越学越扎心,有没啥破解之法?
% O. K; u0 C* t. k& ]2 W, A
" H/ z# y4 w% _9 U- c' F$ y对于算法的学习,我也是从一个小白一步步走来,当然,现在仍然很菜,,,不过,鉴于我觉得还有一些人比我更菜了,我决定谈谈我算法学习过程走过的坑,以及自己总结的一些经验。
% K$ k: S: b$ c2 M: I5 _# a, j2 M$ a& [0 j( p. M; W0 t, {$ C' R
切勿盲目刷题:刷题前的知识积累
* h$ `& K9 q) L1 Z6 C7 L0 E& G' }
2 n  q5 f5 k1 J- b/ f说实话,想要提高自己的算法,真的没啥捷径,我觉得最好的捷径就是脚踏实地着多动手去刷题,多刷题。! A/ S; n& b4 n4 ~& U" Q

% i) g2 S9 O; [. Y' D: y但是,我必须提醒的是,如果你是小白,也就是说,你连常见的数据结构,如链表、树以及常见的算法思想,如递归、枚举、动态规划这些都没学过,那么,我不建议你盲目疯狂着去刷题的。而是先去找本书先去学习这些必要的知识,然后再去刷题。3 ~6 }& \6 X3 \6 }7 O, V3 W

4 @7 O7 A  q9 K/ w; s2 W7 t因为,如果这些基础都不懂的话,估计一道题做了几个小时,然后看答案都看不懂,做题没有任何思路,这是很难受的。久而久之,估计没啥动力了,我刚开始就是这样,一道题答案看一天,然而还是不大懂,什么回溯啊,暴力啊,还不知道是啥意思。; g" D" i+ l! M+ K9 }

5 d' T4 Z5 a8 I, ]也就是说,假如你要去诸如leetcode这些网站刷题,那么,你要先具备一定的基础,这些基础包括:
7 X! E6 |  i- d$ b+ I) y$ s% E/ |: O( P/ Q- d: J; [/ l: t# T
1、常见数据结构:链表、树(如二叉树)。(是的,链表和二叉树是重点,图这些可以先放着)' o4 q0 m, G% s/ ^2 O. j" n6 A

6 v1 _, q9 y8 r8 G5 Y) \- z2、常见算法思想:贪婪法、分治法、穷举法、动态规划,回溯法。(贪婪、穷举、分治是基础,动态规划有难度,可以先放着)
) p3 i/ u7 Q! `" w2 ~: N1 t/ E6 T
以上列出来的算是最基本的吧。就是说你刷题之前,要把这些过一遍再去刷题。如果你连这些最基本的都不知道的话,那么你再刷题的过程中,会很难受的,思路也会相对比较少。
. |$ r6 o! P! Z$ S# P. b$ V& p' l0 N4 s/ d# M. b3 Y
总之,千万不要急,先把这些基本的过一遍,力求理解,再去刷题。
8 x' E1 e, }3 V( }
. N  p8 g/ v' q$ n& j& X9 e在这里,我推荐基本我大一时看过的书籍吧,感觉还是非常不错的,如果对于数据结构时零基础的话,那么我建议你可以看《数据结构与算法分析:C语言描述版》这本书,这本书自认为真的很 nice,当时我把这本书里面的全部都看了,并且 coding 了一遍,感觉整个人有了质的飞跃。
& @5 i4 ?$ k7 P% {7 N; U- U6 I9 S# G( t0 ^0 e4 o
后面我时在一些学校的OJ刷题,当时看的一本书叫做《挑战程序设计大赛》,日本作家写的,我觉得这本书也很nice,里面有分初级,中级和高级三个模块,基础比较差的可以从初级开始看起。2 ?$ v2 _2 ?. [& G9 i$ \( @- P4 d

, x0 L; N3 t, p当然,这两本书,你可以在这个Github上找到:https://github.com/iamshuaidi/CS-Book
9 s$ a2 ~8 S5 Z; W* |9 W总结下:
9 K( ~5 k9 G! Q' ?5 a! K' Q1 p% `/ w7 A$ b& B( D6 E
提高数据结构与算法没啥捷径,最好的捷径就是多刷题。但是,刷题的前提是你要先学会一些基本的数据结构与算法思想。, t4 M. w. x) x
; r; ?5 W" t# G' |( w
AC不是目的,我们要追求完美$ a0 I% n/ `) e2 J

2 O- U; B6 E* G6 K如何刷题?如何对待一道算法题?
" h# ?) G7 ?4 T6 k' C4 {" @! M2 A( o, K5 b- w2 `
我觉得,在做题的时候,一定要追求完美,千万不要把一道题做出来之后,提交通过,然后就赶紧下一道。我认为这意义不大,因为一道题的解法太多了,有些解法态粗糙了,我们应该要寻找最优的方法。# t: S3 |1 C3 F1 U7 [
3 M* B! ~- F9 f5 P+ n! y- `
算法能力的提升和做题的数量是有一定的关系,但并不是线性关系。也就是说,在做题的时候,要力求一题多解,如果自己实在想不出来其他办法了,可以去看看别人是怎么做的,千万不要觉得模仿别人的做法是件丢人的事。% F! m9 w/ w/ ~7 V; M' W1 M

5 v8 M$ B- n- `+ A4 U. c* z8 }! Q% [我做题的时候,我一看到一道题,可能第一想法就是用很粗糙的方式做,因为很多题采用暴力法都会很容易做,就是时间复杂度很高。之后,我就会慢慢思考,看看有没其他方法来降低时间复杂度或空间复杂度。最后,我会去看一下别人的做法,当然,并不是每道题都会这样执行。- A9 y: ~6 j; r: u% Z9 d) |

" k, c+ X5 x3 d& e2 I衡量一道算法题的好坏无非就是时间复杂度和空间复杂度,所以我们要力求完美,就要把这两个降到最低,令他们相辅相成。
6 ^( Q% Z+ h  n' e" k# ~3 F+ M: ?, y
我举道例题吧:3 O, O, [; h5 ^" a
7 W+ B- S7 F' P, w
问题: 一只青蛙一次可以跳上1级台阶,也可以跳上2级。求该青蛙跳上一个n级的台阶总共有多少种跳法?* t3 o1 X5 Y" E0 n; R* o$ \2 D* A: z
% Z5 ^1 Y, w( f: A4 {2 s! b
这道题我在以前的分章分析过,不懂的可以先看下之前写的:递归与动态规划—基础篇14 y: J) s  y1 `4 U4 _
+ p# `: P1 B' y) P/ B" N
方法1::暴力递归7 C" _0 ?; h8 i5 G# v9 T3 H

  P, |( V& \4 u0 a6 U, Q! l: t7 G这道题不难,或许你会采取下面的做法:  X# F4 Y( x4 m6 E, y! x
' ?# r& z7 s1 t
public int solve(int n){
0 w8 ^  l3 h% Z; [" J1 S    if(n <= 2){; N" l- a5 t% q! Y
        return n;
! J# o& j6 t3 c* H3 f' L+ P    }else{
0 ~5 @& F5 o$ Y, F* [; q9 |3 I        return solve(n-1) + solve(n-2);1 F4 @- b6 @. P, t5 `% ]: y
    }: _2 Z$ {* U- [# _% }1 z. t0 }  S
}8 n7 X& J8 o  o3 g2 `* l
1
* O8 e* N( O8 d" W- e2
; \$ u/ D- ^1 f! E3 ~3
' ~& ]- o0 R# D8 M4
' E8 K; P3 _6 R5; M7 }9 Q7 N7 v0 h. T1 L9 A# n
6
) n5 R! N2 t. C' P$ T! a7
; n! m' J1 }% p/ H% [这种做法的时间复杂度很高,指数级别了。但是如果你提交之后侥幸通过了,然后你就接着下一道题了,那么你就要好好想想了。' c! [0 |# Z1 [2 B
0 p, v! F" P7 W9 P% g  z' s
方法二:空间换时间
0 i7 J6 j6 A: h9 d5 W' q3 {/ C
9 g9 ]' h0 J) h2 @8 Q$ D6 ~: q6 {力求完美,我们可以考虑用空间换时间:这道题如何你去仔细想一想,会发现有很多是重复执行了。不行你可以画个图' h; D0 ~# M* Q4 O( ^- }  s
, G7 d) X" W( D8 o
所以可以采取下面的方法:
( i; Z( Z& J; Q- a: H9 r$ q( r4 o- [9 n7 p2 q& S
//用一个HashMap来保存已经计算过的状态
6 G) A/ j4 w  F( k* o: kstatic Map<Integer,Integer> map = new HashMap();
  n" ^7 o( i+ Ipublic static int solve(int n){
+ ~; g( f# z3 i6 L; N) ]6 Q         if(n <= 2){3 O6 J1 i& j0 \+ d. `
        return n;% ^: ^2 @3 L7 f3 s
    }else{//是否计算过
; k- O3 C! N) p2 y6 J1 V        if(map.containsKey(n)){
' }: l" l' j/ t            return map.get(n);+ Z9 f" S% ^: Z
        }else{
; v6 R/ z/ I: X$ N4 v# B/ E            int m = solve(n-1) + solve(n-2);
) u" l& r1 R/ i, B' q9 s9 ]            map.put(n, m);3 }! h, n: M3 ^! T3 c
            return m;+ d0 l  ~8 G. Q5 F( U* a  p8 h7 C
        }1 P, K. t* I6 y5 z% R4 E+ w
    }3 S6 n+ S- X2 m) V3 H
}: d& f% t. e& ]- U4 E
; h  q2 P5 @2 u) J* [
1
  Y$ J" L: Y, S  X0 I2
+ S' N' @8 F  i9 j1 A5 C- n2 l3
7 X  K0 w8 i1 h. F8 h5 L& ^4
/ d: o- ~8 M& X" r! ]50 w# q& g3 g( H6 f3 X1 |) z
6: u+ z* n1 X0 Z* |1 r; J
7/ `% s4 J8 s4 |3 w2 |. p3 f
8
8 Y6 F# X- D) }0 q9 u92 K$ q6 s* [) Q( u) M# \
10
  _+ d$ e$ v5 i+ D' U2 `8 L110 a$ W/ W, `# C9 ^  |
12
& v4 @9 ~" `5 v1 [7 X& B13
! Q" i/ }. k+ {14
% a6 A% }# m1 g& r  G. _+ }0 S  q151 Q0 n; B5 n5 e! {
16
; t* o0 g3 M: O8 H5 U" R这样,可以大大缩短时间。也就是说,当一道题你做了之后,发现时间复杂度很高,那么可以考虑下,是否有更好的方法,是否可以用空间换时间。
' l: a) C2 W  f' [  ^: J& C% G6 X$ G. ^1 D+ \  h' f  V% O' o
方法三:斐波那契数列
+ `6 E( `- w0 M8 u' e# B& B% ]$ p9 w: ]' Q. L
实际上,我们可以把空间复杂度弄的更小,不需要HashMap来保存状态:- `1 W% u8 d: p& i" C$ f0 {& ~4 U
1 X( M; J& r; j3 L3 H# x/ h! Y' m
public static int solve(int n){
! Z" w3 c- Z( l0 q) ^5 g6 I    if(n <= 2){
- V7 s# J& w0 O        return n;
& Y  n4 U  p! {2 Y  _  F$ E    } , Y* U$ F7 w; G& C! A
    int f1 = 0;9 [  J. r+ G5 {; _- K% O" ?: Q9 W
    int f2 = 1;
( q- [2 m0 [4 M/ X. C" [) x    int sum = 0;
- R3 ~. A4 t; @) Z/ N* h    for(int i = 1; i<= n; i++){
! O: d& R5 G3 o4 U5 D3 j* R* W, H        sum = f1 + f2;7 _7 X3 J; L/ j
        f1 = f2;
# Z: I  J! x9 b: i$ I        f2 = sum;
$ @7 f; n, s% T4 O4 h8 b5 u, N$ ^    }
- G/ t, i2 t0 P& k0 b    return sum;. d' E6 ?! N* I: x  R7 d
}
- I5 U3 ?8 s* `' g- r1
8 R) k! o) Q2 o  B2
3 i- l  R4 R6 A+ G7 ?, T9 y3; A( p; ?; K/ I! {8 s
4" j( o0 Z: j, v
5/ c8 @) Q' ]8 G9 P) f8 k3 U6 {
67 C# R: q% m, E' p: z- M
7
% ~" ~2 y' ?0 z& q. K8+ j8 k7 i& y2 d6 p
9
2 M; P2 g6 x  S" c10
/ a7 O1 T# W, A! D8 m; j) \11
8 R: |8 _+ E/ _7 k4 _' `; K12+ e  y# }8 q' w3 Z  Q% E9 ?
13' Y; ]' I( @$ }5 N- ]' e3 S0 ?* N
14" H' E# l+ C+ @5 r% I& n$ ~
我弄这道题给你们看,并不是在教你们这道题怎么做,而是有以下目的:0 g3 T  X4 E: n* F% d

1 W% z% Y4 T" m: x* T7 Z1 ?4 X1、在刷题的时候,我们要力求完美。
2 M6 l( C4 d. S( `0 E% g, ^0 l5 A# M
6 J8 K* ~$ ^! ?' E2、我想不到这些方法啊,怎么办?那么你就可以去看别人的做法,之后,遇到类似的题,你就会更有思路,更知道往哪个方向想。  P2 [; r1 q4 g( C: \+ N! D

0 W7 I$ K% v$ L$ t: l0 z% o3、可以从简单暴力入手做一道题,在考虑空间与时间之间的衡量,一点点去优化。: u* b( \' t% Q4 o$ u- @

7 S" l' c- z% N) j+ d+ K挑战自己,跳出舒适区
+ m! {! x# O$ L' K" x  @$ I' J9 h* D. `9 n2 v& p
什么叫舒适区?在刷题的时候,可能有一类题是你比较懂的,你每次一看就有思路,然后半个小时就撸好代码,提交代码,然后通过了,然后,哇,又多刷了一道题,心里很舒服。3 }, _- s5 r% e7 n

: }; |+ o5 Q: B. J0 Z但是,记住,前期你可以多刷这种题练手,提升自己的乐趣,但,我还是建议你慢慢跳出舒适区,去做一些自己不擅长的题,并且找段时间一直刷这种题。例如,我觉得我在递归方面的题还是挺强的,
$ S3 |! R) V7 l. w3 s但是,我对动态规划的题,很菜,每次都要想好久,每次遇到这种题都有点害怕,没什么信心。不过有段时间我觉得只刷动态规划的题,直接在 leetcode 选定专题,连续做了四五十道,刚开始很难受,后来就慢慢知道了套路了,一道题从两三个小时最后缩到半小时,简单的十几分钟就搞定。感觉自己对这类型的题也不惧怕的。- C8 E! b: X( J# ^

4 z. q2 @% O7 q6 {  H当然,对于动态规划的学习,大家也可以看我这篇广受好评的文章:为什么你学不过动态规划?告别动态规划,谈谈我的经验/ l" W1 C' K2 A) g9 d% x: [
) Y0 ~3 Y% X- F5 e. [/ C  s7 Q, E4 C
所以,建议你,一定要学好跳出自己的舒适区。' N0 z! c0 Z8 T/ t! O9 F

& \# f1 D5 p/ @一定要学会分类总结
, B/ A/ M  ~2 c5 }$ k
; X5 w) a7 S: R0 \有些人以为 leetcode 的题刷的越多,就一定能越厉害,其实不然,leetcode 虽然有 1000 多道题,但题型就那么几类,我们前期在刷的时候,我是建议按照题型分类刷题的,例如我这整理刷二叉树相关,然后刷链表相关,然后二分法,然后递归等等,每刷一种题型,都要研究他们的套路,如果你愿意去总结,那么 leetcode 的题,其实你刷几百道,有目的、挑选的刷,我觉得就差不多了。
& x3 m+ |/ f5 m% e! Z
4 p+ y, R; f" z2 O0 Z1 g2 I& n我看过一本书,叫做《程序员代码面试指南:IT 名企算法与数据结构题目最优解》,这本书就非常不错,里面按照栈,队列,链表,二叉树,字符串等一个专题一个专题来刷的,并且每道题都给出了最优解,而且里面的题有一定的难度,感兴趣的,真心不错,如果你把这本书的题全部搞定,并且总结相关套路,那么你的算法一定有很大的提升。
5 X! ^/ Q7 V8 G$ Y& I
# f1 C& w7 Q* I) y( z2 [推荐一些刷题网站  _% ~9 y7 b0 X* f9 y! a# E5 h
9 |* d; @& Z' K, ?7 B# i9 e! ?
我一般是在leetcode和牛客网刷题,感觉挺不错,题目难度不是很大。- g( g- r5 `* n% p# D

% d5 L& t- K, v" O# N6 A! p5 F在牛客网那里,我主要刷剑指Offer,不过那里也有个在线刷leetcode,不过里面的题量比较少。牛客网刷题有个非常方便的地方就是有个讨论区,那里会有很多大佬分享他们的解题方法,不用我们去百度找题解。所以你做完后,实在想不出,可以很方便着去看别人是怎么做的。4 S1 \0 t6 P: x# {# `' k8 Q
2 K8 o. g: z  h4 R/ k
至于leetcode,也是大部分题目官方都有给出答案,也是个不错的刷题网站。你们可以两个挑选一个,或者两个都刷。
& p+ d3 j' p  e# Q' c# o/ `- `# D
9 N% u6 `" _% c' u- R5 T当然,还有其他刷题的网站,不过,其他网站没刷过,不大清除如何。; \0 @2 S8 O/ P! M; _( j; B5 ~
8 j: M9 P# q: n
至于leetcode,有中文版和英文版
( g: W4 u# v( t1 @7 t
( y% [9 b9 D, q7 b8 W) j  Nleetcode有中文版- q$ R0 {1 R6 J. L

% Q$ \2 Y/ A; d& U& H英文版& S2 `  [* I! _6 H$ t- ^
, g/ @4 P( x! {5 R8 `" H( z
根据自己的兴趣选。
* k) ^  ], |3 `3 |* [0 F0 u2 J  L0 m
8 Y, O, l9 v/ R# v5 T4 f% _- F学习一些解题技巧
( P  Q8 F: `2 ~  O; T4 r  H; }
: }8 p" |, y2 d; ^1 o* {3 t1 p# [说实话,有些题在你没看别人的解法前,你好不知道有这么美妙优雅的解法,看了之后,卧槽,居然还可以这样。而我们在刷题的过程中,就要不断累积这些技巧,当你累计多了,你就会形成一种
2 G8 ~5 }' E& o神经反应,一下子就想到了某种方法。解题技巧很多,例如数组下标法、位图法、双指针等等,我自己也分享过一篇总结一些算法技巧的文章
2 V0 {" B+ Y7 X3 j7 ?# ?
. C1 \+ ~5 C  i% [8 g7 V推荐阅读:一些常用的算法技巧总结2 R+ x- a! D5 Y+ J; F, n0 ^7 u
1 n8 [, R2 l+ ^* {- }' H7 w0 b
例如在刷题的时候,我们要学会巧用双指针、数组下标法、位运算等等技巧来解决问题,可能会有意想不到的效果。我给你再找点我之前写文章的一些例子吧:  |1 Z2 A8 J0 H/ a7 g
1 Q0 \2 ?3 P/ H, E* s' j" L: [$ Y, ]
分享一道解法巧妙的算法题$ y, c( k& O5 D+ W9 i4 b, Y

. |' N& K, ^( S7 |) C【算法技巧】位运算装逼指南. x) R7 ?. n/ d! _  Z- g2 k

' p5 \4 l1 a% s+ c这是个长期累积的过程,我自己也精彩在我的公众号里分享一些解题的文章,感兴趣的可以关注我的公众号:帅地玩编程。$ \. [8 |8 h" l0 |
  o# U- ~( X2 J# F
再说数据结构发重要性
: v* G1 x4 o) `5 C, J
6 n1 a$ @% {5 z' ~9 J: N2 e) _& y前面我主要是说了我平时都是怎么学习算法的。在数据结构方法,我只是列举了你们一定要学习链表和树(二叉堆),但这是最基本的,刷题之前要掌握的,对于数据结构,我列举下一些比较重要的:
- D4 ^' B  Z9 }$ S, S" }: v/ q9 w0 u9 R" x8 h
1、链表(如单向链表、双向链表)。# B, H6 P0 G4 G/ V
5 K2 [9 t, e) V& A/ O7 i6 O
2、树(如二叉树、平衡树、红黑树)。) ]1 Y) C& r; V$ k/ U3 y$ R' J

" C! X* T+ h" }$ i2 E3、图(如最短路径的几种算法)。" c6 O; G) z0 j# g! {/ i# s

$ k7 i/ e8 U) x. I4、队列、栈、矩阵。
9 C, P+ P# D# Z* J0 e! e8 j7 _+ k8 U- e  L: q7 z  ]
对于这些,自己一定要动手实现一遍。你可以看书,也可以看视频,新手可以先看视频,不过前期可以看视频,之后我建议是一定要看书。
9 U* @. _, o. L3 P& E" F6 b# ~4 ?' K9 q% R9 g+ a
例如对于平衡树,可能你跟着书本的代码实现之后,过阵子你就忘记,不过这不要紧,虽然你忘记了,但是如果你之前用代码实现过,理解过,那么当你再次看到的时候,会很快就记起来,很快就知道思路,而且你的抽象能力等等会在不知不觉中提升起来。之后再学习红黑树啊,什么数据结构啊,都会学的很快。2 C. m! x/ E' G2 M

$ ^8 J% I" Q1 m对于有哪些值得学习的算法,我之前也总结过,这里推荐给大家程序员必须掌握的核心算法有哪些?,这篇文章居然 40多万阅读量了,有点受宠若惊。
* n4 t3 a* M, R$ ^0 P1 r, T. T8 v8 N% t- k% a8 j) Z; [
最最重要
8 c3 T# b( I7 Q3 _# h" P
7 N5 d: K' e8 @0 a, ]$ c/ R: v动手去做,动手去做,动手去做。重要的话说三遍。
  J% C8 _; u) j; ?& _3 Z2 K1 ]
0 T+ Y3 |; S# t* \/ s4 \千万不要找了一堆资源,订好了学习计划,我要留到某某天就来去做…
; g9 t$ o2 i# G$ a, I6 x  o( s% s$ X
. D( ?, G1 c5 }千万不要这样,而是当你激情来的时候,就马上去干,千万不要留到某个放假日啊什么鬼了,很多这种想法的人,最后会啥也没做的。0 K- S$ Q+ c/ Z; ^$ M' ]

. E/ M7 R4 }/ o/ l7 T- w, l; _也不要觉得要学习的有好多啊,不知道从哪学习起。我上面说了,可以先学习最基本的,然后刷题,刷题是一个需要长期坚持的事情,一年,两年。在刷题的过程中,可以穿插和学习其他数据结构。* r  o' W( w2 n5 H4 X

$ k- I3 g- {5 D* ^- I1 w0 l% i总结一下吧
* ?8 j9 o0 p2 b& U. [5 J9 e/ s7 Z; H1 @7 ?& w
所以我给大家的建议就是,先学习基本的数据结构以及算法思想,不要盲目刷题,接着刷题的过程中,不能得过且过,尽量追求最优解,还有就是要跳出舒适区,逼自己成长,刷题的过程中,要学会分类总结。
0 j/ m- ]% ?/ j$ W; P6 R2 k" I! u5 q7 t5 K% h+ C$ U
当然,最重要的,就是你去动手了,不然,一切免谈!
1 u+ s% M$ |) n  R1 R, O' ]4 N
' x6 U" U/ V: ~- X. e, I- f看在熬夜写过的份上,送我个赞呗,嘻嘻。# U, @: J, `8 `' z0 k
————————————————
- m- o1 B) `7 n: M! D. f版权声明:本文为CSDN博主「帅地」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
) y8 n: Y; ?- \1 |, X# S5 ]" Z+ [原文链接:https://blog.csdn.net/m0_37907797/article/details/104765116
3 c$ s& Z/ l  m0 W! }" j! i1 f3 O
$ \. N$ `3 n9 X" F% h+ J3 D% `
5 W3 d6 }; V. L




欢迎光临 数学建模社区-数学中国 (http://www.madio.net/) Powered by Discuz! X2.5