递归:如何找到“最终推荐人”?

一、递归

\quad 递归是一种非常高效、简洁的编码技巧,一种应用非常广泛的算法,比如DFS深度优先搜索、前中后序二叉树遍历等都是使用递归。

\quad 方法或函数调用自身的方式称为递归调用,调用称为递,返回称为归。即:的过程叫“”,回来的过程叫“”。

\quad 基本上,所有的递归问题都可以用递推公式来表示:一排人的位置(通过询问前一个人的位置):

\quad f(n)=f(n-1)+1 \quad 其中,f(1)=1

二、递归需要满足的三个条件

① 一个问题的解可以分解为几个子问题的解

\quad 何为子问题?子问题就是数据规模更小的问题。比如,电影院的例子,你要知道,“自己在哪一排”的问题,可以分解为“前一排的人在哪一排”这样一个子问题。2. 这个问题与分解之后的子问题,除了数据规模不同,求解思路完全一样。

② 这个问题与分解之后的子问题,除了数据规模不同,求解思路完全一样

\quad “自己在哪一排”的思路,和前面一排人求解“自己在哪一排”的思路,是一模一样的。

③ 存在递归终止条件

\quad 把问题分解为子问题,把子问题再分解为子子问题,一层一层分解下去,不能存在无限循环,这就需要有终止条件。还是电影院的例子,第一排的人不需要再继续询问任何人,就知道自己在哪一排,也就是 f(1)=1,这就是递归的终止条件。

三、为什么使用递归?递归的优缺点?

1、优点

\quad 代码的表达力很强,写起来简洁。

2、缺点

\quad 空间复杂度高、有堆栈溢出风险、存在重复计算、过多的函数调用会耗时较多等问题。

四、如何实现递归?

1、递归代码编写

\quad 写递归代码的关键就是找到如何将大问题分解为小问题的规律,并且基于此写出递推公式,然后再推敲终止条件,最后将递推公式和终止条件翻译成代码。

2、迭代循环的非递归

\quad 笼统的讲,所有的递归代码都可以改写为迭代循环的非递归写法。如何做?抽象出递推公式、初始值和边界条件,然后用迭代循环实现。

五、递归常见问题及解决方案

1、警惕堆栈溢出

\quad 可以声明一个全局变量来控制递归的深度,从而避免堆栈溢出。

2、警惕重复计算

\quad 通过某种数据结构来保存已经求解过的值,从而避免重复计算。

3、死循环

\quad 数据库里存在脏数据,我们还需要处理由此产生的无限递归问题。如果 A 的推荐人是 B,B 的推荐人是 C,C 的推荐人是 A,这样就会发生死循环。

六、如何找到“最终推荐人”?

\quad actor_id 表示用户 id,referrer_id 表示推荐人 id。
在这里插入图片描述

long findRootReferrerId(long actorId) {
  Long referrerId = select referrer_id from [table] where actor_id = actorId;
  if (referrerId == null) return actorId;
  return findRootReferrerId(referrerId);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ru-willow

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值