数学建模社区-数学中国

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

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

/ H/ a) ~- H; x算法越学越扎心,有没啥破解之法?
1 C7 P& C0 F' B算法越学越扎心,有没啥破解之法?: ^( q9 P, d% d- y/ q( C, M( R

, x6 p& y) H3 a3 F- _8 {1 H' ^9 d对于算法的学习,我也是从一个小白一步步走来,当然,现在仍然很菜,,,不过,鉴于我觉得还有一些人比我更菜了,我决定谈谈我算法学习过程走过的坑,以及自己总结的一些经验。6 S% A2 X  k  y$ p

0 V: m" f& i$ e; x切勿盲目刷题:刷题前的知识积累1 W) D: k3 \6 k: G) n2 l% ~
4 j+ p0 ^- p' [
说实话,想要提高自己的算法,真的没啥捷径,我觉得最好的捷径就是脚踏实地着多动手去刷题,多刷题。! E+ M, s  i; z/ g6 m

, k% L; y$ x0 T但是,我必须提醒的是,如果你是小白,也就是说,你连常见的数据结构,如链表、树以及常见的算法思想,如递归、枚举、动态规划这些都没学过,那么,我不建议你盲目疯狂着去刷题的。而是先去找本书先去学习这些必要的知识,然后再去刷题。
# C- t" Z$ p5 }' a+ I' @
, `% E% f$ m% m7 g4 l- p# Z4 P因为,如果这些基础都不懂的话,估计一道题做了几个小时,然后看答案都看不懂,做题没有任何思路,这是很难受的。久而久之,估计没啥动力了,我刚开始就是这样,一道题答案看一天,然而还是不大懂,什么回溯啊,暴力啊,还不知道是啥意思。% V- y7 c7 R! m. ^( R5 Q. M" u# C
4 [0 }8 y+ \6 D$ }! T2 U5 y6 h
也就是说,假如你要去诸如leetcode这些网站刷题,那么,你要先具备一定的基础,这些基础包括:
8 H' d% S3 N7 l
( C; q  X' W) M! x' ~4 k; C1、常见数据结构:链表、树(如二叉树)。(是的,链表和二叉树是重点,图这些可以先放着)
1 J' Z. t0 P5 A9 P3 L# ^8 Q' _
. o' s1 z2 x' t" v2 }' o/ l2、常见算法思想:贪婪法、分治法、穷举法、动态规划,回溯法。(贪婪、穷举、分治是基础,动态规划有难度,可以先放着)
7 o% L  Z# x3 L+ t# @. p" L0 h. x& \+ P
以上列出来的算是最基本的吧。就是说你刷题之前,要把这些过一遍再去刷题。如果你连这些最基本的都不知道的话,那么你再刷题的过程中,会很难受的,思路也会相对比较少。
2 O. M0 S1 t& ]0 ~
3 f" y3 o0 Y+ C! S9 t总之,千万不要急,先把这些基本的过一遍,力求理解,再去刷题。
: L: y- ^" U6 J. C  |% t' F
7 t: P! r  V7 l& s4 X$ h在这里,我推荐基本我大一时看过的书籍吧,感觉还是非常不错的,如果对于数据结构时零基础的话,那么我建议你可以看《数据结构与算法分析:C语言描述版》这本书,这本书自认为真的很 nice,当时我把这本书里面的全部都看了,并且 coding 了一遍,感觉整个人有了质的飞跃。2 D3 v+ }8 f' W& S, H. y
+ @0 V( h7 r, K' k/ l6 L
后面我时在一些学校的OJ刷题,当时看的一本书叫做《挑战程序设计大赛》,日本作家写的,我觉得这本书也很nice,里面有分初级,中级和高级三个模块,基础比较差的可以从初级开始看起。
- t8 p: j/ y. P; t  n3 l& Q+ ?5 p- H# Q2 I) t
当然,这两本书,你可以在这个Github上找到:https://github.com/iamshuaidi/CS-Book
, M# A' j0 N- {5 |) s6 s总结下:
3 G; T( l9 p- N5 P0 y; ]1 E! }; m- p
提高数据结构与算法没啥捷径,最好的捷径就是多刷题。但是,刷题的前提是你要先学会一些基本的数据结构与算法思想。  [& V) `- u6 O6 x: ^7 |8 E) ?( V7 G

" R& h1 q: N* |& J8 qAC不是目的,我们要追求完美
  v5 K2 p4 t" f4 L" o3 [+ f
6 \" E2 u5 Z3 z/ p如何刷题?如何对待一道算法题?$ V1 C1 Y' n; H$ ^+ m
6 c- G* @) ]3 O) Q
我觉得,在做题的时候,一定要追求完美,千万不要把一道题做出来之后,提交通过,然后就赶紧下一道。我认为这意义不大,因为一道题的解法太多了,有些解法态粗糙了,我们应该要寻找最优的方法。# D$ ?& _$ ]2 T0 d  ]6 f* ^
' t6 _. \* L- A& k
算法能力的提升和做题的数量是有一定的关系,但并不是线性关系。也就是说,在做题的时候,要力求一题多解,如果自己实在想不出来其他办法了,可以去看看别人是怎么做的,千万不要觉得模仿别人的做法是件丢人的事。
1 }2 [4 f3 P" y8 p4 S
9 i" z( b+ Q2 E" t! `我做题的时候,我一看到一道题,可能第一想法就是用很粗糙的方式做,因为很多题采用暴力法都会很容易做,就是时间复杂度很高。之后,我就会慢慢思考,看看有没其他方法来降低时间复杂度或空间复杂度。最后,我会去看一下别人的做法,当然,并不是每道题都会这样执行。  U( a' }& ]' [- w! W
. n8 T5 L, J- |5 A
衡量一道算法题的好坏无非就是时间复杂度和空间复杂度,所以我们要力求完美,就要把这两个降到最低,令他们相辅相成。0 n7 t/ O% a  u; F6 B

! X. V/ }; G3 T2 _# ?  I8 }* Z我举道例题吧:% i# A8 E" `5 u
- K. N) ]' `1 P; B5 V
问题: 一只青蛙一次可以跳上1级台阶,也可以跳上2级。求该青蛙跳上一个n级的台阶总共有多少种跳法?1 C$ X: d0 k3 b' ~" p: ~( s/ f' d

# t0 \# O7 v$ [# z1 _0 J, a8 |' m4 T' m这道题我在以前的分章分析过,不懂的可以先看下之前写的:递归与动态规划—基础篇1
4 Z9 _4 O) e& f' i6 G6 I" h
, }$ Q1 ?% T* ^) R( T5 B. `方法1::暴力递归
( L) M* w: x& P% T9 c3 H2 N0 ^, h' M+ W
这道题不难,或许你会采取下面的做法:& H( ?1 ~. s3 w
# g* [+ t2 U8 K! i2 P. j4 `9 ^
public int solve(int n){! I3 P* t6 \2 J9 I3 s
    if(n <= 2){% h$ b7 o- O) e( c* E$ c- s
        return n;
3 Q* j( E) q/ N9 y+ l    }else{
4 M5 w! R7 M( ^. ]6 d$ P; e        return solve(n-1) + solve(n-2);# |8 `" k& t! j- I3 j/ b& t
    }3 B% z! q3 ^' q- n' V
}* Q9 H/ k2 D5 B, z, ~! ?
1
% F& v/ @& [/ b6 f4 F2- G# P4 ~. ^% t  l4 ^: [& {# l
3% S- k# g, x4 G' i4 j
4  Q+ W) ?: m4 U- }& Q/ B
5. _8 [1 \" y. Y1 S5 j& w# x* {
6
" Y' `$ H0 G" p- u, d# u! J7, r# ~+ n5 o% @  [* A7 {4 n
这种做法的时间复杂度很高,指数级别了。但是如果你提交之后侥幸通过了,然后你就接着下一道题了,那么你就要好好想想了。
; E: U; W, f. m8 U: Z
1 E6 k' n" A" Y4 ~1 j9 E; q, E方法二:空间换时间
) x, M) u& a1 d- F( H  q
/ |3 `3 |* d% G8 O2 D- m力求完美,我们可以考虑用空间换时间:这道题如何你去仔细想一想,会发现有很多是重复执行了。不行你可以画个图5 a4 e7 p) [+ c" b

$ f3 a; d2 g5 V所以可以采取下面的方法:
) Q5 A4 z0 [8 T" u* H
! a$ D; C, h7 C! V! P# a//用一个HashMap来保存已经计算过的状态6 X5 `. \3 k, N4 r: J2 A
static Map<Integer,Integer> map = new HashMap();& w9 [) [- v& B; ?; j' [7 C
public static int solve(int n){
+ ]: @, e+ _% M1 x# u* s1 I         if(n <= 2){& O  H/ b& }( z* ?. ?9 ?4 Q1 |
        return n;. J, G' ?2 a/ t9 F8 P: g; e
    }else{//是否计算过2 ^3 ^% [& Y% t
        if(map.containsKey(n)){
7 p9 T7 ]" @4 d# i3 _6 k" Y: w            return map.get(n);6 q% a, ?5 S4 P$ H+ ~8 j5 c& w1 |
        }else{
  ]4 a! X9 L8 q0 b  N1 u/ d            int m = solve(n-1) + solve(n-2);
& S/ N; d( U3 i            map.put(n, m);
$ l* v( R8 h5 l$ z            return m;
8 U$ Q  ^7 V9 V0 p7 T& Q2 Q& [( X        }! v) E! D# H- M
    }5 N: w6 Z8 l3 d8 f3 B
}
+ K* s- x4 m2 V0 D3 s8 u) E4 {
4 h& ]- o# l2 c+ z9 G1
/ {: t. `: G) W- Q6 I2
, k# Q& J5 J; u1 G  N38 _: f6 _) z3 I- ~* |/ N7 U
4$ j7 B! d" x+ b6 a; p  ]; n
5
. v, i& P- @  L& P+ W6 N6" V+ j" S+ Z# ~& K$ |
7
$ B5 r- P  Z3 N) h: w9 t: B- D% @85 K, ~1 N! c0 v9 L  w
99 Z0 {( j1 I# D3 Y
104 L% X5 Q: Q5 r* q% H. `
118 |; x. z! U( L2 `+ b/ L
12
4 ^9 i1 G( K/ O- b7 s13
. ^6 L/ r9 D* {1 Q3 l/ ?14# q; ?% N, J8 ^" N1 f
15
, x0 C" e/ P' R0 U! ]% Q16
) x7 v/ ]' @. d0 {& h+ ^' E这样,可以大大缩短时间。也就是说,当一道题你做了之后,发现时间复杂度很高,那么可以考虑下,是否有更好的方法,是否可以用空间换时间。- u; k: p  Q, C) w; t

7 o3 ^# ]  S2 p1 s" d: v& f8 g方法三:斐波那契数列
- o0 n% {- ^' O, N3 G7 }7 v- q  f1 c* P' H  H  e! W; e. ~
实际上,我们可以把空间复杂度弄的更小,不需要HashMap来保存状态:5 h9 a) G. C2 \  c. _" N- c
5 w& G, v- [# r7 M; P% z: _
public static int solve(int n){
; N9 e' p% P9 {% m$ |' K    if(n <= 2){. L+ J( O+ r# t3 D: {  H
        return n;% R1 `1 T# z! F2 X
    } 5 C4 j  O& T5 p/ y7 T
    int f1 = 0;) l5 w0 B# K# r+ @+ q2 O( o2 x
    int f2 = 1;, a' d5 \/ L* {/ `: g- M* }; d% f
    int sum = 0;/ V+ Z8 M$ C7 g4 X
    for(int i = 1; i<= n; i++){
( R; m  a6 }+ p5 e$ X. Q  }        sum = f1 + f2;: @: J# x+ s# p) @/ Q2 d
        f1 = f2;
; x4 L  [' X0 G) A1 J, @: m0 M        f2 = sum;, ?/ t$ @, A+ u; L) c
    }
* s+ d% m1 f$ C0 T: c& r    return sum;. N4 c8 R' S( Q
}; I# [: C% U% w# e
1" y! J: I: H0 l7 Q* O* z# H# i9 K
2$ j' H) l7 S  R! e  x
3& s2 b  }" ]2 h$ A- I4 q0 J! t
4
2 w  h. [' v  [  u: y0 `58 }* b9 x; m; b+ ~0 P0 P
6* w. o# P/ Z2 x' R/ b0 R6 h
75 M  a( x: b9 G( T4 T' J
8
. {8 b* L3 ~$ g( H, z) C9
+ G: ?$ f4 Q0 u" O100 I+ u' Z! j: J! X# K9 f7 H
11
& o8 }( F1 P' \  H12- ?0 Y6 k: w/ c0 x, c+ B
13% z  U" p0 E: H* b0 c" ]
140 Q' l& f' [" R8 V
我弄这道题给你们看,并不是在教你们这道题怎么做,而是有以下目的:
5 r9 K0 C% q% R  I  m8 {4 @3 n8 Y" t" c1 ^
1、在刷题的时候,我们要力求完美。
2 `' [# C) A# g( Q9 z3 h7 L
! J8 ~$ B1 o) G2、我想不到这些方法啊,怎么办?那么你就可以去看别人的做法,之后,遇到类似的题,你就会更有思路,更知道往哪个方向想。& O2 v8 t% \5 X5 ^1 ^5 O

; m5 k+ K1 E# Y6 z) f3、可以从简单暴力入手做一道题,在考虑空间与时间之间的衡量,一点点去优化。. B. s: Q# ?4 I! K  P2 z$ f9 k1 ~
8 S: x1 `3 d3 _, u1 q" d& o
挑战自己,跳出舒适区
8 G8 ?! a& k5 \- W6 }3 {
8 L0 l6 W8 K/ j! P( t& K4 J) K1 D5 l什么叫舒适区?在刷题的时候,可能有一类题是你比较懂的,你每次一看就有思路,然后半个小时就撸好代码,提交代码,然后通过了,然后,哇,又多刷了一道题,心里很舒服。
* a; E1 j! w) V+ |
- d% d; w* y' n, n但是,记住,前期你可以多刷这种题练手,提升自己的乐趣,但,我还是建议你慢慢跳出舒适区,去做一些自己不擅长的题,并且找段时间一直刷这种题。例如,我觉得我在递归方面的题还是挺强的,8 W( \! o- y0 w/ ]
但是,我对动态规划的题,很菜,每次都要想好久,每次遇到这种题都有点害怕,没什么信心。不过有段时间我觉得只刷动态规划的题,直接在 leetcode 选定专题,连续做了四五十道,刚开始很难受,后来就慢慢知道了套路了,一道题从两三个小时最后缩到半小时,简单的十几分钟就搞定。感觉自己对这类型的题也不惧怕的。, c" b" f/ K* r; q+ o5 s

& x% W- `* J2 N% X0 O) b2 ^. D当然,对于动态规划的学习,大家也可以看我这篇广受好评的文章:为什么你学不过动态规划?告别动态规划,谈谈我的经验
: {9 s9 R8 J% R% Z5 E" Z) }$ _0 W/ U6 m- P) D% Z1 P1 ~+ F, m+ k
所以,建议你,一定要学好跳出自己的舒适区。
/ e9 f. Y+ T) A% ^/ {) z  U8 p
: V9 |' r& ]: T+ p7 j) h一定要学会分类总结; Z2 I" q7 {0 S: T0 w4 H1 H
. `% k  `- ?& F. I
有些人以为 leetcode 的题刷的越多,就一定能越厉害,其实不然,leetcode 虽然有 1000 多道题,但题型就那么几类,我们前期在刷的时候,我是建议按照题型分类刷题的,例如我这整理刷二叉树相关,然后刷链表相关,然后二分法,然后递归等等,每刷一种题型,都要研究他们的套路,如果你愿意去总结,那么 leetcode 的题,其实你刷几百道,有目的、挑选的刷,我觉得就差不多了。
+ J# g: ~$ F+ v, w7 {3 r; ~- n5 k  C+ @& u' ]$ D
我看过一本书,叫做《程序员代码面试指南:IT 名企算法与数据结构题目最优解》,这本书就非常不错,里面按照栈,队列,链表,二叉树,字符串等一个专题一个专题来刷的,并且每道题都给出了最优解,而且里面的题有一定的难度,感兴趣的,真心不错,如果你把这本书的题全部搞定,并且总结相关套路,那么你的算法一定有很大的提升。
6 Y& g6 A/ m6 y$ v: c4 N4 A4 k
* i0 O. q8 o2 {. h$ p* G推荐一些刷题网站) J. o% t7 W) P" ?' N/ M
6 x1 x1 R8 v" e. f$ l
我一般是在leetcode和牛客网刷题,感觉挺不错,题目难度不是很大。
3 Z- Y4 `1 m: j9 m$ T: `) ^& `( B( z5 H( @4 V
在牛客网那里,我主要刷剑指Offer,不过那里也有个在线刷leetcode,不过里面的题量比较少。牛客网刷题有个非常方便的地方就是有个讨论区,那里会有很多大佬分享他们的解题方法,不用我们去百度找题解。所以你做完后,实在想不出,可以很方便着去看别人是怎么做的。
: W0 {) X% g/ Q0 G. v1 o1 ]8 C$ e) {4 S2 k/ e* z+ Y+ K
至于leetcode,也是大部分题目官方都有给出答案,也是个不错的刷题网站。你们可以两个挑选一个,或者两个都刷。
( e6 c* \' I+ v6 E# ]6 X$ a. A0 H
5 q* O6 a# b; g0 a7 l6 ?  z& U0 m3 \当然,还有其他刷题的网站,不过,其他网站没刷过,不大清除如何。
( N3 Z% i+ U* y+ E' T
7 m3 r' R( M; t至于leetcode,有中文版和英文版
4 I' S  X4 ~' a( p4 O* W7 k+ g  v8 n8 x) r/ W9 D5 w
leetcode有中文版$ F3 y4 T% r( s8 X; y

; `2 N/ p4 q# c" n英文版) `: U, r  t  |
/ h% j4 Z* n* ]9 h8 _
根据自己的兴趣选。' P5 a; W# m( P* ^

! Y' _4 w; ^" V$ W9 x学习一些解题技巧
; e6 M" D4 y3 E' x8 b1 f, }0 ~; W7 D- _
说实话,有些题在你没看别人的解法前,你好不知道有这么美妙优雅的解法,看了之后,卧槽,居然还可以这样。而我们在刷题的过程中,就要不断累积这些技巧,当你累计多了,你就会形成一种
) B! X8 M# [. Z. K神经反应,一下子就想到了某种方法。解题技巧很多,例如数组下标法、位图法、双指针等等,我自己也分享过一篇总结一些算法技巧的文章
9 S4 a% V: v7 t0 x6 y/ Y/ j! f7 n, d1 s5 f* Y% ^( v; K& i. Z+ V  B
推荐阅读:一些常用的算法技巧总结8 N( {# w+ J% I9 g9 g
$ [4 h! F7 m5 r; Z! h4 W0 U) V5 w
例如在刷题的时候,我们要学会巧用双指针、数组下标法、位运算等等技巧来解决问题,可能会有意想不到的效果。我给你再找点我之前写文章的一些例子吧:
* }8 w: }7 R' w: X1 \! s9 ^" S# t: s% k  |1 s
分享一道解法巧妙的算法题
5 u: L5 d% X* |  z/ b
/ \+ N6 ~, F' Q' j【算法技巧】位运算装逼指南
: h2 T+ A+ y. x3 y- b6 r4 ~
) C6 {% y! a2 K% I, F& v这是个长期累积的过程,我自己也精彩在我的公众号里分享一些解题的文章,感兴趣的可以关注我的公众号:帅地玩编程。
5 T4 l- a4 F# U- O
: s# n% @  Y; s' g7 y再说数据结构发重要性4 w/ h( L7 w, Z2 v  G0 w" S

6 x; b5 [. q6 C5 q9 I4 ?. e) L前面我主要是说了我平时都是怎么学习算法的。在数据结构方法,我只是列举了你们一定要学习链表和树(二叉堆),但这是最基本的,刷题之前要掌握的,对于数据结构,我列举下一些比较重要的:8 Q1 k1 y4 }- j

1 w9 h8 D* G0 H; Q6 x( I7 R/ B1、链表(如单向链表、双向链表)。
" @5 S" `) w4 d4 k+ S& W2 t
, o0 c# k$ B. K0 {* M+ P/ j2、树(如二叉树、平衡树、红黑树)。
6 b3 p# p' F" h+ r2 k! W
! v# z7 `% W3 o3、图(如最短路径的几种算法)。  z* m( m6 e# y  o( o* G5 w# ]
3 H, ?3 l3 Z# `) f( `
4、队列、栈、矩阵。  {9 o0 V6 v2 U

4 H* U- F+ [* e4 O6 B对于这些,自己一定要动手实现一遍。你可以看书,也可以看视频,新手可以先看视频,不过前期可以看视频,之后我建议是一定要看书。0 H/ f& q9 \) `7 O9 V

& ^0 S% j/ K) s/ A' n例如对于平衡树,可能你跟着书本的代码实现之后,过阵子你就忘记,不过这不要紧,虽然你忘记了,但是如果你之前用代码实现过,理解过,那么当你再次看到的时候,会很快就记起来,很快就知道思路,而且你的抽象能力等等会在不知不觉中提升起来。之后再学习红黑树啊,什么数据结构啊,都会学的很快。# R& e8 G+ G# r7 F' s* m
' H- a8 b" Y7 y+ D
对于有哪些值得学习的算法,我之前也总结过,这里推荐给大家程序员必须掌握的核心算法有哪些?,这篇文章居然 40多万阅读量了,有点受宠若惊。, v  N4 }$ U  u6 Y( ]

2 B5 V% ?2 z: |2 L最最重要
8 O+ L1 S/ D5 a. F: D- _8 C5 V' H: a; L% s8 I
动手去做,动手去做,动手去做。重要的话说三遍。3 u7 J6 K$ W# M* _2 G4 R. ?

7 }9 d, g, |' }6 a- |# y6 Y千万不要找了一堆资源,订好了学习计划,我要留到某某天就来去做…
4 r- n/ B1 v: ^/ T3 K$ V6 G( N& }* D+ b3 w3 l1 o  U1 m
千万不要这样,而是当你激情来的时候,就马上去干,千万不要留到某个放假日啊什么鬼了,很多这种想法的人,最后会啥也没做的。4 X/ u& h/ L& [5 S+ ~5 _- ]

9 k8 N& g/ }; N也不要觉得要学习的有好多啊,不知道从哪学习起。我上面说了,可以先学习最基本的,然后刷题,刷题是一个需要长期坚持的事情,一年,两年。在刷题的过程中,可以穿插和学习其他数据结构。
2 y/ a8 x: A9 a
1 h& G1 T! ]4 L总结一下吧
5 H0 R% N0 E6 x+ {7 K5 C0 W: J* v$ ?- ~( \8 F! K6 ?4 e. f, a
所以我给大家的建议就是,先学习基本的数据结构以及算法思想,不要盲目刷题,接着刷题的过程中,不能得过且过,尽量追求最优解,还有就是要跳出舒适区,逼自己成长,刷题的过程中,要学会分类总结。
; b. `( d* p5 p4 Y& J" ]% ^2 [7 h
) ]' _: u5 i$ n9 A当然,最重要的,就是你去动手了,不然,一切免谈!
5 x: ^( l( A, z" [* K/ D
" K+ [# ~* }$ n2 D2 N) b看在熬夜写过的份上,送我个赞呗,嘻嘻。
/ g( J8 x2 ^2 ?————————————————
4 _$ \1 R. A+ l% Z版权声明:本文为CSDN博主「帅地」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。3 a7 v5 K' Q. Z
原文链接:https://blog.csdn.net/m0_37907797/article/details/104765116
8 `# S1 ?; B. O6 O3 |4 M
: k6 @. c/ l$ i0 ]% v- [& n
, _7 n* W7 s  W8 M




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