Next.js 实现原理系列(1):Link 基础

背景

这周开始尝试阅读 Next.js 源码,有难度,但也很有意思。

对我来说,一个能够极大提高效率的技巧就是:每次先抛出一个自己感兴趣的小问题(多是关于实现原理的细节),然后阅读源码,尝试回答这个问题。解决疑问之后,再放下源码,抛出下一个感兴趣的问题(很可能是解决了上一个问题所引发的新问题),再阅读源码,如此循环往复。

这样的好处是,每次阅读源码都有一个清晰且易实现的目标(解答当前问题),不易陷入无关细节。如此反复数次,解决一个又一个问题,对实现原理的理解自然层层加深了。

首先从 Next.js 的 Link 与 Router 着手,因为感觉这一部分比较直观,好上手;而且还因为我刚写了一篇 一箭双雕:浅谈 Next.js 页面导航(page navigation)原理,和这个关系也比较大。

本篇涉及的源码文件:

packages/next/client/link.tsx # Link 的实现

注:本系列文章基于 Next.js v12.0.4 版本。

热身:a 链接 的 preventDefault

第一个问题非常简单。在上一篇文章 一箭双雕:浅谈 Next.js 页面导航(page navigation)原理 中我提到,Next.js 中每个 <Link /> 下都有一个 <a> 元素,用户正常点击的时候 <a> 不会生效,但在 JS 禁用的时候(例如对于不允许 JS 的爬虫来说) <a> 就能派上用场了。

下面通过源码印证一下,正常情况下 Next.js 真的会让 <a> 失效。

我们平常使用的 <Link /> 组件,其定义在 packages/next/client/link.tsx 的第 109 行。这里略去大部分无关代码,截取其中最重要的一段做注释:

代码片段 1:Link

// packages/next/client/link.tsx
// 第 109 行
function Link(props: React.PropsWithChildren<LinkProps>) {
   
  // 前面代码略,从第 277 行开始
  const childProps: {
   
    onMouseEnter?: React
  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
index.vue:201 Uncaught (in promise) TypeError: Cannot read properties of undefined (reading 'type') at _callee$ (index.vue:201:1) at tryCatch (regeneratorRuntime.js:44:1) at Generator.eval (regeneratorRuntime.js:125:1) at Generator.eval [as next] (regeneratorRuntime.js:69:1) at asyncGeneratorStep (asyncToGenerator.js:3:1) at _next (asyncToGenerator.js:22:1) at eval (asyncToGenerator.js:27:1) at new Promise (<anonymous>) at eval (asyncToGenerator.js:19:1) at VueComponent.handleNodeClick (index.vue:227:1) _callee$ @ index.vue:201 tryCatch @ regeneratorRuntime.js:44 eval @ regeneratorRuntime.js:125 eval @ regeneratorRuntime.js:69 asyncGeneratorStep @ asyncToGenerator.js:3 _next @ asyncToGenerator.js:22 eval @ asyncToGenerator.js:27 eval @ asyncToGenerator.js:19 handleNodeClick @ index.vue:227 handleCurrentChange @ index.vue:197 invokeWithErrorHandling @ vue.runtime.esm.js:3971 invoker @ vue.runtime.esm.js:1188 invokeWithErrorHandling @ vue.runtime.esm.js:3971 Vue.$emit @ vue.runtime.esm.js:2874 eval @ element-ui.common.js:1116 eval @ vue.runtime.esm.js:4097 flushCallbacks @ vue.runtime.esm.js:4019 Promise.then(异步) timerFunc @ vue.runtime.esm.js:4044 nextTick @ vue.runtime.esm.js:4109 queueWatcher @ vue.runtime.esm.js:3346 Watcher.update @ vue.runtime.esm.js:3584 Dep.notify @ vue.runtime.esm.js:710 reactiveSetter @ vue.runtime.esm.js:4380 proxySetter @ vue.runtime.esm.js:5158 handleCurrentChange @ element-ui.common.js:1069 invokeWithErrorHandling @ vue.runtime.esm.js:3971 invoker @ vue.runtime.esm.js:1188 invokeWithErrorHandling @ vue.runtime.esm.js:3971 Vue.$emit @ vue.runtime.esm.js:2874 onPagerClick @ element-ui.common.js:547 invokeWithErrorHandling @ vue.runtime.esm.js:3971 invoker @ vue.runtime.esm.js:1188 original_1._wrapper @ vue.runtime.esm.js:7265
06-12

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值