小弟最近在学习算法和数据结构,奈何经常搞不懂递归的问题,也查阅了不少资料,但还是没搞懂用递归去反转链表
所以发了这个贴请教一下 v2 上面的大佬,谢谢大家啦
下面是从 leetcode 看到的 solution
class Solution: def reverseList(self, head): if not head or not head.next: return head new_head = self.reverseList(head.next) next_node = head.next # head -> next_node next_node.next = head # head <- next_node head.next = None # [x] <- head <- next_node return new_head
不能理解的地方是,返回最后一个的节点指针时候这三行代码的意思
next_node = head.next next_node.next = head head.next = None
![]() | 1 quickma 2018-10-18 09:43:18 +08:00 递归都是靠悟性的 |
![]() | 2 swulling 2018-10-18 09:50:30 +08:00 via iPhone 恕我眼拙,这个算递归么?不就是正常的遍历反转么 |
![]() | 4 swulling 2018-10-18 09:52:53 +08:00 via iPhone 呀,看错了,中间加了递归……不过这个解太复杂了吧,直接遍历反转就行了,这个递归加的没有意义 |
![]() | 5 canbingzt 2018-10-18 09:53:59 +08:00 注释很明显了,前 2 行把 2 个节点反向,第 3 行把节点断开(不然就循环了) |
7 xilixjd 2018-10-18 10:04:36 +08:00 从我刷 200 题的菜鸡来看,递归靠悟性,1 楼说的很对 这边递归只是为了将 new_head 指到最后一个节点 那三行建议你在纸上画个图就理解了 |
![]() | 8 BingoXuan 2018-10-18 10:13:47 +08:00 head 的实现能给一下吗? 函数返回的是节点。递归最后调用时候 head.next 是倒数第一个,head 是倒数第二个。第一句用临时变量指向倒数第一个节点(即原下一个节点),第二句是将倒数第二个作为倒数第一个下一个节点(原上一个节点是原下一个节点的下一个节点),第三句是把倒数第二个节点指向一个空(原上一个节点的下一个节点是空),以便最后返回的时候节点末尾是空,以及避免两个节点相互指向。最后返回最后一个节点。然后一路往回走,原第一个节点指向空,原最后一个节点按照原先顺序往回一路指向。 如果还是想不通就直接把函数代码嵌套一层,打印出来把。 |
![]() | 9 BingoXuan 2018-10-18 10:19:57 +08:00 @xilixjd 我是直接想象一下先嵌套一层,嵌套那一层是用什么条件实现不递归直接跳出来的,按照这个思路推导递归进去,结束条件,递归出来整个过程的。就是纯想就真的烧脑,如果有笔纸会好很多(但是懒。。。)。 |
![]() | 10 reus 2018-10-18 10:25:33 +08:00 ![]() 翻转 [head, n1, n2, n3, ...] 等于先翻转 [n1, n2, n3, ...] 再把 head 放到最后 head.next 就是 [n1, n2, n3, ...],翻转就是 reverseList(head.next) 结果是 [..., n3, n2, n1],注意 head.next 现在仍然指向 n1,也就是最后 所以,next_node = head.next 等于 next_node 赋值为 n1,也就是末尾的结点 然后 next_node.next = head,就是构造 [..., n3, n2, n1, head] head.next = None,就是把 head 指向 n1 去掉 就翻转了 next_node 应该命名为 last_node,这样一看就明白了 |
11 bucky 2018-10-18 10:30:20 +08:00 new_head = self.reverseList(head.next) # 除了 head 节点已经反转好的链表 # 将 head 节点和 head.next 交换 原始 head -> head_next -> None 1 -> 2-> None head.next.next = head # head -> head_next -> head 1 -> 2-> 1 存在循环 head.next = None # head 断开 head_next -> head -> None 1 断开 2-> 1 -> None return new_head |
12 bucky 2018-10-18 10:36:13 +08:00 # head.next.next = head 和 next_node = head.next next_node.next = head 效果一样 # 除了 head 节点已经反转好的链表 new_head = self.reverseList(head.next) # 将 head 节点和 head.next 交换 # 原始 head -> head_next -> None 1 -> 2 -> None # head -> head_next -> head 1 -> 2 -> 1 存在循环 head.next.next = head # head 断开 head_next -> head -> None 1 断开 2-> 1 -> None 这就是反转两个结果的结果 head.next = None return new_head |
![]() | 13 reus 2018-10-18 10:36:59 +08:00 开始是 head -> n1 -> n2 -> ... reverseList(head.next)之后是: new_head -> ... -> n2 -> n1 <- head 没有循环,是 head 和 n1 的方向不对,那三行就是将 n1 <- head 改成 n1 -> head |
14 bucky 2018-10-18 10:53:30 +08:00 你只要注意这个递归是从后向前处理的,而不是从前往后处理的就容易理解了, 比如 [1, 2, 3, 4, 5] 是先处理后面 [4, 5] , 把 [4, 5] 变成 [5, 4], 这就是那三行代码的意思 |
![]() | 15 SpiderXiantang 2018-10-18 10:57:28 +08:00 leetcode 60 道小白说一句 学递归之前先理解栈 |
![]() | 17 zhaogaz 2018-10-18 11:07:29 +08:00 递归的核心是一种递推式,就是能把问题拆成子问题,子问题和现有的问题解法一样,就能用递归写出来了。 leetcode 我只写了几十道题,没到 100,能总结出来规律的。写多了就会了。 |
![]() | 19 YaphetYin 2018-10-18 11:27:11 +08:00 via iPhone 这三句把当前 head 放到反序链表的最后面 |
20 flight2006 2018-10-18 11:37:33 +08:00 @reus 你这思路我看了下是错的,最后三行不是调最后的 head,而是反转每个节点的 next 指向,每次执行出栈都会执行,而不是只执行一次,其他人的思路还没看 |
![]() | 21 reus 2018-10-18 11:57:50 +08:00 @flight2006 懂递归吗你? 例如 [1, 2, 3, 4, 5] 就是翻转 [2, 3, 4, 5],再把 1 放最后 翻转 [2, 3, 4, 5],就是翻转 [3, 4, 5], 2 放最后 翻转 [3, 4, 5] 就是 翻转 [4, 5] ,3 放最后 翻转 [4, 5],就是翻转 [5],4 放最后 翻转 [5],就是 [5],4 放最后得到 [5, 4] 再 3 放最后,得到 [5, 4, 3] 再 2 放最后,得到 [5, 4, 3, 2] 再 1 放最后,得到 [5, 4, 3, 2, 1] 每一步概括起来,就是先翻转除了第一个的余下的列表,然后把第一个放最后 最后三行就是”把第一个元素放在最后“这个操作 哪来什么“只执行一次”? @bucky [1 2 3 4 5]是先处理 [2 3 4 5],当然最后也会处理到 [4 5] |
22 bucky 2018-10-18 12:07:05 +08:00 @reus 真搞笑你,前面是递归的展开(在全部展开之前还没进行过一次处理),后面才是处理,代码里面进行递归是在反转(后面的三行代码)的前面的,你看不到还是看不懂。本来探讨问题有错误没什么,你这说话的口气真大,口气大有真本事也忍了,关键是你自己没本事还一直 diss 别人,别丢人了行吗? 还先处理 [1], 你自己在纸上写一写看是不是先处理[1], 真是丢人丢到家了 |
24 iceheart 2018-10-18 12:50:23 +08:00 via Android 看我写这个吧,刚出炉 node *revert(node* left, node* right){ if(!right) return left; node* next = right->next; right->next = left; return revert(right,next); } //use: node *rlist = revert(nullptr, list); |
![]() | 25 reus 2018-10-18 12:56:18 +08:00 @bucky 我对“处理”的理解,就是调用 reverseList,你对“处理”的理解是改变 next。“先处理[1]”是哪里来的?我有说过?这个帖子就你说了[1]吧? 不争了,没意义。 |
26 Justin13 2018-10-18 13:13:33 +08:00 via Android 递归是很优雅的实现方式,也很好理解。 核心是把公共的处理逻辑抽出来,以递归的方式实现遍历,结束递归的条件就是遍历结束的条件。 有些语言根本没有循环五路,像 for,while 这些都用递归来实现。 |
27 bucky 2018-10-18 13:14:40 +08:00 @reus 要点脸行吗?自己错了就开始狡辩,递归里面什么时候是式子的展开,什么时候是处理是你说的算吗?真搞笑,就你这水平就别出来丢人了,现在 v 站什么人都感回答问题,你是做销售的吧 |
29 flight2006 2018-10-18 13:25:39 +08:00 @reus 你这人真喜欢给人扣帽子,就你懂递归,head 就是当前处理节点的引用,链表不是单纯数字的 list 还有节点的 next 指向,那三步就是在反转当前处理 head 的 next 指向问题,代码后面的注释箭头很清楚了 |
30 Deville 2018-10-18 13:28:46 +08:00 别吵了,PHP 是世界上最好的语言 |
31 xilixjd 2018-10-18 13:37:16 +08:00 @BingoXuan 画三个节点在纸上,跟着代码过一遍,你连动手都不舍得,自己又不是那种聪明绝顶抽象能力很好的,看再多的资料有啥用?楼上回答基本不用看了,因为看了我感觉你依然想象不出来 |
![]() | 33 reus 2018-10-18 13:58:00 +08:00 @flight2006 你说的这些和我说的,有什么冲突吗? |
![]() | 34 BingoXuan 2018-10-18 14:09:51 +08:00 @xilixjd 我没说想象不出来啊,反转链表,二叉树遍历这一类基础本身就不难。lz 代码又不多,嵌套一层也就 14 行代码。只是我习惯就是除非真的太难,否则不会动笔。自我挑战时烧脑的感觉也是很有意思。 |
![]() | 35 ECHO777 2018-10-18 16:17:48 +08:00 要理解递归的过程中挂起的概念,这一点很重要 |
![]() | 36 loryyang 2018-10-18 16:30:44 +08:00 你自己画一下就知道了啊。如果画不出来,你就中间 print 出来,看看每个 node 都是谁 如果这个困难都过不去,感觉算法对你来说还是太难了一点 至于递归的概念,最重要的就是认定一点:这个函数能帮你搞定什么 比如例子中是:能帮你反转从 head 开始的链表,那么每次递归就是帮你反转除了 head (也就是 head.next 开始)的所有链表,我们可以称为子链表 那么你递归调用之后,剩下的就是要把 head 塞到反转之后( self.reverseList(head.next))的链表的尾巴上面去。这个尾巴是什么呢?就是原子链表的头:head.next,原来他是头,现在反转就是尾巴了。(需要注意的是,head.next 依然指向这个节点,即使递归里面做了许多操作) 所以我们要做的就是把 head,塞到 head.next 的后面去,要做的就是 head.next.next 指向 head,head.next 指向 None (尾巴的后面就没有别的东西了)。所以那三行冗余了,其实就两行:head.next.next = head; head.next = None 其实理解递归最好的公式还是斐波那契数列,你可以去参考下。这个题目完全没必要递归,违反人类直觉 |
37 raingorain OP @loryyang 谢谢你,真的很用心,现在一个个 node 都 print 出来看了,谢谢 |
![]() | 38 loryyang 2018-10-18 16:42:46 +08:00 另外,我看了下,@reus 的说明是对的,别的不清楚在质疑什么 按照我的理解,reus 的解释还是比较容易理解的。next_node 是否要命名为 last_node 看你从什么角度看吧 当然每个人思维方式不一样,建议 lz 自己画一画 |
![]() | 39 loryyang 2018-10-18 16:44:57 +08:00 @raingorain 客气,希望你能不断成长,路还很长 |
43 bucky 2018-10-18 20:10:00 +08:00 @loryyang 一个连递归展开都不清楚,不清楚反转链表的执行从后面还是从前面开始的人,你觉得他说的是对的,那你们好好交流一下 |
![]() | 44 loryyang 2018-10-18 20:18:24 +08:00 |
![]() | 45 chengluyu 2018-10-18 20:24:42 +08:00 reverse [] = [] reverse (x:xs) = reverse xs ++ [x] |
47 bucky 2018-10-18 20:39:57 +08:00 @loryyang 说真的,你们两个应该交个朋友,没理可说了就去翻别人的记录,你们两个这么像不交朋友可惜了,当然可能就是一个人呀,哈哈 |
![]() | 48 icedx 2018-10-18 20:58:06 +08:00 写递归的时候最重要的是头脑清晰 而且一般能跑通(不 SF) 就能正常工作 |