diff --git a/404.html b/404.html index 15eecf1cf..65ab9b2a7 100644 --- a/404.html +++ b/404.html @@ -20,7 +20,7 @@
Skip to content

404

PAGE NOT FOUND

But if you don't change your direction, and if you keep looking, you may end up where you are heading.

Released under the MIT License.

- diff --git a/assets/index.md.0d318df1.js b/assets/index.md.4abc46fe.js similarity index 95% rename from assets/index.md.0d318df1.js rename to assets/index.md.4abc46fe.js index fd2f31e0e..a6a5ea9cf 100644 --- a/assets/index.md.0d318df1.js +++ b/assets/index.md.4abc46fe.js @@ -1 +1 @@ -import{_ as t,o as e,c as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"Home","description":"","frontmatter":{"layout":"home","title":"Home","hero":{"name":"ran","text":"风起于青萍之末","tagline":"A ship in harbor is safe, but that is not what ships are built for.","image":{"src":"/home.svg","alt":"logo"},"actions":[{"theme":"brand","text":"更多详情","link":"https://github.com/chaxus/ran"},{"theme":"alt","text":"访问我的GitHub","link":"https://github.com/chaxus/ran"}]},"features":[{"icon":"⚡️","title":"记录","details":"每当遇到问题或者可以改进的事情的时候,会把想法先写下来,如果这是你遇到的问题,那么很可能有一群人也遇到同样的困难。"},{"icon":"🖖","title":"解决","details":"过一段时间后,再回来看看这些问题是否还有意义。然后再研究这个问题,确定是否存在解决方案。"},{"icon":"🛠️","title":"改进","details":"通常会发现一个工具是为了解决一个问题,我会去考虑如何改进或者简化这个工具。创造一个更好的轮子。"}]},"headers":[],"relativePath":"index.md","lastUpdated":1694276611000}'),i={name:"index.md"};function o(s,n,r,c,l,h){return e(),a("div")}const p=t(i,[["render",o]]);export{m as __pageData,p as default}; +import{_ as t,o as e,c as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"Home","description":"","frontmatter":{"layout":"home","title":"Home","hero":{"name":"ran","text":"风起于青萍之末","tagline":"A ship in harbor is safe, but that is not what ships are built for.","image":{"src":"/home.svg","alt":"logo"},"actions":[{"theme":"brand","text":"更多详情","link":"https://github.com/chaxus/ran"},{"theme":"alt","text":"访问我的GitHub","link":"https://github.com/chaxus/ran"}]},"features":[{"icon":"⚡️","title":"记录","details":"每当遇到问题或者可以改进的事情的时候,会把想法先写下来,如果这是你遇到的问题,那么很可能有一群人也遇到同样的困难。"},{"icon":"🖖","title":"解决","details":"过一段时间后,再回来看看这些问题是否还有意义。然后再研究这个问题,确定是否存在解决方案。"},{"icon":"🛠️","title":"改进","details":"通常会发现一个工具是为了解决一个问题,我会去考虑如何改进或者简化这个工具。创造一个更好的轮子。"}]},"headers":[],"relativePath":"index.md","lastUpdated":1694419899000}'),i={name:"index.md"};function o(s,n,r,c,l,h){return e(),a("div")}const p=t(i,[["render",o]]);export{m as __pageData,p as default}; diff --git a/assets/index.md.0d318df1.lean.js b/assets/index.md.4abc46fe.lean.js similarity index 95% rename from assets/index.md.0d318df1.lean.js rename to assets/index.md.4abc46fe.lean.js index fd2f31e0e..a6a5ea9cf 100644 --- a/assets/index.md.0d318df1.lean.js +++ b/assets/index.md.4abc46fe.lean.js @@ -1 +1 @@ -import{_ as t,o as e,c as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"Home","description":"","frontmatter":{"layout":"home","title":"Home","hero":{"name":"ran","text":"风起于青萍之末","tagline":"A ship in harbor is safe, but that is not what ships are built for.","image":{"src":"/home.svg","alt":"logo"},"actions":[{"theme":"brand","text":"更多详情","link":"https://github.com/chaxus/ran"},{"theme":"alt","text":"访问我的GitHub","link":"https://github.com/chaxus/ran"}]},"features":[{"icon":"⚡️","title":"记录","details":"每当遇到问题或者可以改进的事情的时候,会把想法先写下来,如果这是你遇到的问题,那么很可能有一群人也遇到同样的困难。"},{"icon":"🖖","title":"解决","details":"过一段时间后,再回来看看这些问题是否还有意义。然后再研究这个问题,确定是否存在解决方案。"},{"icon":"🛠️","title":"改进","details":"通常会发现一个工具是为了解决一个问题,我会去考虑如何改进或者简化这个工具。创造一个更好的轮子。"}]},"headers":[],"relativePath":"index.md","lastUpdated":1694276611000}'),i={name:"index.md"};function o(s,n,r,c,l,h){return e(),a("div")}const p=t(i,[["render",o]]);export{m as __pageData,p as default}; +import{_ as t,o as e,c as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"Home","description":"","frontmatter":{"layout":"home","title":"Home","hero":{"name":"ran","text":"风起于青萍之末","tagline":"A ship in harbor is safe, but that is not what ships are built for.","image":{"src":"/home.svg","alt":"logo"},"actions":[{"theme":"brand","text":"更多详情","link":"https://github.com/chaxus/ran"},{"theme":"alt","text":"访问我的GitHub","link":"https://github.com/chaxus/ran"}]},"features":[{"icon":"⚡️","title":"记录","details":"每当遇到问题或者可以改进的事情的时候,会把想法先写下来,如果这是你遇到的问题,那么很可能有一群人也遇到同样的困难。"},{"icon":"🖖","title":"解决","details":"过一段时间后,再回来看看这些问题是否还有意义。然后再研究这个问题,确定是否存在解决方案。"},{"icon":"🛠️","title":"改进","details":"通常会发现一个工具是为了解决一个问题,我会去考虑如何改进或者简化这个工具。创造一个更好的轮子。"}]},"headers":[],"relativePath":"index.md","lastUpdated":1694419899000}'),i={name:"index.md"};function o(s,n,r,c,l,h){return e(),a("div")}const p=t(i,[["render",o]]);export{m as __pageData,p as default}; diff --git a/assets/src_article_astParse_tokenizer.md.ccc361f6.js b/assets/src_article_astParse_tokenizer.md.99a1983d.js similarity index 99% rename from assets/src_article_astParse_tokenizer.md.ccc361f6.js rename to assets/src_article_astParse_tokenizer.md.99a1983d.js index bfcef6352..34c5771a5 100644 --- a/assets/src_article_astParse_tokenizer.md.ccc361f6.js +++ b/assets/src_article_astParse_tokenizer.md.99a1983d.js @@ -1,4 +1,4 @@ -import{_ as a,o as n,c as l,N as o}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/Literal.4f8173cf.jpeg",e="/ran/assets/Identifier.91b58315.jpeg",t="/ran/assets/Statement.98dca542.jpeg",c="/ran/assets/Declaration.1f7b4e23.jpeg",r="/ran/assets/Expression.d8010e56.jpeg",D="/ran/assets/ExpressionStatement.15a42b4e.jpeg",y="/ran/assets/Class.614486e5.jpeg",F="/ran/assets/import.32c7331a.jpeg",B="/ran/assets/export.84f41d4d.jpeg",i="/ran/assets/Program.e2066113.jpeg",d="/ran/assets/File.9a43a095.jpeg",E="/ran/assets/axtexplorer.2ebc951a.jpeg",A="/ran/assets/axtexplorerSave.21965993.jpeg",s="/ran/assets/Comment.4c895119.jpeg",C="/ran/assets/extra.ff9e7b7d.jpeg",v=JSON.parse('{"title":"Abstract Syntax Tree","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/astParse/tokenizer.md","lastUpdated":1694276611000}'),u={name:"src/article/astParse/tokenizer.md"},h=o('

Abstract Syntax Tree

一.(abstract syntax tree)抽象语法树的作用

源码是一串按照语法格式来组织的字符串,人能够认识,但是计算机并不认识,想让计算机认识就要转成一种数据结构,通过不同的对象来保存不同的数据,并且按照依赖关系组织起来,这种数据结构就是抽象语法树(abstract syntax tree)。

之所以叫“抽象”语法树是因为数据结构中省略掉了一些无具体意义的分隔符比如 ; { } 等。

有了 AST,计算机就能理解源码字符串的意思,而理解是能够转换的前提,所以编译的第一步需要把源码 parseAST

转成 AST 之后就可以通过修改 AST ,分析 AST 的方式来修改和分析代码,比如 babel 就通过这种方式进行代码的转换,比如 rollupTree Shaking ,就是通过分析 AST的 导入导出语法,从而分析出没有使用的代码,进行去除。

二.常见的 AST 节点

常见的 AST 节点 AST 是对源码的抽象,字面量、标识符、表达式、语句、模块语法、class 语法都有各自的 AST。

我们分别来了解一下:

Literal

Literal 是字面量的意思,比如 let name = 'value'中,'value'就是一个字符串字面量 StringLiteral,相应的还有数字字面量 NumericLiteral,布尔字面量 BooleanLiteral,字符串字面量 StringLiteral,正则表达式字面量 RegExpLiteral 等。

下面这些字面量都有对应的 Literal 节点:

代码中的字面量很多,babel 就是通过 xxLiteral 来抽象这部分内容的。

Identifier

Identifer 是标识符的意思,变量名、属性名、参数名等各种声明和引用的名字,都是Identifer

我们知道, JS 中的标识符只能包含字母或数字或下划线 (“_”) 或美元符号 (“$”) ,且不能以数字开头。这是 Identifier 的词法特点。

尝试分析一下,下面这一段代码里面有多少 Identifier 呢?

js
const name = 'value'
+import{_ as a,o as n,c as l,N as o}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/Literal.4f8173cf.jpeg",e="/ran/assets/Identifier.91b58315.jpeg",t="/ran/assets/Statement.98dca542.jpeg",c="/ran/assets/Declaration.1f7b4e23.jpeg",r="/ran/assets/Expression.d8010e56.jpeg",D="/ran/assets/ExpressionStatement.15a42b4e.jpeg",y="/ran/assets/Class.614486e5.jpeg",F="/ran/assets/import.32c7331a.jpeg",B="/ran/assets/export.84f41d4d.jpeg",i="/ran/assets/Program.e2066113.jpeg",d="/ran/assets/File.9a43a095.jpeg",E="/ran/assets/axtexplorer.2ebc951a.jpeg",A="/ran/assets/axtexplorerSave.21965993.jpeg",s="/ran/assets/Comment.4c895119.jpeg",C="/ran/assets/extra.ff9e7b7d.jpeg",v=JSON.parse('{"title":"Abstract Syntax Tree","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/astParse/tokenizer.md","lastUpdated":1694419899000}'),u={name:"src/article/astParse/tokenizer.md"},h=o('

Abstract Syntax Tree

一.(abstract syntax tree)抽象语法树的作用

源码是一串按照语法格式来组织的字符串,人能够认识,但是计算机并不认识,想让计算机认识就要转成一种数据结构,通过不同的对象来保存不同的数据,并且按照依赖关系组织起来,这种数据结构就是抽象语法树(abstract syntax tree)。

之所以叫“抽象”语法树是因为数据结构中省略掉了一些无具体意义的分隔符比如 ; { } 等。

有了 AST,计算机就能理解源码字符串的意思,而理解是能够转换的前提,所以编译的第一步需要把源码 parseAST

转成 AST 之后就可以通过修改 AST ,分析 AST 的方式来修改和分析代码,比如 babel 就通过这种方式进行代码的转换,比如 rollupTree Shaking ,就是通过分析 AST的 导入导出语法,从而分析出没有使用的代码,进行去除。

二.常见的 AST 节点

常见的 AST 节点 AST 是对源码的抽象,字面量、标识符、表达式、语句、模块语法、class 语法都有各自的 AST。

我们分别来了解一下:

Literal

Literal 是字面量的意思,比如 let name = 'value'中,'value'就是一个字符串字面量 StringLiteral,相应的还有数字字面量 NumericLiteral,布尔字面量 BooleanLiteral,字符串字面量 StringLiteral,正则表达式字面量 RegExpLiteral 等。

下面这些字面量都有对应的 Literal 节点:

代码中的字面量很多,babel 就是通过 xxLiteral 来抽象这部分内容的。

Identifier

Identifer 是标识符的意思,变量名、属性名、参数名等各种声明和引用的名字,都是Identifer

我们知道, JS 中的标识符只能包含字母或数字或下划线 (“_”) 或美元符号 (“$”) ,且不能以数字开头。这是 Identifier 的词法特点。

尝试分析一下,下面这一段代码里面有多少 Identifier 呢?

js
const name = 'value'
 
 function say(name) {
   console.log(name)
diff --git a/assets/src_article_astParse_tokenizer.md.ccc361f6.lean.js b/assets/src_article_astParse_tokenizer.md.99a1983d.lean.js
similarity index 93%
rename from assets/src_article_astParse_tokenizer.md.ccc361f6.lean.js
rename to assets/src_article_astParse_tokenizer.md.99a1983d.lean.js
index 3e0c08532..395d4b42e 100644
--- a/assets/src_article_astParse_tokenizer.md.ccc361f6.lean.js
+++ b/assets/src_article_astParse_tokenizer.md.99a1983d.lean.js
@@ -1 +1 @@
-import{_ as a,o as n,c as l,N as o}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/Literal.4f8173cf.jpeg",e="/ran/assets/Identifier.91b58315.jpeg",t="/ran/assets/Statement.98dca542.jpeg",c="/ran/assets/Declaration.1f7b4e23.jpeg",r="/ran/assets/Expression.d8010e56.jpeg",D="/ran/assets/ExpressionStatement.15a42b4e.jpeg",y="/ran/assets/Class.614486e5.jpeg",F="/ran/assets/import.32c7331a.jpeg",B="/ran/assets/export.84f41d4d.jpeg",i="/ran/assets/Program.e2066113.jpeg",d="/ran/assets/File.9a43a095.jpeg",E="/ran/assets/axtexplorer.2ebc951a.jpeg",A="/ran/assets/axtexplorerSave.21965993.jpeg",s="/ran/assets/Comment.4c895119.jpeg",C="/ran/assets/extra.ff9e7b7d.jpeg",v=JSON.parse('{"title":"Abstract Syntax Tree","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/astParse/tokenizer.md","lastUpdated":1694276611000}'),u={name:"src/article/astParse/tokenizer.md"},h=o("",150),f=[h];function m(g,b,T,k,x,_){return n(),l("div",null,f)}const j=a(u,[["render",m]]);export{v as __pageData,j as default};
+import{_ as a,o as n,c as l,N as o}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/Literal.4f8173cf.jpeg",e="/ran/assets/Identifier.91b58315.jpeg",t="/ran/assets/Statement.98dca542.jpeg",c="/ran/assets/Declaration.1f7b4e23.jpeg",r="/ran/assets/Expression.d8010e56.jpeg",D="/ran/assets/ExpressionStatement.15a42b4e.jpeg",y="/ran/assets/Class.614486e5.jpeg",F="/ran/assets/import.32c7331a.jpeg",B="/ran/assets/export.84f41d4d.jpeg",i="/ran/assets/Program.e2066113.jpeg",d="/ran/assets/File.9a43a095.jpeg",E="/ran/assets/axtexplorer.2ebc951a.jpeg",A="/ran/assets/axtexplorerSave.21965993.jpeg",s="/ran/assets/Comment.4c895119.jpeg",C="/ran/assets/extra.ff9e7b7d.jpeg",v=JSON.parse('{"title":"Abstract Syntax Tree","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/astParse/tokenizer.md","lastUpdated":1694419899000}'),u={name:"src/article/astParse/tokenizer.md"},h=o("",150),f=[h];function m(g,b,T,k,x,_){return n(),l("div",null,f)}const j=a(u,[["render",m]]);export{v as __pageData,j as default};
diff --git a/assets/src_article_babel.md.31e21b4d.js b/assets/src_article_babel.md.90287ca6.js
similarity index 95%
rename from assets/src_article_babel.md.31e21b4d.js
rename to assets/src_article_babel.md.90287ca6.js
index 6db16cd0b..c1a06612c 100644
--- a/assets/src_article_babel.md.31e21b4d.js
+++ b/assets/src_article_babel.md.90287ca6.js
@@ -1 +1 @@
-import{_ as e,o as a,c as t,N as l}from"./chunks/framework.6fe2e870.js";const u=JSON.parse('{"title":"Babel","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/babel.md","lastUpdated":1694276611000}'),r={name:"src/article/babel.md"},i=l('

Babel

babel 核心库主要是:

  • @babel/parser 对源码进行 parse,可以通过 plugins、sourceType 等来指定 parse 语法,功能是把源码转成 AST。
  • @babel/traverse 通过 visitor 函数对遍历到的 ast 进行处理,分为 enter 和 exit 两个阶段,具体操作 AST 使用 path 的 api,还可以通过 state 来在遍历过程中传递一些数据
  • @babel/types 用于创建、判断 AST 节点,提供了 xxx、isXxx、assertXxx 的 api
  • @babel/template 当需要批量创建 AST 的时候可以使用 @babel/template 来简化 AST 创建逻辑。
  • @babel/code-frame 可以创建友好的报错信息
  • @babel/generator 打印 AST 成目标代码字符串,支持 comments、minified、sourceMaps 等选项。
  • @babel/core 基于上面的包来完成 babel 的编译流程,并应用 plugin 和 preset。
',3),s=[i];function b(o,c,p,n,_,d){return a(),t("div",null,s)}const T=e(r,[["render",b]]);export{u as __pageData,T as default}; +import{_ as e,o as a,c as t,N as l}from"./chunks/framework.6fe2e870.js";const u=JSON.parse('{"title":"Babel","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/babel.md","lastUpdated":1694419899000}'),r={name:"src/article/babel.md"},i=l('

Babel

babel 核心库主要是:

  • @babel/parser 对源码进行 parse,可以通过 plugins、sourceType 等来指定 parse 语法,功能是把源码转成 AST。
  • @babel/traverse 通过 visitor 函数对遍历到的 ast 进行处理,分为 enter 和 exit 两个阶段,具体操作 AST 使用 path 的 api,还可以通过 state 来在遍历过程中传递一些数据
  • @babel/types 用于创建、判断 AST 节点,提供了 xxx、isXxx、assertXxx 的 api
  • @babel/template 当需要批量创建 AST 的时候可以使用 @babel/template 来简化 AST 创建逻辑。
  • @babel/code-frame 可以创建友好的报错信息
  • @babel/generator 打印 AST 成目标代码字符串,支持 comments、minified、sourceMaps 等选项。
  • @babel/core 基于上面的包来完成 babel 的编译流程,并应用 plugin 和 preset。
',3),s=[i];function b(o,c,p,n,_,d){return a(),t("div",null,s)}const T=e(r,[["render",b]]);export{u as __pageData,T as default}; diff --git a/assets/src_article_babel.md.31e21b4d.lean.js b/assets/src_article_babel.md.90287ca6.lean.js similarity index 83% rename from assets/src_article_babel.md.31e21b4d.lean.js rename to assets/src_article_babel.md.90287ca6.lean.js index e642ea63d..ff63c76cc 100644 --- a/assets/src_article_babel.md.31e21b4d.lean.js +++ b/assets/src_article_babel.md.90287ca6.lean.js @@ -1 +1 @@ -import{_ as e,o as a,c as t,N as l}from"./chunks/framework.6fe2e870.js";const u=JSON.parse('{"title":"Babel","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/babel.md","lastUpdated":1694276611000}'),r={name:"src/article/babel.md"},i=l("",3),s=[i];function b(o,c,p,n,_,d){return a(),t("div",null,s)}const T=e(r,[["render",b]]);export{u as __pageData,T as default}; +import{_ as e,o as a,c as t,N as l}from"./chunks/framework.6fe2e870.js";const u=JSON.parse('{"title":"Babel","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/babel.md","lastUpdated":1694419899000}'),r={name:"src/article/babel.md"},i=l("",3),s=[i];function b(o,c,p,n,_,d){return a(),t("div",null,s)}const T=e(r,[["render",b]]);export{u as __pageData,T as default}; diff --git a/assets/src_article_bundle.md.7a4db991.js b/assets/src_article_bundle.md.1794ffc6.js similarity index 93% rename from assets/src_article_bundle.md.7a4db991.js rename to assets/src_article_bundle.md.1794ffc6.js index f15d76302..360ed3a7e 100644 --- a/assets/src_article_bundle.md.7a4db991.js +++ b/assets/src_article_bundle.md.1794ffc6.js @@ -1 +1 @@ -import{_ as l,o as t,c as a,x as e,a as n}from"./chunks/framework.6fe2e870.js";const B=JSON.parse('{"title":"Bundle","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/bundle.md","lastUpdated":1694276611000}'),s={name:"src/article/bundle.md"},d=e("h1",{id:"bundle",tabindex:"-1"},[n("Bundle "),e("a",{class:"header-anchor",href:"#bundle","aria-label":'Permalink to "Bundle"'},"​")],-1),o=e("p",null,"Bundle 的本质就是输入,转换,输出。在机器上直接运行的代码,往往都难以维护和理解,我们需要将开发者方便理解和维护的代码,通过打包等工具转换成方便机器或者程序使用的代码。对于 web 前端来说,打包工具,至少需要以下功能:",-1),r=e("ul",null,[e("li",null,"编译能力"),e("li",null,"插件机制"),e("li",null,"HMR"),e("li",null,"cli 和命令行能力")],-1),c=[d,o,r];function i(u,_,p,h,m,b){return t(),a("div",null,c)}const x=l(s,[["render",i]]);export{B as __pageData,x as default}; +import{_ as l,o as t,c as a,x as e,a as n}from"./chunks/framework.6fe2e870.js";const B=JSON.parse('{"title":"Bundle","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/bundle.md","lastUpdated":1694419899000}'),s={name:"src/article/bundle.md"},d=e("h1",{id:"bundle",tabindex:"-1"},[n("Bundle "),e("a",{class:"header-anchor",href:"#bundle","aria-label":'Permalink to "Bundle"'},"​")],-1),o=e("p",null,"Bundle 的本质就是输入,转换,输出。在机器上直接运行的代码,往往都难以维护和理解,我们需要将开发者方便理解和维护的代码,通过打包等工具转换成方便机器或者程序使用的代码。对于 web 前端来说,打包工具,至少需要以下功能:",-1),r=e("ul",null,[e("li",null,"编译能力"),e("li",null,"插件机制"),e("li",null,"HMR"),e("li",null,"cli 和命令行能力")],-1),c=[d,o,r];function i(u,_,p,h,m,b){return t(),a("div",null,c)}const x=l(s,[["render",i]]);export{B as __pageData,x as default}; diff --git a/assets/src_article_bundle.md.7a4db991.lean.js b/assets/src_article_bundle.md.1794ffc6.lean.js similarity index 93% rename from assets/src_article_bundle.md.7a4db991.lean.js rename to assets/src_article_bundle.md.1794ffc6.lean.js index f15d76302..360ed3a7e 100644 --- a/assets/src_article_bundle.md.7a4db991.lean.js +++ b/assets/src_article_bundle.md.1794ffc6.lean.js @@ -1 +1 @@ -import{_ as l,o as t,c as a,x as e,a as n}from"./chunks/framework.6fe2e870.js";const B=JSON.parse('{"title":"Bundle","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/bundle.md","lastUpdated":1694276611000}'),s={name:"src/article/bundle.md"},d=e("h1",{id:"bundle",tabindex:"-1"},[n("Bundle "),e("a",{class:"header-anchor",href:"#bundle","aria-label":'Permalink to "Bundle"'},"​")],-1),o=e("p",null,"Bundle 的本质就是输入,转换,输出。在机器上直接运行的代码,往往都难以维护和理解,我们需要将开发者方便理解和维护的代码,通过打包等工具转换成方便机器或者程序使用的代码。对于 web 前端来说,打包工具,至少需要以下功能:",-1),r=e("ul",null,[e("li",null,"编译能力"),e("li",null,"插件机制"),e("li",null,"HMR"),e("li",null,"cli 和命令行能力")],-1),c=[d,o,r];function i(u,_,p,h,m,b){return t(),a("div",null,c)}const x=l(s,[["render",i]]);export{B as __pageData,x as default}; +import{_ as l,o as t,c as a,x as e,a as n}from"./chunks/framework.6fe2e870.js";const B=JSON.parse('{"title":"Bundle","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/bundle.md","lastUpdated":1694419899000}'),s={name:"src/article/bundle.md"},d=e("h1",{id:"bundle",tabindex:"-1"},[n("Bundle "),e("a",{class:"header-anchor",href:"#bundle","aria-label":'Permalink to "Bundle"'},"​")],-1),o=e("p",null,"Bundle 的本质就是输入,转换,输出。在机器上直接运行的代码,往往都难以维护和理解,我们需要将开发者方便理解和维护的代码,通过打包等工具转换成方便机器或者程序使用的代码。对于 web 前端来说,打包工具,至少需要以下功能:",-1),r=e("ul",null,[e("li",null,"编译能力"),e("li",null,"插件机制"),e("li",null,"HMR"),e("li",null,"cli 和命令行能力")],-1),c=[d,o,r];function i(u,_,p,h,m,b){return t(),a("div",null,c)}const x=l(s,[["render",i]]);export{B as __pageData,x as default}; diff --git a/assets/src_article_designMode.md.fdb686ab.js b/assets/src_article_designMode.md.c8355f94.js similarity index 99% rename from assets/src_article_designMode.md.fdb686ab.js rename to assets/src_article_designMode.md.c8355f94.js index 7de719ff6..8922ef965 100644 --- a/assets/src_article_designMode.md.fdb686ab.js +++ b/assets/src_article_designMode.md.c8355f94.js @@ -1,4 +1,4 @@ -import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/继承.d41dee10.png",o="",e="/ran/assets/组合.df5fa5b1.png",t="",c="",r="",y="",F="",D="/ran/assets/抽象工厂.786dc879.png",A="/ran/assets/单例.c53b9bde.jpg",B="/ran/assets/建造者.8aaf3334.jpeg",i="/ran/assets/原型.38aa29a0.jpg",E="/ran/assets/装饰.54bb2dbb.jpg",d="/ran/assets/外观.77eeb1fd.png",C="",u="/ran/assets/享元.6ee46b37.png",h="/ran/assets/桥接.de284493.png",f="/ran/assets/适配器.1e901e8e.png",g="",m="",k="/ran/assets/解释器.07d142dd.jpg",b="",v="",R="/ran/assets/备忘录.f4491fa8.jpg",q="",P="/ran/assets/状态.10166acb.png",x="/ran/assets/策略.3d82c7be.png",V="",S="/ran/assets/访问者.4ef81e5c.png",L=JSON.parse('{"title":"23 种经典设计模式","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/designMode.md","lastUpdated":1694276611000}'),J={name:"src/article/designMode.md"},M=l(`

23 种经典设计模式

设计模式 Design Pattern 是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结,使用设计模式是为了可重用代码、让代码更容易被他人理解并且保证代码可靠性。。

在《设计模式:可复用面向对象软件的基础》一书中所介绍的 23 种经典设计模式,不过设计模式并不仅仅只有这 23 种,随着软件开发行业的发展,越来越多的新模式不断诞生并得以应用。有经验的开发者在学习设计模式可以和过往的经验互相印证,更容易理解这些设计模式。

设计模式一般包含模式名称、问题、目的、解决方案、效果等组成要素。问题描述了应该在何时使用模式,它包含了设计中存在的问题以及问题存在的原因。解决方案描述了一个设计模式的组成成分,以及这些组成成分之间的相互关系,各自的职责和协作方式,通常解决方案通过 UML 类图和核心代码来进行描述。效果描述了模式的优缺点以及在使用模式时应权衡的问题。

为什么要学习设计模式:

  • 设计模式来源众多专家的经验和智慧,它们是从许多优秀的软件系统中总结出的成功的、能够实现可维护性复用的设计方案,使用这些方案将可以让我们避免做一些重复性的工作

  • 设计模式提供了一套通用的设计词汇和一种通用的形式来方便开发人员之间沟通和交流,使得设计方案更加通俗易懂

  • 大部分设计模式都兼顾了系统的可重用性和可扩展性,这使得我们可以更好地重用一些已有的设计方案、功能模块甚至一个完整的软件系统,避免我们经常做一些重复的设计、编写一些重复的代码

  • 合理使用设计模式并对设计模式的使用情况进行文档化,将有助于别人更快地理解系统

  • 学习设计模式将有助于初学者更加深入地理解面向对象思想

储备知识

  • 抽象类:一般抽象类都是作为基类,比如说「电脑」就可以作为一个抽象类,根据抽象类派生出「台式电脑」和「笔记本电脑」2 种具体类。一般不对抽象类进行实例化。

  • 组合优于继承:不能滥用继承来拓展功能,配合组合会更灵活。同样拿「电脑」抽象类来举例,如果使用继承,区分不同类型的「电脑」我们可以派生出「台式电脑」和「笔记本电脑」,如果再增加一个维度,根据品牌又能继续细分出「联想台式电脑」、「联想笔记本电脑」、「苹果台式电脑」和「苹果笔记本电脑」等等,如果再增加一个维度继续细分下去,显然继承是无法胜任的。这个时候可以使用继承加组合方式,组合的对象也可以进行抽象化设计:

    ts
    // 品牌
    +import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/继承.d41dee10.png",o="",e="/ran/assets/组合.df5fa5b1.png",t="",c="",r="",y="",F="",D="/ran/assets/抽象工厂.786dc879.png",A="/ran/assets/单例.c53b9bde.jpg",B="/ran/assets/建造者.8aaf3334.jpeg",i="/ran/assets/原型.38aa29a0.jpg",E="/ran/assets/装饰.54bb2dbb.jpg",d="/ran/assets/外观.77eeb1fd.png",C="",u="/ran/assets/享元.6ee46b37.png",h="/ran/assets/桥接.de284493.png",f="/ran/assets/适配器.1e901e8e.png",g="",m="",k="/ran/assets/解释器.07d142dd.jpg",b="",v="",R="/ran/assets/备忘录.f4491fa8.jpg",q="",P="/ran/assets/状态.10166acb.png",x="/ran/assets/策略.3d82c7be.png",V="",S="/ran/assets/访问者.4ef81e5c.png",L=JSON.parse('{"title":"23 种经典设计模式","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/designMode.md","lastUpdated":1694419899000}'),J={name:"src/article/designMode.md"},M=l(`

    23 种经典设计模式

    设计模式 Design Pattern 是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结,使用设计模式是为了可重用代码、让代码更容易被他人理解并且保证代码可靠性。。

    在《设计模式:可复用面向对象软件的基础》一书中所介绍的 23 种经典设计模式,不过设计模式并不仅仅只有这 23 种,随着软件开发行业的发展,越来越多的新模式不断诞生并得以应用。有经验的开发者在学习设计模式可以和过往的经验互相印证,更容易理解这些设计模式。

    设计模式一般包含模式名称、问题、目的、解决方案、效果等组成要素。问题描述了应该在何时使用模式,它包含了设计中存在的问题以及问题存在的原因。解决方案描述了一个设计模式的组成成分,以及这些组成成分之间的相互关系,各自的职责和协作方式,通常解决方案通过 UML 类图和核心代码来进行描述。效果描述了模式的优缺点以及在使用模式时应权衡的问题。

    为什么要学习设计模式:

    • 设计模式来源众多专家的经验和智慧,它们是从许多优秀的软件系统中总结出的成功的、能够实现可维护性复用的设计方案,使用这些方案将可以让我们避免做一些重复性的工作

    • 设计模式提供了一套通用的设计词汇和一种通用的形式来方便开发人员之间沟通和交流,使得设计方案更加通俗易懂

    • 大部分设计模式都兼顾了系统的可重用性和可扩展性,这使得我们可以更好地重用一些已有的设计方案、功能模块甚至一个完整的软件系统,避免我们经常做一些重复的设计、编写一些重复的代码

    • 合理使用设计模式并对设计模式的使用情况进行文档化,将有助于别人更快地理解系统

    • 学习设计模式将有助于初学者更加深入地理解面向对象思想

    储备知识

    • 抽象类:一般抽象类都是作为基类,比如说「电脑」就可以作为一个抽象类,根据抽象类派生出「台式电脑」和「笔记本电脑」2 种具体类。一般不对抽象类进行实例化。

    • 组合优于继承:不能滥用继承来拓展功能,配合组合会更灵活。同样拿「电脑」抽象类来举例,如果使用继承,区分不同类型的「电脑」我们可以派生出「台式电脑」和「笔记本电脑」,如果再增加一个维度,根据品牌又能继续细分出「联想台式电脑」、「联想笔记本电脑」、「苹果台式电脑」和「苹果笔记本电脑」等等,如果再增加一个维度继续细分下去,显然继承是无法胜任的。这个时候可以使用继承加组合方式,组合的对象也可以进行抽象化设计:

      ts
      // 品牌
       interface Brand {
         // ...
       }
      diff --git a/assets/src_article_designMode.md.fdb686ab.lean.js b/assets/src_article_designMode.md.c8355f94.lean.js
      similarity index 99%
      rename from assets/src_article_designMode.md.fdb686ab.lean.js
      rename to assets/src_article_designMode.md.c8355f94.lean.js
      index 158609ba6..3089cf77d 100644
      --- a/assets/src_article_designMode.md.fdb686ab.lean.js
      +++ b/assets/src_article_designMode.md.c8355f94.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/继承.d41dee10.png",o="",e="/ran/assets/组合.df5fa5b1.png",t="",c="",r="",y="",F="",D="/ran/assets/抽象工厂.786dc879.png",A="/ran/assets/单例.c53b9bde.jpg",B="/ran/assets/建造者.8aaf3334.jpeg",i="/ran/assets/原型.38aa29a0.jpg",E="/ran/assets/装饰.54bb2dbb.jpg",d="/ran/assets/外观.77eeb1fd.png",C="",u="/ran/assets/享元.6ee46b37.png",h="/ran/assets/桥接.de284493.png",f="/ran/assets/适配器.1e901e8e.png",g="",m="",k="/ran/assets/解释器.07d142dd.jpg",b="",v="",R="/ran/assets/备忘录.f4491fa8.jpg",q="",P="/ran/assets/状态.10166acb.png",x="/ran/assets/策略.3d82c7be.png",V="",S="/ran/assets/访问者.4ef81e5c.png",L=JSON.parse('{"title":"23 种经典设计模式","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/designMode.md","lastUpdated":1694276611000}'),J={name:"src/article/designMode.md"},M=l("",384),j=[M];function N(Q,O,I,Z,U,w){return n(),a("div",null,j)}const K=s(J,[["render",N]]);export{L as __pageData,K as default};
      +import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/继承.d41dee10.png",o="",e="/ran/assets/组合.df5fa5b1.png",t="",c="",r="",y="",F="",D="/ran/assets/抽象工厂.786dc879.png",A="/ran/assets/单例.c53b9bde.jpg",B="/ran/assets/建造者.8aaf3334.jpeg",i="/ran/assets/原型.38aa29a0.jpg",E="/ran/assets/装饰.54bb2dbb.jpg",d="/ran/assets/外观.77eeb1fd.png",C="",u="/ran/assets/享元.6ee46b37.png",h="/ran/assets/桥接.de284493.png",f="/ran/assets/适配器.1e901e8e.png",g="",m="",k="/ran/assets/解释器.07d142dd.jpg",b="",v="",R="/ran/assets/备忘录.f4491fa8.jpg",q="",P="/ran/assets/状态.10166acb.png",x="/ran/assets/策略.3d82c7be.png",V="",S="/ran/assets/访问者.4ef81e5c.png",L=JSON.parse('{"title":"23 种经典设计模式","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/designMode.md","lastUpdated":1694419899000}'),J={name:"src/article/designMode.md"},M=l("",384),j=[M];function N(Q,O,I,Z,U,w){return n(),a("div",null,j)}const K=s(J,[["render",N]]);export{L as __pageData,K as default};
      diff --git a/assets/src_article_functionalProgramming.md.e9f945b3.js b/assets/src_article_functionalProgramming.md.0ef4df92.js
      similarity index 99%
      rename from assets/src_article_functionalProgramming.md.e9f945b3.js
      rename to assets/src_article_functionalProgramming.md.0ef4df92.js
      index d6b57947b..7bd8fb596 100644
      --- a/assets/src_article_functionalProgramming.md.e9f945b3.js
      +++ b/assets/src_article_functionalProgramming.md.0ef4df92.js
      @@ -1,4 +1,4 @@
      -import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"函数式编程","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/functionalProgramming.md","lastUpdated":1694276611000}'),p={name:"src/article/functionalProgramming.md"},o=l(`

      函数式编程

      • 概述: 函数式编程(Functional Programming)FP就是编程规范之一,我们常听说的编程规范还有面向对象编程,面向过程编程。
      • 面向对象的编程思维方式:把现实世界中的事物抽象成程序世界的类和对象,通过封装,继承和多态演示事物事件的联系
      • 函数编程的思维方式:把现实世界的事物和事物之间的联系抽象到程序世界(对运算过程进行抽象)
        • 程序的本质:根据输入,通过某种运算,获得相应的输出
        • 函数式编程中的函数不是指程序中的(函数)方法,而是数学中的函数,即映射关系
        • 相同的输入始终要得到相同的输出(纯函数)
        • 函数式编程用来描述数据(函数)之间的映射关系
      js
      //非函数式编程,面向过程的编程方式
      +import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"函数式编程","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/functionalProgramming.md","lastUpdated":1694419899000}'),p={name:"src/article/functionalProgramming.md"},o=l(`

      函数式编程

      • 概述: 函数式编程(Functional Programming)FP就是编程规范之一,我们常听说的编程规范还有面向对象编程,面向过程编程。
      • 面向对象的编程思维方式:把现实世界中的事物抽象成程序世界的类和对象,通过封装,继承和多态演示事物事件的联系
      • 函数编程的思维方式:把现实世界的事物和事物之间的联系抽象到程序世界(对运算过程进行抽象)
        • 程序的本质:根据输入,通过某种运算,获得相应的输出
        • 函数式编程中的函数不是指程序中的(函数)方法,而是数学中的函数,即映射关系
        • 相同的输入始终要得到相同的输出(纯函数)
        • 函数式编程用来描述数据(函数)之间的映射关系
      js
      //非函数式编程,面向过程的编程方式
       let num1 = 1
       let num2 = 2
       let sum = num1 + num2
      diff --git a/assets/src_article_functionalProgramming.md.e9f945b3.lean.js b/assets/src_article_functionalProgramming.md.0ef4df92.lean.js
      similarity index 85%
      rename from assets/src_article_functionalProgramming.md.e9f945b3.lean.js
      rename to assets/src_article_functionalProgramming.md.0ef4df92.lean.js
      index 0690f21c7..2b4964e36 100644
      --- a/assets/src_article_functionalProgramming.md.e9f945b3.lean.js
      +++ b/assets/src_article_functionalProgramming.md.0ef4df92.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"函数式编程","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/functionalProgramming.md","lastUpdated":1694276611000}'),p={name:"src/article/functionalProgramming.md"},o=l("",66),e=[o];function t(c,r,y,F,D,B){return n(),a("div",null,e)}const E=s(p,[["render",t]]);export{A as __pageData,E as default};
      +import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"函数式编程","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/functionalProgramming.md","lastUpdated":1694419899000}'),p={name:"src/article/functionalProgramming.md"},o=l("",66),e=[o];function t(c,r,y,F,D,B){return n(),a("div",null,e)}const E=s(p,[["render",t]]);export{A as __pageData,E as default};
      diff --git a/assets/src_article_imagemin.md.05c1ce1a.js b/assets/src_article_imagemin.md.32ccabea.js
      similarity index 90%
      rename from assets/src_article_imagemin.md.05c1ce1a.js
      rename to assets/src_article_imagemin.md.32ccabea.js
      index 6c78ad6ac..cf300c7eb 100644
      --- a/assets/src_article_imagemin.md.05c1ce1a.js
      +++ b/assets/src_article_imagemin.md.32ccabea.js
      @@ -1 +1 @@
      -import{_ as a,o as t,c as i,x as e,a as n}from"./chunks/framework.6fe2e870.js";const f=JSON.parse('{"title":"imagemin 图片压缩源码分析","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/imagemin.md","lastUpdated":1694276611000}'),r={name:"src/article/imagemin.md"},s=e("h1",{id:"imagemin-图片压缩源码分析",tabindex:"-1"},[n("imagemin 图片压缩源码分析 "),e("a",{class:"header-anchor",href:"#imagemin-图片压缩源码分析","aria-label":'Permalink to "imagemin 图片压缩源码分析"'},"​")],-1),c=[s];function m(o,d,l,_,p,g){return t(),i("div",null,c)}const x=a(r,[["render",m]]);export{f as __pageData,x as default};
      +import{_ as a,o as t,c as i,x as e,a as n}from"./chunks/framework.6fe2e870.js";const f=JSON.parse('{"title":"imagemin 图片压缩源码分析","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/imagemin.md","lastUpdated":1694419899000}'),r={name:"src/article/imagemin.md"},s=e("h1",{id:"imagemin-图片压缩源码分析",tabindex:"-1"},[n("imagemin 图片压缩源码分析 "),e("a",{class:"header-anchor",href:"#imagemin-图片压缩源码分析","aria-label":'Permalink to "imagemin 图片压缩源码分析"'},"​")],-1),c=[s];function m(o,d,l,_,p,g){return t(),i("div",null,c)}const x=a(r,[["render",m]]);export{f as __pageData,x as default};
      diff --git a/assets/src_article_imagemin.md.05c1ce1a.lean.js b/assets/src_article_imagemin.md.32ccabea.lean.js
      similarity index 90%
      rename from assets/src_article_imagemin.md.05c1ce1a.lean.js
      rename to assets/src_article_imagemin.md.32ccabea.lean.js
      index 6c78ad6ac..cf300c7eb 100644
      --- a/assets/src_article_imagemin.md.05c1ce1a.lean.js
      +++ b/assets/src_article_imagemin.md.32ccabea.lean.js
      @@ -1 +1 @@
      -import{_ as a,o as t,c as i,x as e,a as n}from"./chunks/framework.6fe2e870.js";const f=JSON.parse('{"title":"imagemin 图片压缩源码分析","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/imagemin.md","lastUpdated":1694276611000}'),r={name:"src/article/imagemin.md"},s=e("h1",{id:"imagemin-图片压缩源码分析",tabindex:"-1"},[n("imagemin 图片压缩源码分析 "),e("a",{class:"header-anchor",href:"#imagemin-图片压缩源码分析","aria-label":'Permalink to "imagemin 图片压缩源码分析"'},"​")],-1),c=[s];function m(o,d,l,_,p,g){return t(),i("div",null,c)}const x=a(r,[["render",m]]);export{f as __pageData,x as default};
      +import{_ as a,o as t,c as i,x as e,a as n}from"./chunks/framework.6fe2e870.js";const f=JSON.parse('{"title":"imagemin 图片压缩源码分析","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/imagemin.md","lastUpdated":1694419899000}'),r={name:"src/article/imagemin.md"},s=e("h1",{id:"imagemin-图片压缩源码分析",tabindex:"-1"},[n("imagemin 图片压缩源码分析 "),e("a",{class:"header-anchor",href:"#imagemin-图片压缩源码分析","aria-label":'Permalink to "imagemin 图片压缩源码分析"'},"​")],-1),c=[s];function m(o,d,l,_,p,g){return t(),i("div",null,c)}const x=a(r,[["render",m]]);export{f as __pageData,x as default};
      diff --git a/assets/src_article_javascript_domLoad.md.c8afd20d.js b/assets/src_article_javascript_domLoad.md.f4dd6368.js
      similarity index 97%
      rename from assets/src_article_javascript_domLoad.md.c8afd20d.js
      rename to assets/src_article_javascript_domLoad.md.f4dd6368.js
      index db3108062..aa0dfd601 100644
      --- a/assets/src_article_javascript_domLoad.md.c8afd20d.js
      +++ b/assets/src_article_javascript_domLoad.md.f4dd6368.js
      @@ -1 +1 @@
      -import{_ as a,o as e,c as o,N as t}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"页面加载完成后事件","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/javascript/domLoad.md","lastUpdated":1694276611000}'),n={name:"src/article/javascript/domLoad.md"},s=t('

      页面加载完成后事件

      window.onload

      DOMContentLoaded

      js
      document.addEventListener('DOMContentLoaded', fun)

      <body onload="fun()">

      readyState

      js
      document.readyState\n\ndocument.onreadystatechange

      一个文档的 readyState 可以是以下之一:

      • loading / 加载 。document 仍在加载。
      • interactive / 互动。文档已经完成加载,文档已被解析,但是诸如图像,样式表和框架之类的子资源仍在加载。
      • complete / 完成。T 文档和所有子资源已完成加载。状态表示 load 事件即将被触发。
      ',9),l=[s];function d(r,c,p,i,h,u){return e(),o("div",null,l)}const _=a(n,[["render",d]]);export{m as __pageData,_ as default}; +import{_ as a,o as e,c as o,N as t}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"页面加载完成后事件","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/javascript/domLoad.md","lastUpdated":1694419899000}'),n={name:"src/article/javascript/domLoad.md"},s=t('

      页面加载完成后事件

      window.onload

      DOMContentLoaded

      js
      document.addEventListener('DOMContentLoaded', fun)

      <body onload="fun()">

      readyState

      js
      document.readyState\n\ndocument.onreadystatechange

      一个文档的 readyState 可以是以下之一:

      • loading / 加载 。document 仍在加载。
      • interactive / 互动。文档已经完成加载,文档已被解析,但是诸如图像,样式表和框架之类的子资源仍在加载。
      • complete / 完成。T 文档和所有子资源已完成加载。状态表示 load 事件即将被触发。
      ',9),l=[s];function d(r,c,p,i,h,u){return e(),o("div",null,l)}const _=a(n,[["render",d]]);export{m as __pageData,_ as default}; diff --git a/assets/src_article_javascript_domLoad.md.c8afd20d.lean.js b/assets/src_article_javascript_domLoad.md.f4dd6368.lean.js similarity index 85% rename from assets/src_article_javascript_domLoad.md.c8afd20d.lean.js rename to assets/src_article_javascript_domLoad.md.f4dd6368.lean.js index 32d62741b..3e7627488 100644 --- a/assets/src_article_javascript_domLoad.md.c8afd20d.lean.js +++ b/assets/src_article_javascript_domLoad.md.f4dd6368.lean.js @@ -1 +1 @@ -import{_ as a,o as e,c as o,N as t}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"页面加载完成后事件","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/javascript/domLoad.md","lastUpdated":1694276611000}'),n={name:"src/article/javascript/domLoad.md"},s=t("",9),l=[s];function d(r,c,p,i,h,u){return e(),o("div",null,l)}const _=a(n,[["render",d]]);export{m as __pageData,_ as default}; +import{_ as a,o as e,c as o,N as t}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"页面加载完成后事件","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/javascript/domLoad.md","lastUpdated":1694419899000}'),n={name:"src/article/javascript/domLoad.md"},s=t("",9),l=[s];function d(r,c,p,i,h,u){return e(),o("div",null,l)}const _=a(n,[["render",d]]);export{m as __pageData,_ as default}; diff --git a/assets/src_article_typescript_calculate.md.f31139e2.js b/assets/src_article_typescript_calculate.md.9ee3c861.js similarity index 99% rename from assets/src_article_typescript_calculate.md.f31139e2.js rename to assets/src_article_typescript_calculate.md.9ee3c861.js index 0489fd518..7b3af38b4 100644 --- a/assets/src_article_typescript_calculate.md.f31139e2.js +++ b/assets/src_article_typescript_calculate.md.9ee3c861.js @@ -1,4 +1,4 @@ -import{_ as s,o as n,c as a,N as p}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"数组长度做计数","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/calculate.md","lastUpdated":1694276611000}'),l={name:"src/article/typescript/calculate.md"},o=p(`

      数组长度做计数

      类型系统不是图灵完备,各种逻辑都能写么,但好像没发现数值相关的逻辑。

      没错,数值相关的逻辑比较绕,被我单独摘了出来,就是这节要讲的内容。

      这是类型体操的第四个套路:数组长度做计数。

      TypeScript 类型系统没有加减乘除运算符,怎么做数值运算呢?

      不知道大家有没有注意到数组类型取 length 就是数值。

      比如:

      ts
      type num1 = [unknown]['length']
      +import{_ as s,o as n,c as a,N as p}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"数组长度做计数","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/calculate.md","lastUpdated":1694419899000}'),l={name:"src/article/typescript/calculate.md"},o=p(`

      数组长度做计数

      类型系统不是图灵完备,各种逻辑都能写么,但好像没发现数值相关的逻辑。

      没错,数值相关的逻辑比较绕,被我单独摘了出来,就是这节要讲的内容。

      这是类型体操的第四个套路:数组长度做计数。

      TypeScript 类型系统没有加减乘除运算符,怎么做数值运算呢?

      不知道大家有没有注意到数组类型取 length 就是数值。

      比如:

      ts
      type num1 = [unknown]['length']
       // type num1 = 1
       type num2 = [unknown, unknown]['length']
       // type num1 = 2
      diff --git a/assets/src_article_typescript_calculate.md.f31139e2.lean.js b/assets/src_article_typescript_calculate.md.9ee3c861.lean.js
      similarity index 85%
      rename from assets/src_article_typescript_calculate.md.f31139e2.lean.js
      rename to assets/src_article_typescript_calculate.md.9ee3c861.lean.js
      index c4a869194..32999e96b 100644
      --- a/assets/src_article_typescript_calculate.md.f31139e2.lean.js
      +++ b/assets/src_article_typescript_calculate.md.9ee3c861.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as n,c as a,N as p}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"数组长度做计数","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/calculate.md","lastUpdated":1694276611000}'),l={name:"src/article/typescript/calculate.md"},o=p("",99),e=[o];function t(r,c,B,y,F,D){return n(),a("div",null,e)}const i=s(l,[["render",t]]);export{E as __pageData,i as default};
      +import{_ as s,o as n,c as a,N as p}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"数组长度做计数","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/calculate.md","lastUpdated":1694419899000}'),l={name:"src/article/typescript/calculate.md"},o=p("",99),e=[o];function t(r,c,B,y,F,D){return n(),a("div",null,e)}const i=s(l,[["render",t]]);export{E as __pageData,i as default};
      diff --git a/assets/src_article_typescript_index.md.9450f83c.js b/assets/src_article_typescript_index.md.8330b87a.js
      similarity index 99%
      rename from assets/src_article_typescript_index.md.9450f83c.js
      rename to assets/src_article_typescript_index.md.8330b87a.js
      index bf4ab3087..74f0fd76a 100644
      --- a/assets/src_article_typescript_index.md.9450f83c.js
      +++ b/assets/src_article_typescript_index.md.8330b87a.js
      @@ -1,4 +1,4 @@
      -import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"TypeScript 的类型系统","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/index.md","lastUpdated":1694276611000}'),l={name:"src/article/typescript/index.md"},o=p(`

      TypeScript 的类型系统

      一.类型是什么

      类型具体点来说就是指 number、boolean、string 等基础类型和 Object、Function 等复合类型,它们是编程语言提供的对不同内容的抽象:

      • 不同类型变量占据的内存大小不同: boolean 类型的变量会分配 4 个字节的内存,而 number 类型的变量则会分配 8 个字节的内存,给变量声明了不同的类型就代表了会占据不同的内存空间。

      • 不同类型变量可做的操作不同: number 类型可以做加减乘除等运算,boolean 就不可以,复合类型中不同类型的对象可用的方法不同,比如 Date 和 RegExp,变量的类型不同代表可以对该变量做的操作就不同。

      有了类型,那我们的操作必须和类型相匹配,否则就会报错,这就是类型检查。

      如果能保证对某种类型只做该类型允许的操作,这就叫做类型安全。

      类型检查可以在运行时做,也可以运行之前的编译期做。这是两种不同的类型,前者叫做动态类型检查,后者叫做静态类型检查。

      两种类型检查各有优缺点。动态类型检查 在源码中不保留类型信息,对某个变量赋什么值、做什么操作都是允许的,写代码很灵活。但这也埋下了类型不安全的隐患,比如对 string 做了乘除,对 Date 对象调用了 exec 方法,这些都是运行时才能检查出来的错误。

      其中,最常见的错误应该是 “null is not an object”、“undefined is not a function” 之类的了,写代码时没发现类型不匹配,到了运行的时候才发现,就会有很多这种报错。

      所以,动态类型虽然代码写起来简单,但代码中很容易藏着一些类型不匹配的隐患。

      静态类型检查则是在源码中保留类型信息,声明变量要指定类型,对变量做的操作要和类型匹配,会有专门的编译器在编译期间做检查。

      静态类型给写代码增加了一些难度,因为你除了要考虑代码要表达的逻辑之外,还要考虑类型逻辑:变量是什么类型的、是不是匹配、要不要做类型转换等。

      不过,静态类型也消除了类型不安全的隐患,因为在编译期间就做了类型检查,就不会出现对 string 做了乘除,调用了 Date 的 exec 方法这类问题。

      所以,静态类型虽然代码写起来要考虑的问题多一些,会复杂一些,但是却消除了代码中潜藏类型不安全问题的可能。

      知道了动态类型检查和静态类型检查的区别,我们自然可以得出这样的结论:

      动态类型只适合简单的场景,对于大项目却不太合适,因为代码中可能藏着的隐患太多了,万一线上报一个类型不匹配的错误,那可能就是大问题。

      而静态类型虽然会增加写代码的成本,但是却能更好的保证代码的健壮性,减少 Bug 率。

      所以,大型项目注定会用静态类型语言开发。

      二.类型系统的分类

      1.简单的类型系统

      变量、函数、类等都可以声明类型,编译器会基于声明的类型做类型检查,类型不匹配时会报错。

      这是最基础的类型系统,能保证类型安全,但有些死板。

      比如一个 add 函数既可以做整数加法、又可以做浮点数加法,却需要声明两个函数:

      c
      int add(int a, int b) {
      +import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"TypeScript 的类型系统","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/index.md","lastUpdated":1694419899000}'),l={name:"src/article/typescript/index.md"},o=p(`

      TypeScript 的类型系统

      一.类型是什么

      类型具体点来说就是指 number、boolean、string 等基础类型和 Object、Function 等复合类型,它们是编程语言提供的对不同内容的抽象:

      • 不同类型变量占据的内存大小不同: boolean 类型的变量会分配 4 个字节的内存,而 number 类型的变量则会分配 8 个字节的内存,给变量声明了不同的类型就代表了会占据不同的内存空间。

      • 不同类型变量可做的操作不同: number 类型可以做加减乘除等运算,boolean 就不可以,复合类型中不同类型的对象可用的方法不同,比如 Date 和 RegExp,变量的类型不同代表可以对该变量做的操作就不同。

      有了类型,那我们的操作必须和类型相匹配,否则就会报错,这就是类型检查。

      如果能保证对某种类型只做该类型允许的操作,这就叫做类型安全。

      类型检查可以在运行时做,也可以运行之前的编译期做。这是两种不同的类型,前者叫做动态类型检查,后者叫做静态类型检查。

      两种类型检查各有优缺点。动态类型检查 在源码中不保留类型信息,对某个变量赋什么值、做什么操作都是允许的,写代码很灵活。但这也埋下了类型不安全的隐患,比如对 string 做了乘除,对 Date 对象调用了 exec 方法,这些都是运行时才能检查出来的错误。

      其中,最常见的错误应该是 “null is not an object”、“undefined is not a function” 之类的了,写代码时没发现类型不匹配,到了运行的时候才发现,就会有很多这种报错。

      所以,动态类型虽然代码写起来简单,但代码中很容易藏着一些类型不匹配的隐患。

      静态类型检查则是在源码中保留类型信息,声明变量要指定类型,对变量做的操作要和类型匹配,会有专门的编译器在编译期间做检查。

      静态类型给写代码增加了一些难度,因为你除了要考虑代码要表达的逻辑之外,还要考虑类型逻辑:变量是什么类型的、是不是匹配、要不要做类型转换等。

      不过,静态类型也消除了类型不安全的隐患,因为在编译期间就做了类型检查,就不会出现对 string 做了乘除,调用了 Date 的 exec 方法这类问题。

      所以,静态类型虽然代码写起来要考虑的问题多一些,会复杂一些,但是却消除了代码中潜藏类型不安全问题的可能。

      知道了动态类型检查和静态类型检查的区别,我们自然可以得出这样的结论:

      动态类型只适合简单的场景,对于大项目却不太合适,因为代码中可能藏着的隐患太多了,万一线上报一个类型不匹配的错误,那可能就是大问题。

      而静态类型虽然会增加写代码的成本,但是却能更好的保证代码的健壮性,减少 Bug 率。

      所以,大型项目注定会用静态类型语言开发。

      二.类型系统的分类

      1.简单的类型系统

      变量、函数、类等都可以声明类型,编译器会基于声明的类型做类型检查,类型不匹配时会报错。

      这是最基础的类型系统,能保证类型安全,但有些死板。

      比如一个 add 函数既可以做整数加法、又可以做浮点数加法,却需要声明两个函数:

      c
      int add(int a, int b) {
           return a + b;
       }
       
      diff --git a/assets/src_article_typescript_index.md.9450f83c.lean.js b/assets/src_article_typescript_index.md.8330b87a.lean.js
      similarity index 85%
      rename from assets/src_article_typescript_index.md.9450f83c.lean.js
      rename to assets/src_article_typescript_index.md.8330b87a.lean.js
      index 259692787..fd07944f3 100644
      --- a/assets/src_article_typescript_index.md.9450f83c.lean.js
      +++ b/assets/src_article_typescript_index.md.8330b87a.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"TypeScript 的类型系统","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/index.md","lastUpdated":1694276611000}'),l={name:"src/article/typescript/index.md"},o=p("",253),e=[o];function t(c,r,y,B,D,F){return a(),n("div",null,e)}const E=s(l,[["render",t]]);export{A as __pageData,E as default};
      +import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"TypeScript 的类型系统","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/index.md","lastUpdated":1694419899000}'),l={name:"src/article/typescript/index.md"},o=p("",253),e=[o];function t(c,r,y,B,D,F){return a(),n("div",null,e)}const E=s(l,[["render",t]]);export{A as __pageData,E as default};
      diff --git a/assets/src_article_typescript_pattern.md.e19bc8e4.js b/assets/src_article_typescript_pattern.md.ed4c198d.js
      similarity index 99%
      rename from assets/src_article_typescript_pattern.md.e19bc8e4.js
      rename to assets/src_article_typescript_pattern.md.ed4c198d.js
      index acf010848..19de70b3f 100644
      --- a/assets/src_article_typescript_pattern.md.e19bc8e4.js
      +++ b/assets/src_article_typescript_pattern.md.ed4c198d.js
      @@ -1,4 +1,4 @@
      -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"模式匹配提取","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/pattern.md","lastUpdated":1694276611000}'),p={name:"src/article/typescript/pattern.md"},o=l(`

      模式匹配提取

      字符串可以和正则做模式匹配,找到匹配的部分,提取子组,之后可以用 1,2 等引用匹配的子组。

      ts
      'abc'.replace(/a(b)c/, '$1,$1,$1')
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"模式匹配提取","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/pattern.md","lastUpdated":1694419899000}'),p={name:"src/article/typescript/pattern.md"},o=l(`

      模式匹配提取

      字符串可以和正则做模式匹配,找到匹配的部分,提取子组,之后可以用 1,2 等引用匹配的子组。

      ts
      'abc'.replace(/a(b)c/, '$1,$1,$1')
       // 'b,b,b'

      Typescript 的类型也同样可以做模式匹配。

      比如这样一个 Promise 类型:

      ts
      type p = Promise<'value'>

      我们想提取 value 的类型,可以这样做:

      ts
      type GetValueType<P> = P extends Promise<infer Value> ? Value : never

      通过 extends 对传入的类型参数 P 做模式匹配,其中值的类型是需要提取的,通过 infer 声明一个局部变量 Value 来保存,如果匹配,就返回匹配到的 Value,否则就返回 never 代表没匹配到。

      ts
      // type GetValueResult = 'value'
       type GetValueResult = GetValueType<Promise<'value'>>

      这就是 Typescript 类型的模式匹配:

      Typescript 类型的模式匹配是通过 extends 对类型参数做匹配,结果保存到通过 infer 声明的局部类型变量里,如果匹配就能从该局部变量里拿到提取出的类型。

      这个模式匹配的套路有多有用呢?我们来看下在数组、字符串、函数、构造器等类型里的应用。

      1.数组类型

      提取第一个元素

      数组类型想提取第一个元素的类型怎么做呢?

      ts
      type arr = [1, 2, 3]

      用它来匹配一个模式类型,提取第一个元素的类型到通过 infer 声明的局部变量里返回。

      ts
      type GetFirst<Arr extends unknown[]> = Arr extends [infer First, ...unknown[]]
         ? First
      diff --git a/assets/src_article_typescript_pattern.md.e19bc8e4.lean.js b/assets/src_article_typescript_pattern.md.ed4c198d.lean.js
      similarity index 85%
      rename from assets/src_article_typescript_pattern.md.e19bc8e4.lean.js
      rename to assets/src_article_typescript_pattern.md.ed4c198d.lean.js
      index 5e7389022..96638dcf4 100644
      --- a/assets/src_article_typescript_pattern.md.e19bc8e4.lean.js
      +++ b/assets/src_article_typescript_pattern.md.ed4c198d.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"模式匹配提取","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/pattern.md","lastUpdated":1694276611000}'),p={name:"src/article/typescript/pattern.md"},o=l("",145),e=[o];function t(r,c,y,B,F,D){return a(),n("div",null,e)}const A=s(p,[["render",t]]);export{E as __pageData,A as default};
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"模式匹配提取","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/pattern.md","lastUpdated":1694419899000}'),p={name:"src/article/typescript/pattern.md"},o=l("",145),e=[o];function t(r,c,y,B,F,D){return a(),n("div",null,e)}const A=s(p,[["render",t]]);export{E as __pageData,A as default};
      diff --git a/assets/src_article_typescript_reconstruction.md.a186fc17.js b/assets/src_article_typescript_reconstruction.md.21d449a5.js
      similarity index 99%
      rename from assets/src_article_typescript_reconstruction.md.a186fc17.js
      rename to assets/src_article_typescript_reconstruction.md.21d449a5.js
      index afae3e37f..36a624a47 100644
      --- a/assets/src_article_typescript_reconstruction.md.a186fc17.js
      +++ b/assets/src_article_typescript_reconstruction.md.21d449a5.js
      @@ -1,4 +1,4 @@
      -import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"重新构造做变换","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/reconstruction.md","lastUpdated":1694276611000}'),l={name:"src/article/typescript/reconstruction.md"},o=p(`

      重新构造做变换

      类型编程主要的目的就是对类型做各种转换,那么如何对类型做修改呢?

      TypeScript 类型系统支持 3 种可以声明任意类型的变量: type、infer、类型参数。

      type 叫做类型别名,其实就是声明一个变量存储某个类型:

      ts
      type ttt = Promise<number>

      infer 用于类型的提取,然后存到一个变量里,相当于局部变量:

      ts
      type GetValueType<P> = P extends Promise<infer Value> ? Value : never

      类型参数用于接受具体的类型,在类型运算中也相当于局部变量:

      ts
      type isTwo<T> = T extends 2 ? true : false

      但是,严格来说这三种也都不叫变量,因为它们不能被重新赋值。

      TypeScript 设计可以做类型编程的类型系统的目的就是为了产生各种复杂的类型,那不能修改怎么产生新类型呢?

      答案是重新构造。

      这就涉及到了第二个类型体操套路:重新构造做变换。

      重新构造

      TypeScript 的 type、infer、类型参数声明的变量都不能修改,想对类型做各种变换产生新的类型就需要重新构造。

      数组、字符串、函数等类型的重新构造比较简单。

      索引类型,也就是多个元素的聚合类型的重新构造复杂一些,涉及到了映射类型的语法。

      我们先从简单的开始:

      数组类型的重新构造

      Push

      有这样一个元组类型:

      ts
      type tuple = [1, 2, 3]

      我想给这个元组类型再添加一些类型,怎么做呢?

      TypeScript 类型变量不支持修改,我们可以构造一个新的元组类型:

      ts
      type Push<Arr extends unknown[], Ele> = [...Arr, Ele]

      类型参数 Arr 是要修改的数组/元组类型,元素的类型任意,也就是 unknown。

      类型参数 Ele 是添加的元素的类型。

      返回的是用 Arr 已有的元素加上 Ele 构造的新的元组类型。

      ts
      type PushResult = Push<[1, 2, 3], 4>
      +import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"重新构造做变换","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/reconstruction.md","lastUpdated":1694419899000}'),l={name:"src/article/typescript/reconstruction.md"},o=p(`

      重新构造做变换

      类型编程主要的目的就是对类型做各种转换,那么如何对类型做修改呢?

      TypeScript 类型系统支持 3 种可以声明任意类型的变量: type、infer、类型参数。

      type 叫做类型别名,其实就是声明一个变量存储某个类型:

      ts
      type ttt = Promise<number>

      infer 用于类型的提取,然后存到一个变量里,相当于局部变量:

      ts
      type GetValueType<P> = P extends Promise<infer Value> ? Value : never

      类型参数用于接受具体的类型,在类型运算中也相当于局部变量:

      ts
      type isTwo<T> = T extends 2 ? true : false

      但是,严格来说这三种也都不叫变量,因为它们不能被重新赋值。

      TypeScript 设计可以做类型编程的类型系统的目的就是为了产生各种复杂的类型,那不能修改怎么产生新类型呢?

      答案是重新构造。

      这就涉及到了第二个类型体操套路:重新构造做变换。

      重新构造

      TypeScript 的 type、infer、类型参数声明的变量都不能修改,想对类型做各种变换产生新的类型就需要重新构造。

      数组、字符串、函数等类型的重新构造比较简单。

      索引类型,也就是多个元素的聚合类型的重新构造复杂一些,涉及到了映射类型的语法。

      我们先从简单的开始:

      数组类型的重新构造

      Push

      有这样一个元组类型:

      ts
      type tuple = [1, 2, 3]

      我想给这个元组类型再添加一些类型,怎么做呢?

      TypeScript 类型变量不支持修改,我们可以构造一个新的元组类型:

      ts
      type Push<Arr extends unknown[], Ele> = [...Arr, Ele]

      类型参数 Arr 是要修改的数组/元组类型,元素的类型任意,也就是 unknown。

      类型参数 Ele 是添加的元素的类型。

      返回的是用 Arr 已有的元素加上 Ele 构造的新的元组类型。

      ts
      type PushResult = Push<[1, 2, 3], 4>
       // type PushResult = [1,2,3,4]

      这就是数组/元组的重新构造。

      数组和元组的区别:数组类型是指任意多个同一类型的元素构成的,比如 number[]Array<number>,而元组则是数量固定,类型可以不同的元素构成的,比如 [1, true, 'name']

      Unshift

      可以在后面添加,同样也可以在前面添加:

      ts
      type Unshift<Arr extends unknown[], Ele> = [Ele, ...Arr]

      Zip

      有这样两个元组:

      ts
      type tuple1 = [1, 2]
       type tuple2 = ['name', 'value']

      我们想把它们合并成这样的元组:

      ts
      type tuple = [[1, 'name'], [2, 'value']]

      思路很容易想到,提取元组中的两个元素,构造成新的元组:

      ts
      type Zip<
         One extends [unknown, unknown],
      diff --git a/assets/src_article_typescript_reconstruction.md.a186fc17.lean.js b/assets/src_article_typescript_reconstruction.md.21d449a5.lean.js
      similarity index 85%
      rename from assets/src_article_typescript_reconstruction.md.a186fc17.lean.js
      rename to assets/src_article_typescript_reconstruction.md.21d449a5.lean.js
      index 657adfb86..004374e39 100644
      --- a/assets/src_article_typescript_reconstruction.md.a186fc17.lean.js
      +++ b/assets/src_article_typescript_reconstruction.md.21d449a5.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"重新构造做变换","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/reconstruction.md","lastUpdated":1694276611000}'),l={name:"src/article/typescript/reconstruction.md"},o=p("",144),e=[o];function t(r,c,B,y,F,D){return a(),n("div",null,e)}const E=s(l,[["render",t]]);export{A as __pageData,E as default};
      +import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"重新构造做变换","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/reconstruction.md","lastUpdated":1694419899000}'),l={name:"src/article/typescript/reconstruction.md"},o=p("",144),e=[o];function t(r,c,B,y,F,D){return a(),n("div",null,e)}const E=s(l,[["render",t]]);export{A as __pageData,E as default};
      diff --git a/assets/src_article_typescript_recursion.md.9db97e45.js b/assets/src_article_typescript_recursion.md.061a1d67.js
      similarity index 99%
      rename from assets/src_article_typescript_recursion.md.9db97e45.js
      rename to assets/src_article_typescript_recursion.md.061a1d67.js
      index 4c1996645..4e5e472c1 100644
      --- a/assets/src_article_typescript_recursion.md.9db97e45.js
      +++ b/assets/src_article_typescript_recursion.md.061a1d67.js
      @@ -1,4 +1,4 @@
      -import{_ as s,o as n,c as a,N as p}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"递归复用","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/recursion.md","lastUpdated":1694276611000}'),l={name:"src/article/typescript/recursion.md"},o=p(`

      递归复用

      递归是把问题分解为一系列相似的小问题,通过函数不断调用自身来解决这一个个小问题,直到满足结束条件,就完成了问题的求解。

      TypeScript 的高级类型支持类型参数,可以做各种类型运算逻辑,返回新的类型,和函数调用是对应的,自然也支持递归。

      TypeScript 类型系统不支持循环,但支持递归。当处理数量(个数、长度、层数)不固定的类型的时候,可以只处理一个类型,然后递归的调用自身处理下一个类型,直到结束条件也就是所有的类型都处理完了,就完成了不确定数量的类型编程,达到循环的效果。

      既然提到了数组、字符串、对象等类型,那么我们就来看一下这些类型的递归案例吧。

      Promise 的递归复用

      DeepPromiseValueType

      先用 Promise 热热身,实现一个提取不确定层数的 Promise 中的 value 类型的高级类型。

      ts
      type ttt = Promise<Promise<Promise<Record<string, any>>>>

      这里是 3 层 Promise,value 类型是索引类型。

      数量不确定,一涉及到这个就要想到用递归来做,每次只处理一层的提取,然后剩下的到下次递归做,直到结束条件。

      所以高级类型是这样的:

      ts
      type DeepPromiseValueType<P extends Promise<unknown>> = P extends Promise<
      +import{_ as s,o as n,c as a,N as p}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"递归复用","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/recursion.md","lastUpdated":1694419899000}'),l={name:"src/article/typescript/recursion.md"},o=p(`

      递归复用

      递归是把问题分解为一系列相似的小问题,通过函数不断调用自身来解决这一个个小问题,直到满足结束条件,就完成了问题的求解。

      TypeScript 的高级类型支持类型参数,可以做各种类型运算逻辑,返回新的类型,和函数调用是对应的,自然也支持递归。

      TypeScript 类型系统不支持循环,但支持递归。当处理数量(个数、长度、层数)不固定的类型的时候,可以只处理一个类型,然后递归的调用自身处理下一个类型,直到结束条件也就是所有的类型都处理完了,就完成了不确定数量的类型编程,达到循环的效果。

      既然提到了数组、字符串、对象等类型,那么我们就来看一下这些类型的递归案例吧。

      Promise 的递归复用

      DeepPromiseValueType

      先用 Promise 热热身,实现一个提取不确定层数的 Promise 中的 value 类型的高级类型。

      ts
      type ttt = Promise<Promise<Promise<Record<string, any>>>>

      这里是 3 层 Promise,value 类型是索引类型。

      数量不确定,一涉及到这个就要想到用递归来做,每次只处理一层的提取,然后剩下的到下次递归做,直到结束条件。

      所以高级类型是这样的:

      ts
      type DeepPromiseValueType<P extends Promise<unknown>> = P extends Promise<
         infer ValueType
       >
         ? ValueType extends Promise<unknown>
      diff --git a/assets/src_article_typescript_recursion.md.9db97e45.lean.js b/assets/src_article_typescript_recursion.md.061a1d67.lean.js
      similarity index 85%
      rename from assets/src_article_typescript_recursion.md.9db97e45.lean.js
      rename to assets/src_article_typescript_recursion.md.061a1d67.lean.js
      index aa5d43310..99f80af54 100644
      --- a/assets/src_article_typescript_recursion.md.9db97e45.lean.js
      +++ b/assets/src_article_typescript_recursion.md.061a1d67.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as n,c as a,N as p}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"递归复用","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/recursion.md","lastUpdated":1694276611000}'),l={name:"src/article/typescript/recursion.md"},o=p("",123),e=[o];function t(r,c,B,y,F,D){return n(),a("div",null,e)}const i=s(l,[["render",t]]);export{E as __pageData,i as default};
      +import{_ as s,o as n,c as a,N as p}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"递归复用","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/recursion.md","lastUpdated":1694419899000}'),l={name:"src/article/typescript/recursion.md"},o=p("",123),e=[o];function t(r,c,B,y,F,D){return n(),a("div",null,e)}const i=s(l,[["render",t]]);export{E as __pageData,i as default};
      diff --git a/assets/src_article_typescript_unionType.md.0ba9cf91.js b/assets/src_article_typescript_unionType.md.2b84a746.js
      similarity index 99%
      rename from assets/src_article_typescript_unionType.md.0ba9cf91.js
      rename to assets/src_article_typescript_unionType.md.2b84a746.js
      index b3291c5f2..4944580ab 100644
      --- a/assets/src_article_typescript_unionType.md.0ba9cf91.js
      +++ b/assets/src_article_typescript_unionType.md.2b84a746.js
      @@ -1 +1 @@
      -import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const i=JSON.parse('{"title":"分布式条件类型","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/unionType.md","lastUpdated":1694276611000}'),l={name:"src/article/typescript/unionType.md"},o=p('

      分布式条件类型

      当类型参数为联合类型,并且在条件类型左边直接引用该类型参数的时候,TypeScript 会把每一个元素单独传入来做类型运算,最后再合并成联合类型,这种语法叫做分布式条件类型。

      比如这样一个联合类型:

      ts
      type Union = 'a' | 'b' | 'c'

      我们想把其中的 a 大写,就可以这样写:

      ts
      type UppercaseA<Item extends string> = Item extends 'a' ? Uppercase<Item> : Item
      ts
      type result = UppercaseA<Union>\n// type result = 'A' | 'b' | 'c';

      可以看到,我们类型参数 Item 约束为 string,条件类型的判断中也是判断是否是 a,但传入的是联合类型。

      这就是 TypeScript 对联合类型在条件类型中使用时的特殊处理:会把联合类型的每一个元素单独传入做类型计算,最后合并。

      这和联合类型遇到字符串时的处理一样:

      这样确实是简化了类型编程逻辑的,不需要递归提取每个元素再处理。

      TypeScript 之所以这样处理联合类型也很容易理解,因为联合类型的每个元素都是互不相关的,不像数组、索引、字符串那样元素之间是有关系的。所以设计成了每一个单独处理,最后合并。

      知道了 TypeScript 怎么处理的联合类型,趁热打铁来练习一下:

      CamelcaseUnion

      Camelcase 我们实现过,就是提取字符串中的字符,首字母大写以后重新构造一个新的。

      ts
      type Camelcase<Str extends string> =\n  Str extends `${infer Left}_${infer Right}${infer Rest}`\n    ? `${Left}${Uppercase<Right>}${Camelcase<Rest>}`\n    : Str

      提取 _ 左右的字符,把右边字符大写之后构造成新的字符串,余下的字符串递归处理。

      ts
      type CamelcaseResult = Camelcase<'aa_aa_aa'>\n// type CamelcaseResult = 'aaAaAa'

      如果是对字符串数组做 Camelcase,那就要递归处理每一个元素:

      ts
      type CamelcaseArr<Arr extends unknown[]> = Arr extends [\n  infer Item,\n  ...infer RestArr,\n]\n  ? [Camelcase<Item & string>, ...CamelcaseArr<RestArr>]\n  : []

      类型参数 Arr 为待处理数组。

      递归提取每一个元素做 Camelcase,因为 Camelcase 要求传入 string,这里要 & string 来变成 string 类型。

      那如果是联合类型呢?

      联合类型不需要递归提取每个元素,TypeScript 内部会把每一个元素传入单独做计算,之后把每个元素的计算结果合并成联合类型。

      ts
      type CamelcaseUnion<Item extends string> =\n  Item extends `${infer Left}_${infer Right}${infer Rest}`\n    ? `${Left}${Uppercase<Right>}${CamelcaseUnion<Rest>}`\n    : Item

      这不和单个字符串的处理没区别么?

      没错,对联合类型的处理和对单个类型的处理没什么区别,TypeScript 会把每个单独的类型拆开传入。不需要像数组类型那样需要递归提取每个元素做处理。

      确实简化了很多,好像都是优点?

      也不全是,其实这样处理也增加了一些认知成本,不信我们再来看个例子:

      IsUnion

      判断联合类型我们会这样写:

      ts
      type IsUnion<A, B = A> = A extends A ? ([B] extends [A] ? false : true) : never

      当传入联合类型时,会返回 true:

      ts
      type IsUnionResult = IsUnion<'a' | 'b' 'c'>\n// type IsUnionResult = true

      当传入其他类型时,会返回 false:

      ts
      type IsUnionResult = IsUnion<['a' | 'b' 'c']>\n// type IsUnionResult = false

      这就是分布式条件类型带来的认知成本。

      我们先来看这样一个类型:

      ts
      type TestUnion<A, B = A> = A extends A ? { a: A; b: B } : never\n\ntype TestUnionResult = TestUnion<'a' | 'b' | 'c'>

      传入联合类型 'a' | 'b' | 'c' 的时候,结果是这样的:

      A 和 B 都是同一个联合类型,为啥值还不一样呢?

      因为条件类型中如果左边的类型是联合类型,会把每个元素单独传入做计算,而右边不会。

      所以 A 是 'a' 的时候,B 是 'a' | 'b' | 'c', A 是 'b' 的时候,B 是 'a' | 'b' | 'c'。。。

      那么利用这个特点就可以实现 Union 类型的判断:

      ts
      type IsUnion<A, B = A> = A extends A ? ([B] extends [A] ? false : true) : never

      类型参数 A、B 是待判断的联合类型,B 默认值为 A,也就是同一个类型。

      A extends A 这段看似没啥意义,主要是为了触发分布式条件类型,让 A 的每个类型单独传入。

      [B] extends [A] 这样不直接写 B 就可以避免触发分布式条件类型,那么 B 就是整个联合类型。

      B 是联合类型整体,而 A 是单个类型,自然不成立,而其它类型没有这种特殊处理,A 和 B 都是同一个,怎么判断都成立。

      利用这个特点就可以判断出是否是联合类型。

      其中有两个点比较困惑,我们重点记一下:

      当 A 是联合类型时:

      A extends A 这种写法是为了触发分布式条件类型,让每个类型单独传入处理的,没别的意义。

      A extends A 和 [A] extends [A] 是不同的处理,前者是单个类型和整个类型做判断,后者两边都是整个联合类型,因为只有 extends 左边直接是类型参数才会触发分布式条件类型。

      理解了这两点,分布式条件类型就算掌握了。

      BEM

      bem 是 css 命名规范,用 block__element--modifier 的形式来描述某个区块下面的某个元素的某个状态的样式。

      那么我们可以写这样一个高级类型,传入 block、element、modifier,返回构造出的 class 名:

      这样使用:

      ts
      type bemResult = BEM<'guang', ['aaa', 'bbb'], ['warning', 'success']>

      它的实现就是三部分的合并,但传入的是数组,要递归遍历取出每一个元素来和其他部分组合,这样太麻烦了。

      而如果是联合类型就不用递归遍历了,因为联合类型遇到字符串也是会单独每个元素单独传入做处理。

      数组转联合类型可以这样写:

      ts
      type union = ['aaa', 'bbb'][number]\n// type union = 'aaa' | 'bbb'

      那么 BEM 就可以这样实现:

      ts
      type BEM<\n  Block extends string,\n  Element extends string[],\n  Modifiers extends string[],\n> = `${Block}__${Element[number]}--${Modifiers[number]}`

      类型参数 Block、Element、Modifiers 分别是 bem 规范的三部分,其中 Element 和 Modifiers 都可能多个,约束为 string[]。

      构造一个字符串类型,其中 Element 和 Modifiers 通过索引访问来变为联合类型。

      字符串类型中遇到联合类型的时候,会每个元素单独传入计算,也就是这样的效果:

      ts
      type RemResult = BEM<'a', ['b', 'c'], ['d', 'e']>\n// type RemResult = 'a__b--d' | 'a__b--e' | 'a__c--d' | 'a__b--e'

      可以看到,用好了联合类型,确实能简化类型编程逻辑。

      AllCombinations

      我们再来实现一个全组合的高级类型,也是联合类型相关的:

      希望传入 'A' | 'B' 的时候,能够返回所有的组合: 'A' | 'B' | 'BA' | 'AB'。

      这种全组合问题的实现思路就是两两组合,组合出的字符串再和其他字符串两两组和:

      比如 'A' | 'B' | 'c',就是 A 和 B、C 组合,B 和 A、C 组合,C 和 A、B 组合。然后组合出来的字符串再和其他字符串组合。

      任何两个类型的组合有四种:A、B、AB、BA

      ts
      type Combination<A extends string, B extends string> =\n  | A\n  | B\n  | `${A}${B}`\n  | `${B}${A}`

      然后构造出来的字符串再和其他字符串组合。

      所以全组合的高级类型就是这样:

      ts
      type AllCombinations<A extends string, B extends string = A> = A extends A\n  ? Combination<A, AllCombinations<Exclude<B, A>>>\n  : never

      类型参数 A、B 是待组合的两个联合类型,B 默认是 A 也就是同一个。

      A extends A 的意义就是让联合类型每个类型单独传入做处理,上面我们刚学会。

      A 的处理就是 A 和 B 中去掉 A 以后的所有类型组合,也就是 Combination<A, B 去掉 A 以后的所有组合>。

      而 B 去掉 A 以后的所有组合就是 AllCombinations<Exclude<B, A>>,所以全组合就是 Combination<A, AllCombinations<Exclude<B, A>>>。

      总结

      联合类型中的每个类型都是相互独立的,TypeScript 对它做了特殊处理,也就是遇到字符串类型、条件类型的时候会把每个类型单独传入做计算,最后把每个类型的计算结果合并成联合类型。

      条件类型左边是联合类型的时候就会触法这种处理,叫做分布式条件类型。

      有两点特别要注意:

      • A extends A 不是没意义,意义是取出联合类型中的单个类型放入 A

      • A extends A 才是分布式条件类型, [A] extends [A] 就不是了,只有左边是单独的类型参数才可以。

      我们后面做了一些案例,发现联合类型的这种 distributive 的特性确实能简化类型编程,但是也增加了认知成本,不过这也是不可避免的事。

      ',91),e=[o];function t(c,r,B,y,F,D){return a(),n("div",null,e)}const E=s(l,[["render",t]]);export{i as __pageData,E as default}; +import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const i=JSON.parse('{"title":"分布式条件类型","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/unionType.md","lastUpdated":1694419899000}'),l={name:"src/article/typescript/unionType.md"},o=p('

      分布式条件类型

      当类型参数为联合类型,并且在条件类型左边直接引用该类型参数的时候,TypeScript 会把每一个元素单独传入来做类型运算,最后再合并成联合类型,这种语法叫做分布式条件类型。

      比如这样一个联合类型:

      ts
      type Union = 'a' | 'b' | 'c'

      我们想把其中的 a 大写,就可以这样写:

      ts
      type UppercaseA<Item extends string> = Item extends 'a' ? Uppercase<Item> : Item
      ts
      type result = UppercaseA<Union>\n// type result = 'A' | 'b' | 'c';

      可以看到,我们类型参数 Item 约束为 string,条件类型的判断中也是判断是否是 a,但传入的是联合类型。

      这就是 TypeScript 对联合类型在条件类型中使用时的特殊处理:会把联合类型的每一个元素单独传入做类型计算,最后合并。

      这和联合类型遇到字符串时的处理一样:

      这样确实是简化了类型编程逻辑的,不需要递归提取每个元素再处理。

      TypeScript 之所以这样处理联合类型也很容易理解,因为联合类型的每个元素都是互不相关的,不像数组、索引、字符串那样元素之间是有关系的。所以设计成了每一个单独处理,最后合并。

      知道了 TypeScript 怎么处理的联合类型,趁热打铁来练习一下:

      CamelcaseUnion

      Camelcase 我们实现过,就是提取字符串中的字符,首字母大写以后重新构造一个新的。

      ts
      type Camelcase<Str extends string> =\n  Str extends `${infer Left}_${infer Right}${infer Rest}`\n    ? `${Left}${Uppercase<Right>}${Camelcase<Rest>}`\n    : Str

      提取 _ 左右的字符,把右边字符大写之后构造成新的字符串,余下的字符串递归处理。

      ts
      type CamelcaseResult = Camelcase<'aa_aa_aa'>\n// type CamelcaseResult = 'aaAaAa'

      如果是对字符串数组做 Camelcase,那就要递归处理每一个元素:

      ts
      type CamelcaseArr<Arr extends unknown[]> = Arr extends [\n  infer Item,\n  ...infer RestArr,\n]\n  ? [Camelcase<Item & string>, ...CamelcaseArr<RestArr>]\n  : []

      类型参数 Arr 为待处理数组。

      递归提取每一个元素做 Camelcase,因为 Camelcase 要求传入 string,这里要 & string 来变成 string 类型。

      那如果是联合类型呢?

      联合类型不需要递归提取每个元素,TypeScript 内部会把每一个元素传入单独做计算,之后把每个元素的计算结果合并成联合类型。

      ts
      type CamelcaseUnion<Item extends string> =\n  Item extends `${infer Left}_${infer Right}${infer Rest}`\n    ? `${Left}${Uppercase<Right>}${CamelcaseUnion<Rest>}`\n    : Item

      这不和单个字符串的处理没区别么?

      没错,对联合类型的处理和对单个类型的处理没什么区别,TypeScript 会把每个单独的类型拆开传入。不需要像数组类型那样需要递归提取每个元素做处理。

      确实简化了很多,好像都是优点?

      也不全是,其实这样处理也增加了一些认知成本,不信我们再来看个例子:

      IsUnion

      判断联合类型我们会这样写:

      ts
      type IsUnion<A, B = A> = A extends A ? ([B] extends [A] ? false : true) : never

      当传入联合类型时,会返回 true:

      ts
      type IsUnionResult = IsUnion<'a' | 'b' 'c'>\n// type IsUnionResult = true

      当传入其他类型时,会返回 false:

      ts
      type IsUnionResult = IsUnion<['a' | 'b' 'c']>\n// type IsUnionResult = false

      这就是分布式条件类型带来的认知成本。

      我们先来看这样一个类型:

      ts
      type TestUnion<A, B = A> = A extends A ? { a: A; b: B } : never\n\ntype TestUnionResult = TestUnion<'a' | 'b' | 'c'>

      传入联合类型 'a' | 'b' | 'c' 的时候,结果是这样的:

      A 和 B 都是同一个联合类型,为啥值还不一样呢?

      因为条件类型中如果左边的类型是联合类型,会把每个元素单独传入做计算,而右边不会。

      所以 A 是 'a' 的时候,B 是 'a' | 'b' | 'c', A 是 'b' 的时候,B 是 'a' | 'b' | 'c'。。。

      那么利用这个特点就可以实现 Union 类型的判断:

      ts
      type IsUnion<A, B = A> = A extends A ? ([B] extends [A] ? false : true) : never

      类型参数 A、B 是待判断的联合类型,B 默认值为 A,也就是同一个类型。

      A extends A 这段看似没啥意义,主要是为了触发分布式条件类型,让 A 的每个类型单独传入。

      [B] extends [A] 这样不直接写 B 就可以避免触发分布式条件类型,那么 B 就是整个联合类型。

      B 是联合类型整体,而 A 是单个类型,自然不成立,而其它类型没有这种特殊处理,A 和 B 都是同一个,怎么判断都成立。

      利用这个特点就可以判断出是否是联合类型。

      其中有两个点比较困惑,我们重点记一下:

      当 A 是联合类型时:

      A extends A 这种写法是为了触发分布式条件类型,让每个类型单独传入处理的,没别的意义。

      A extends A 和 [A] extends [A] 是不同的处理,前者是单个类型和整个类型做判断,后者两边都是整个联合类型,因为只有 extends 左边直接是类型参数才会触发分布式条件类型。

      理解了这两点,分布式条件类型就算掌握了。

      BEM

      bem 是 css 命名规范,用 block__element--modifier 的形式来描述某个区块下面的某个元素的某个状态的样式。

      那么我们可以写这样一个高级类型,传入 block、element、modifier,返回构造出的 class 名:

      这样使用:

      ts
      type bemResult = BEM<'guang', ['aaa', 'bbb'], ['warning', 'success']>

      它的实现就是三部分的合并,但传入的是数组,要递归遍历取出每一个元素来和其他部分组合,这样太麻烦了。

      而如果是联合类型就不用递归遍历了,因为联合类型遇到字符串也是会单独每个元素单独传入做处理。

      数组转联合类型可以这样写:

      ts
      type union = ['aaa', 'bbb'][number]\n// type union = 'aaa' | 'bbb'

      那么 BEM 就可以这样实现:

      ts
      type BEM<\n  Block extends string,\n  Element extends string[],\n  Modifiers extends string[],\n> = `${Block}__${Element[number]}--${Modifiers[number]}`

      类型参数 Block、Element、Modifiers 分别是 bem 规范的三部分,其中 Element 和 Modifiers 都可能多个,约束为 string[]。

      构造一个字符串类型,其中 Element 和 Modifiers 通过索引访问来变为联合类型。

      字符串类型中遇到联合类型的时候,会每个元素单独传入计算,也就是这样的效果:

      ts
      type RemResult = BEM<'a', ['b', 'c'], ['d', 'e']>\n// type RemResult = 'a__b--d' | 'a__b--e' | 'a__c--d' | 'a__b--e'

      可以看到,用好了联合类型,确实能简化类型编程逻辑。

      AllCombinations

      我们再来实现一个全组合的高级类型,也是联合类型相关的:

      希望传入 'A' | 'B' 的时候,能够返回所有的组合: 'A' | 'B' | 'BA' | 'AB'。

      这种全组合问题的实现思路就是两两组合,组合出的字符串再和其他字符串两两组和:

      比如 'A' | 'B' | 'c',就是 A 和 B、C 组合,B 和 A、C 组合,C 和 A、B 组合。然后组合出来的字符串再和其他字符串组合。

      任何两个类型的组合有四种:A、B、AB、BA

      ts
      type Combination<A extends string, B extends string> =\n  | A\n  | B\n  | `${A}${B}`\n  | `${B}${A}`

      然后构造出来的字符串再和其他字符串组合。

      所以全组合的高级类型就是这样:

      ts
      type AllCombinations<A extends string, B extends string = A> = A extends A\n  ? Combination<A, AllCombinations<Exclude<B, A>>>\n  : never

      类型参数 A、B 是待组合的两个联合类型,B 默认是 A 也就是同一个。

      A extends A 的意义就是让联合类型每个类型单独传入做处理,上面我们刚学会。

      A 的处理就是 A 和 B 中去掉 A 以后的所有类型组合,也就是 Combination<A, B 去掉 A 以后的所有组合>。

      而 B 去掉 A 以后的所有组合就是 AllCombinations<Exclude<B, A>>,所以全组合就是 Combination<A, AllCombinations<Exclude<B, A>>>。

      总结

      联合类型中的每个类型都是相互独立的,TypeScript 对它做了特殊处理,也就是遇到字符串类型、条件类型的时候会把每个类型单独传入做计算,最后把每个类型的计算结果合并成联合类型。

      条件类型左边是联合类型的时候就会触法这种处理,叫做分布式条件类型。

      有两点特别要注意:

      • A extends A 不是没意义,意义是取出联合类型中的单个类型放入 A

      • A extends A 才是分布式条件类型, [A] extends [A] 就不是了,只有左边是单独的类型参数才可以。

      我们后面做了一些案例,发现联合类型的这种 distributive 的特性确实能简化类型编程,但是也增加了认知成本,不过这也是不可避免的事。

      ',91),e=[o];function t(c,r,B,y,F,D){return a(),n("div",null,e)}const E=s(l,[["render",t]]);export{i as __pageData,E as default}; diff --git a/assets/src_article_typescript_unionType.md.0ba9cf91.lean.js b/assets/src_article_typescript_unionType.md.2b84a746.lean.js similarity index 85% rename from assets/src_article_typescript_unionType.md.0ba9cf91.lean.js rename to assets/src_article_typescript_unionType.md.2b84a746.lean.js index b1b5b6818..5b051a3a0 100644 --- a/assets/src_article_typescript_unionType.md.0ba9cf91.lean.js +++ b/assets/src_article_typescript_unionType.md.2b84a746.lean.js @@ -1 +1 @@ -import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const i=JSON.parse('{"title":"分布式条件类型","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/unionType.md","lastUpdated":1694276611000}'),l={name:"src/article/typescript/unionType.md"},o=p("",91),e=[o];function t(c,r,B,y,F,D){return a(),n("div",null,e)}const E=s(l,[["render",t]]);export{i as __pageData,E as default}; +import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const i=JSON.parse('{"title":"分布式条件类型","description":"","frontmatter":{},"headers":[],"relativePath":"src/article/typescript/unionType.md","lastUpdated":1694419899000}'),l={name:"src/article/typescript/unionType.md"},o=p("",91),e=[o];function t(c,r,B,y,F,D){return a(),n("div",null,e)}const E=s(l,[["render",t]]);export{i as __pageData,E as default}; diff --git a/assets/src_ranui_button_index.md.32a9d03f.js b/assets/src_ranui_button_index.md.eb788bcc.js similarity index 99% rename from assets/src_ranui_button_index.md.32a9d03f.js rename to assets/src_ranui_button_index.md.eb788bcc.js index cf6ba87e8..c6ac41944 100644 --- a/assets/src_ranui_button_index.md.32a9d03f.js +++ b/assets/src_ranui_button_index.md.eb788bcc.js @@ -1,4 +1,4 @@ -import{_ as n,o,c as t,N as s,x as a}from"./chunks/framework.6fe2e870.js";const h=JSON.parse('{"title":"Button 按钮","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/button/index.md","lastUpdated":1694276611000}'),l={name:"src/ranui/button/index.md"},p=s(`

      Button 按钮

      按钮用于开始一个即时操作。

      代码演示

      Button
      xml
      <r-button >Button</r-button>

      属性

      类型type

      按钮有四种类型

      主要按钮
      警告按钮
      文本按钮
      默认按钮
      xml
      <r-button type="primary">主要按钮</r-button>
      +import{_ as n,o,c as t,N as s,x as a}from"./chunks/framework.6fe2e870.js";const h=JSON.parse('{"title":"Button 按钮","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/button/index.md","lastUpdated":1694419899000}'),l={name:"src/ranui/button/index.md"},p=s(`

      Button 按钮

      按钮用于开始一个即时操作。

      代码演示

      Button
      xml
      <r-button >Button</r-button>

      属性

      类型type

      按钮有四种类型

      主要按钮
      警告按钮
      文本按钮
      默认按钮
      xml
      <r-button type="primary">主要按钮</r-button>
        <r-button type="warning">警告按钮</r-button>
        <r-button type="text">文本按钮</r-button>
        <r-button >默认按钮</r-button>

      不可用状态disabled

      添加 disabled 属性即可让按钮处于不可用状态,同时按钮样式也会改变。

      主要按钮
      警告按钮
      文本按钮
      默认按钮
      xml
      <r-button type="primary" disabled>主要按钮</r-button>
      diff --git a/assets/src_ranui_button_index.md.32a9d03f.lean.js b/assets/src_ranui_button_index.md.eb788bcc.lean.js
      similarity index 88%
      rename from assets/src_ranui_button_index.md.32a9d03f.lean.js
      rename to assets/src_ranui_button_index.md.eb788bcc.lean.js
      index 751350cad..6c1438d71 100644
      --- a/assets/src_ranui_button_index.md.32a9d03f.lean.js
      +++ b/assets/src_ranui_button_index.md.eb788bcc.lean.js
      @@ -1 +1 @@
      -import{_ as n,o,c as t,N as s,x as a}from"./chunks/framework.6fe2e870.js";const h=JSON.parse('{"title":"Button 按钮","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/button/index.md","lastUpdated":1694276611000}'),l={name:"src/ranui/button/index.md"},p=s("",28),e=a("r-button",{type:"default",effect:"fase",icon:"user"},"默认按钮",-1),r=a("r-button",{type:"primary",effect:"fase",icon:"home"},"主要按钮",-1),c=s("",1),D=[p,e,r,c];function y(F,i,u,d,b,g){return o(),t("div",null,D)}const E=n(l,[["render",y]]);export{h as __pageData,E as default};
      +import{_ as n,o,c as t,N as s,x as a}from"./chunks/framework.6fe2e870.js";const h=JSON.parse('{"title":"Button 按钮","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/button/index.md","lastUpdated":1694419899000}'),l={name:"src/ranui/button/index.md"},p=s("",28),e=a("r-button",{type:"default",effect:"fase",icon:"user"},"默认按钮",-1),r=a("r-button",{type:"primary",effect:"fase",icon:"home"},"主要按钮",-1),c=s("",1),D=[p,e,r,c];function y(F,i,u,d,b,g){return o(),t("div",null,D)}const E=n(l,[["render",y]]);export{h as __pageData,E as default};
      diff --git a/assets/src_ranui_icon_index.md.11f7a3d7.js b/assets/src_ranui_icon_index.md.e1b014ce.js
      similarity index 99%
      rename from assets/src_ranui_icon_index.md.11f7a3d7.js
      rename to assets/src_ranui_icon_index.md.e1b014ce.js
      index f88348e55..47407666c 100644
      --- a/assets/src_ranui_icon_index.md.11f7a3d7.js
      +++ b/assets/src_ranui_icon_index.md.e1b014ce.js
      @@ -1,4 +1,4 @@
      -import{_ as c,o as r,c as F,N as t,x as a}from"./chunks/framework.6fe2e870.js";const D=()=>{setTimeout(()=>{const e=["add-user","book","check-circle","close-circle","eye-close","eye","info-circle","loading","lock","message","power-off","setting","team","unlock","user"];if(typeof document<"u"){const o=document.getElementById("icon-list");e.forEach(l=>{const s=document.createElement("div");s.style.setProperty("display","flex"),s.style.setProperty("align-items","center"),s.style.setProperty("margin","15px"),s.style.setProperty("justify-content","center"),s.style.setProperty("flex-flow","column nowrap");const n=document.createElement("r-icon");n.setAttribute("name",l),n.setAttribute("size","50"),s.appendChild(n);const p=document.createElement("span");p.innerHTML=l,s.appendChild(p),o==null||o.appendChild(s)})}},0)};D();const q=JSON.parse('{"title":"Icon 图标","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/icon/index.md","lastUpdated":1694276611000}'),y={name:"src/ranui/icon/index.md"},i=t(`

      Icon 图标

      语义化的矢量图形

      代码演示

      xml
      <r-icon name="lock"  ></r-icon>
      +import{_ as c,o as r,c as F,N as t,x as a}from"./chunks/framework.6fe2e870.js";const D=()=>{setTimeout(()=>{const e=["add-user","book","check-circle","close-circle","eye-close","eye","info-circle","loading","lock","message","power-off","setting","team","unlock","user"];if(typeof document<"u"){const o=document.getElementById("icon-list");e.forEach(l=>{const s=document.createElement("div");s.style.setProperty("display","flex"),s.style.setProperty("align-items","center"),s.style.setProperty("margin","15px"),s.style.setProperty("justify-content","center"),s.style.setProperty("flex-flow","column nowrap");const n=document.createElement("r-icon");n.setAttribute("name",l),n.setAttribute("size","50"),s.appendChild(n);const p=document.createElement("span");p.innerHTML=l,s.appendChild(p),o==null||o.appendChild(s)})}},0)};D();const q=JSON.parse('{"title":"Icon 图标","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/icon/index.md","lastUpdated":1694419899000}'),y={name:"src/ranui/icon/index.md"},i=t(`

      Icon 图标

      语义化的矢量图形

      代码演示

      xml
      <r-icon name="lock"  ></r-icon>
        <r-icon name="eye"  ></r-icon>
        <r-icon name="user"  ></r-icon>

      属性

      名称name

      根据名称选择不同的图标

      html
      <r-icon name="lock"></r-icon>
       <r-icon name="eye"></r-icon>
      diff --git a/assets/src_ranui_icon_index.md.11f7a3d7.lean.js b/assets/src_ranui_icon_index.md.e1b014ce.lean.js
      similarity index 95%
      rename from assets/src_ranui_icon_index.md.11f7a3d7.lean.js
      rename to assets/src_ranui_icon_index.md.e1b014ce.lean.js
      index 231513e5e..65fbb2941 100644
      --- a/assets/src_ranui_icon_index.md.11f7a3d7.lean.js
      +++ b/assets/src_ranui_icon_index.md.e1b014ce.lean.js
      @@ -1 +1 @@
      -import{_ as c,o as r,c as F,N as t,x as a}from"./chunks/framework.6fe2e870.js";const D=()=>{setTimeout(()=>{const e=["add-user","book","check-circle","close-circle","eye-close","eye","info-circle","loading","lock","message","power-off","setting","team","unlock","user"];if(typeof document<"u"){const o=document.getElementById("icon-list");e.forEach(l=>{const s=document.createElement("div");s.style.setProperty("display","flex"),s.style.setProperty("align-items","center"),s.style.setProperty("margin","15px"),s.style.setProperty("justify-content","center"),s.style.setProperty("flex-flow","column nowrap");const n=document.createElement("r-icon");n.setAttribute("name",l),n.setAttribute("size","50"),s.appendChild(n);const p=document.createElement("span");p.innerHTML=l,s.appendChild(p),o==null||o.appendChild(s)})}},0)};D();const q=JSON.parse('{"title":"Icon 图标","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/icon/index.md","lastUpdated":1694276611000}'),y={name:"src/ranui/icon/index.md"},i=t("",18),u=a("div",{style:{display:"flex"}},[a("r-icon",{name:"loading",size:"50",color:"#1E90FF",spin:"0.7"}),a("r-icon",{name:"loading",size:"50",color:"#1E90FF",spin:""}),a("r-icon",{name:"loading",size:"50",color:"#1E90FF",spin:"5"})],-1),d=t("",3),E=[i,u,d];function m(e,o,l,s,n,p){return r(),F("div",null,E)}const h=c(y,[["render",m]]);export{q as __pageData,h as default};
      +import{_ as c,o as r,c as F,N as t,x as a}from"./chunks/framework.6fe2e870.js";const D=()=>{setTimeout(()=>{const e=["add-user","book","check-circle","close-circle","eye-close","eye","info-circle","loading","lock","message","power-off","setting","team","unlock","user"];if(typeof document<"u"){const o=document.getElementById("icon-list");e.forEach(l=>{const s=document.createElement("div");s.style.setProperty("display","flex"),s.style.setProperty("align-items","center"),s.style.setProperty("margin","15px"),s.style.setProperty("justify-content","center"),s.style.setProperty("flex-flow","column nowrap");const n=document.createElement("r-icon");n.setAttribute("name",l),n.setAttribute("size","50"),s.appendChild(n);const p=document.createElement("span");p.innerHTML=l,s.appendChild(p),o==null||o.appendChild(s)})}},0)};D();const q=JSON.parse('{"title":"Icon 图标","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/icon/index.md","lastUpdated":1694419899000}'),y={name:"src/ranui/icon/index.md"},i=t("",18),u=a("div",{style:{display:"flex"}},[a("r-icon",{name:"loading",size:"50",color:"#1E90FF",spin:"0.7"}),a("r-icon",{name:"loading",size:"50",color:"#1E90FF",spin:""}),a("r-icon",{name:"loading",size:"50",color:"#1E90FF",spin:"5"})],-1),d=t("",3),E=[i,u,d];function m(e,o,l,s,n,p){return r(),F("div",null,E)}const h=c(y,[["render",m]]);export{q as __pageData,h as default};
      diff --git a/assets/src_ranui_image_index.md.08e89593.js b/assets/src_ranui_image_index.md.c0b5dda1.js
      similarity index 98%
      rename from assets/src_ranui_image_index.md.08e89593.js
      rename to assets/src_ranui_image_index.md.c0b5dda1.js
      index 550771f7c..44092f4d2 100644
      --- a/assets/src_ranui_image_index.md.08e89593.js
      +++ b/assets/src_ranui_image_index.md.c0b5dda1.js
      @@ -1 +1 @@
      -import{_ as A,o as a,c as E,N as Q,x as o}from"./chunks/framework.6fe2e870.js";const I=JSON.parse('{"title":"Image 图片","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/image/index.md","lastUpdated":1694276611000}'),e={name:"src/ranui/image/index.md"},s=Q('

      Image 图片

      代码演示

      xml
      <r-img src="" fallback=""></r-img>

      属性

      图片加载地址src

      图片的地址

      图片加载失败fallback

      src配置的图片加载失败,兜底的图片地址,下面是默认加载失败图片

      ',8),n=o("r-img",{fallback:""},null,-1),g=[s,n];function l(t,c,i,r,B,d){return a(),E("div",null,g)}const C=A(e,[["render",l]]);export{I as __pageData,C as default}; +import{_ as A,o as a,c as E,N as Q,x as o}from"./chunks/framework.6fe2e870.js";const I=JSON.parse('{"title":"Image 图片","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/image/index.md","lastUpdated":1694419899000}'),e={name:"src/ranui/image/index.md"},s=Q('

      Image 图片

      代码演示

      xml
      <r-img src="" fallback=""></r-img>

      属性

      图片加载地址src

      图片的地址

      图片加载失败fallback

      src配置的图片加载失败,兜底的图片地址,下面是默认加载失败图片

      ',8),n=o("r-img",{fallback:""},null,-1),g=[s,n];function l(t,c,i,r,B,d){return a(),E("div",null,g)}const C=A(e,[["render",l]]);export{I as __pageData,C as default}; diff --git a/assets/src_ranui_image_index.md.08e89593.lean.js b/assets/src_ranui_image_index.md.c0b5dda1.lean.js similarity index 98% rename from assets/src_ranui_image_index.md.08e89593.lean.js rename to assets/src_ranui_image_index.md.c0b5dda1.lean.js index fc4b2cfbe..f3e3dc722 100644 --- a/assets/src_ranui_image_index.md.08e89593.lean.js +++ b/assets/src_ranui_image_index.md.c0b5dda1.lean.js @@ -1 +1 @@ -import{_ as A,o as a,c as E,N as Q,x as o}from"./chunks/framework.6fe2e870.js";const I=JSON.parse('{"title":"Image 图片","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/image/index.md","lastUpdated":1694276611000}'),e={name:"src/ranui/image/index.md"},s=Q("",8),n=o("r-img",{fallback:""},null,-1),g=[s,n];function l(t,c,i,r,B,d){return a(),E("div",null,g)}const C=A(e,[["render",l]]);export{I as __pageData,C as default}; +import{_ as A,o as a,c as E,N as Q,x as o}from"./chunks/framework.6fe2e870.js";const I=JSON.parse('{"title":"Image 图片","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/image/index.md","lastUpdated":1694419899000}'),e={name:"src/ranui/image/index.md"},s=Q("",8),n=o("r-img",{fallback:""},null,-1),g=[s,n];function l(t,c,i,r,B,d){return a(),E("div",null,g)}const C=A(e,[["render",l]]);export{I as __pageData,C as default}; diff --git a/assets/src_ranui_index.md.0d4c041d.js b/assets/src_ranui_index.md.8ed6a12e.js similarity index 99% rename from assets/src_ranui_index.md.0d4c041d.js rename to assets/src_ranui_index.md.8ed6a12e.js index 7ae81fba1..caee98f58 100644 --- a/assets/src_ranui_index.md.0d4c041d.js +++ b/assets/src_ranui_index.md.8ed6a12e.js @@ -1,4 +1,4 @@ -import{_ as n,o as l,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/customElements.ed27d2e2.png",v=JSON.parse('{"title":"ranui","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/index.md","lastUpdated":1694276611000}'),e={name:"src/ranui/index.md"},t=a(`

      ranui

      基于 Web Components UI组件库

      Feature 特点

      1. 基于Web Components开发,跨框架复用,统一所有情况。
      2. TypeScript开发,有声明和类型文件。
      3. 纯原生手写,基础组件无依赖。
      4. 文档基于vitepress,所有组件实例可交互。
      5. MIT协议。

      Situation 项目情况

      Build Statusnpm-vnpm-dbrotlimodule formats: umd, esm

      • githttps://github.com/chaxus/ran/tree/main/packages/ranui
      • npmhttps://www.npmjs.com/package/ranui

      Usage 使用

      大多数情况都可以像原生的 div 标签一样使用。

      接下来是一些使用例子

      1. html
      2. js
      3. jsx
      4. vue
      5. tsx

      1.html

      html
      <script src="./ranui/dist/umd/index.umd.cjs"></script>
      +import{_ as n,o as l,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/customElements.ed27d2e2.png",v=JSON.parse('{"title":"ranui","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/index.md","lastUpdated":1694419899000}'),e={name:"src/ranui/index.md"},t=a(`

      ranui

      基于 Web Components UI组件库

      Feature 特点

      1. 基于Web Components开发,跨框架复用,统一所有情况。
      2. TypeScript开发,有声明和类型文件。
      3. 纯原生手写,基础组件无依赖。
      4. 文档基于vitepress,所有组件实例可交互。
      5. MIT协议。

      Situation 项目情况

      Build Statusnpm-vnpm-dbrotlimodule formats: umd, esm

      • githttps://github.com/chaxus/ran/tree/main/packages/ranui
      • npmhttps://www.npmjs.com/package/ranui

      Usage 使用

      大多数情况都可以像原生的 div 标签一样使用。

      接下来是一些使用例子

      1. html
      2. js
      3. jsx
      4. vue
      5. tsx

      1.html

      html
      <script src="./ranui/dist/umd/index.umd.cjs"></script>
       
       <body>
         <r-button>Button</r-button>
      diff --git a/assets/src_ranui_index.md.0d4c041d.lean.js b/assets/src_ranui_index.md.8ed6a12e.lean.js
      similarity index 93%
      rename from assets/src_ranui_index.md.0d4c041d.lean.js
      rename to assets/src_ranui_index.md.8ed6a12e.lean.js
      index c7a0a9a65..e75790ed6 100644
      --- a/assets/src_ranui_index.md.0d4c041d.lean.js
      +++ b/assets/src_ranui_index.md.8ed6a12e.lean.js
      @@ -1 +1 @@
      -import{_ as n,o as l,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/customElements.ed27d2e2.png",v=JSON.parse('{"title":"ranui","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/index.md","lastUpdated":1694276611000}'),e={name:"src/ranui/index.md"},t=a("",40),c=s("div",{style:{display:"flex"}},[s("r-icon",{name:"lock",size:"50"}),s("r-icon",{name:"user",size:"50"}),s("r-icon",{name:"loading",size:"50",color:"#1E90FF",spin:""})],-1),r=a("",9),F=s("r-button",{onclick:"message.info('这是一条提示')"},"信息提示",-1),i=s("r-button",{onclick:"message.warning('这是一条提示')"},"警告提示",-1),y=s("r-button",{onclick:"message.error('这是一条提示')"},"错误提示",-1),D=s("r-button",{onclick:"message.success('这是一条提示')"},"成功提示",-1),B=s("r-button",{onclick:"message.toast('这是一条提示')"},"toast 提示",-1),d=a("",16),u=[t,c,r,F,i,y,D,B,d];function h(m,g,E,A,b,C){return l(),o("div",null,u)}const _=n(e,[["render",h]]);export{v as __pageData,_ as default};
      +import{_ as n,o as l,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/customElements.ed27d2e2.png",v=JSON.parse('{"title":"ranui","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/index.md","lastUpdated":1694419899000}'),e={name:"src/ranui/index.md"},t=a("",40),c=s("div",{style:{display:"flex"}},[s("r-icon",{name:"lock",size:"50"}),s("r-icon",{name:"user",size:"50"}),s("r-icon",{name:"loading",size:"50",color:"#1E90FF",spin:""})],-1),r=a("",9),F=s("r-button",{onclick:"message.info('这是一条提示')"},"信息提示",-1),i=s("r-button",{onclick:"message.warning('这是一条提示')"},"警告提示",-1),y=s("r-button",{onclick:"message.error('这是一条提示')"},"错误提示",-1),D=s("r-button",{onclick:"message.success('这是一条提示')"},"成功提示",-1),B=s("r-button",{onclick:"message.toast('这是一条提示')"},"toast 提示",-1),d=a("",16),u=[t,c,r,F,i,y,D,B,d];function h(m,g,E,A,b,C){return l(),o("div",null,u)}const _=n(e,[["render",h]]);export{v as __pageData,_ as default};
      diff --git a/assets/src_ranui_input_index.md.24d6061b.js b/assets/src_ranui_input_index.md.755c9dc7.js
      similarity index 99%
      rename from assets/src_ranui_input_index.md.24d6061b.js
      rename to assets/src_ranui_input_index.md.755c9dc7.js
      index ace2f6e2e..7f265cce1 100644
      --- a/assets/src_ranui_input_index.md.24d6061b.js
      +++ b/assets/src_ranui_input_index.md.755c9dc7.js
      @@ -1,4 +1,4 @@
      -import{_ as n,o as l,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/input-input.f80a45d3.jpg",q=JSON.parse('{"title":"Input 输入框","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/input/index.md","lastUpdated":1694276611000}'),e={name:"src/ranui/input/index.md"},t=a('

      Input 输入框

      通过鼠标或键盘输入内容,是最基础的表单域的包装。

      代码演示

      输入框:
      xml
      <r-input></r-input>

      属性

      标签label

      提供类似于 Metiral Design 的输入体验。

      html
      <r-input label="user"></r-input>

      占位placeholder

      与原生placeholder一致。

      html
      <r-input placeholder="user"></r-input>

      禁用disabled

      通过disabled可以禁用输入框,禁用后该按钮上的事件失效。

      html
      <r-input label="user" disabled></r-input>

      value

      设置或返回输入框的value属性值。

      类型type

      目前支持passwordnumber这几种类型,设置后会出现额外的ui控件。

      密码输入框

      支持密码明文和密文切换。

      html
      <r-input icon="lock" type="password"></r-input>

      图标icon

      可以设置一个icon来表示标签标识。

      html
      <r-input icon="user"></r-input>

      数字输入框

      数字输入框,类似于原生input[type=number],支持minmaxstep属性,支持键盘上下键切换数字。

      html
      <r-input type="number" min="-10" max="10" step="0.5"></r-input>

      name 属性名

      跟 form 组件联动的时候有效,form 提交时收集的字段名字

      status 状态

      • error

      默认色值: #ff4d4f

      ',40),c=s("div",null,[s("r-input",{status:"error"})],-1),r=a('
      xml
      <r-input status="error"></r-input>
      • warning

      默认色值: #ff7875

      ',3),D=s("div",null,[s("r-input",{status:"warning"})],-1),i=a('
      xml
      <r-input  status="warning"></r-input>

      事件event

      常见的回调事件。

      onchange

      文本改变的时候触发。

      ',5),F=s("r-input",{onchange:"console.log(this.value)"},null,-1),y=a(`
      html
      <r-input onchange="func(this.value)"></r-input>
      js
      const input = document.createElement('r-input')
      +import{_ as n,o as l,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/input-input.f80a45d3.jpg",q=JSON.parse('{"title":"Input 输入框","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/input/index.md","lastUpdated":1694419899000}'),e={name:"src/ranui/input/index.md"},t=a('

      Input 输入框

      通过鼠标或键盘输入内容,是最基础的表单域的包装。

      代码演示

      输入框:
      xml
      <r-input></r-input>

      属性

      标签label

      提供类似于 Metiral Design 的输入体验。

      html
      <r-input label="user"></r-input>

      占位placeholder

      与原生placeholder一致。

      html
      <r-input placeholder="user"></r-input>

      禁用disabled

      通过disabled可以禁用输入框,禁用后该按钮上的事件失效。

      html
      <r-input label="user" disabled></r-input>

      value

      设置或返回输入框的value属性值。

      类型type

      目前支持passwordnumber这几种类型,设置后会出现额外的ui控件。

      密码输入框

      支持密码明文和密文切换。

      html
      <r-input icon="lock" type="password"></r-input>

      图标icon

      可以设置一个icon来表示标签标识。

      html
      <r-input icon="user"></r-input>

      数字输入框

      数字输入框,类似于原生input[type=number],支持minmaxstep属性,支持键盘上下键切换数字。

      html
      <r-input type="number" min="-10" max="10" step="0.5"></r-input>

      name 属性名

      跟 form 组件联动的时候有效,form 提交时收集的字段名字

      status 状态

      • error

      默认色值: #ff4d4f

      ',40),c=s("div",null,[s("r-input",{status:"error"})],-1),r=a('
      xml
      <r-input status="error"></r-input>
      • warning

      默认色值: #ff7875

      ',3),D=s("div",null,[s("r-input",{status:"warning"})],-1),i=a('
      xml
      <r-input  status="warning"></r-input>

      事件event

      常见的回调事件。

      onchange

      文本改变的时候触发。

      ',5),F=s("r-input",{onchange:"console.log(this.value)"},null,-1),y=a(`
      html
      <r-input onchange="func(this.value)"></r-input>
      js
      const input = document.createElement('r-input')
       input.setAttribute('label', 'home')
       const func = (e) => {
         console.log(e)
      diff --git a/assets/src_ranui_input_index.md.24d6061b.lean.js b/assets/src_ranui_input_index.md.755c9dc7.lean.js
      similarity index 91%
      rename from assets/src_ranui_input_index.md.24d6061b.lean.js
      rename to assets/src_ranui_input_index.md.755c9dc7.lean.js
      index 88e08a3ec..881369f4b 100644
      --- a/assets/src_ranui_input_index.md.24d6061b.lean.js
      +++ b/assets/src_ranui_input_index.md.755c9dc7.lean.js
      @@ -1 +1 @@
      -import{_ as n,o as l,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/input-input.f80a45d3.jpg",q=JSON.parse('{"title":"Input 输入框","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/input/index.md","lastUpdated":1694276611000}'),e={name:"src/ranui/input/index.md"},t=a("",40),c=s("div",null,[s("r-input",{status:"error"})],-1),r=a("",3),D=s("div",null,[s("r-input",{status:"warning"})],-1),i=a("",5),F=s("r-input",{onchange:"console.log(this.value)"},null,-1),y=a("",4),d=s("r-input",{oninput:"console.log(this.value)"},null,-1),u=a("",2),h=[t,c,r,D,i,F,y,d,u];function _(m,A,E,g,b,B){return l(),o("div",null,h)}const v=n(e,[["render",_]]);export{q as __pageData,v as default};
      +import{_ as n,o as l,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/input-input.f80a45d3.jpg",q=JSON.parse('{"title":"Input 输入框","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/input/index.md","lastUpdated":1694419899000}'),e={name:"src/ranui/input/index.md"},t=a("",40),c=s("div",null,[s("r-input",{status:"error"})],-1),r=a("",3),D=s("div",null,[s("r-input",{status:"warning"})],-1),i=a("",5),F=s("r-input",{onchange:"console.log(this.value)"},null,-1),y=a("",4),d=s("r-input",{oninput:"console.log(this.value)"},null,-1),u=a("",2),h=[t,c,r,D,i,F,y,d,u];function _(m,A,E,g,b,B){return l(),o("div",null,h)}const v=n(e,[["render",_]]);export{q as __pageData,v as default};
      diff --git a/assets/src_ranui_message_index.md.fa827f88.js b/assets/src_ranui_message_index.md.2e7373db.js
      similarity index 99%
      rename from assets/src_ranui_message_index.md.fa827f88.js
      rename to assets/src_ranui_message_index.md.2e7373db.js
      index e395c4f69..a5f0443cc 100644
      --- a/assets/src_ranui_message_index.md.fa827f88.js
      +++ b/assets/src_ranui_message_index.md.2e7373db.js
      @@ -1,4 +1,4 @@
      -import{_ as n,o as t,c as l,x as s,a as o,N as a}from"./chunks/framework.6fe2e870.js";const q=JSON.parse('{"title":"message 全局提示","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/message/index.md","lastUpdated":1694276611000}'),e={name:"src/ranui/message/index.md"},p=s("h1",{id:"message-全局提示",tabindex:"-1"},[o("message 全局提示 "),s("a",{class:"header-anchor",href:"#message-全局提示","aria-label":'Permalink to "message 全局提示"'},"​")],-1),c=s("p",null,"全局展示操作反馈信息。",-1),r=s("h2",{id:"代码演示",tabindex:"-1"},[o("代码演示 "),s("a",{class:"header-anchor",href:"#代码演示","aria-label":'Permalink to "代码演示"'},"​")],-1),D=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{type:"primary",onclick:"message.info('这是一条提示')"},"点击触发全局提示")],-1),F=a('
      xml
      <r-button type="primary" onclick="message.info('这是一条提示')">点击触发全局提示</r-button>

      属性

      类型type

      不同的提示类型

      ',4),i=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.info('这是一条提示')"},"信息提示")],-1),y=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.warning('这是一条提示')"},"警告提示")],-1),d=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.error('这是一条提示')"},"错误提示")],-1),g=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.success('这是一条提示')"},"成功提示")],-1),m=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.toast('这是一条提示')"},"toast提示")],-1),u=a(`
      html
      <r-button onclick="message.info('这是一条提示')">信息提示</r-button>
      +import{_ as n,o as t,c as l,x as s,a as o,N as a}from"./chunks/framework.6fe2e870.js";const q=JSON.parse('{"title":"message 全局提示","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/message/index.md","lastUpdated":1694419899000}'),e={name:"src/ranui/message/index.md"},p=s("h1",{id:"message-全局提示",tabindex:"-1"},[o("message 全局提示 "),s("a",{class:"header-anchor",href:"#message-全局提示","aria-label":'Permalink to "message 全局提示"'},"​")],-1),c=s("p",null,"全局展示操作反馈信息。",-1),r=s("h2",{id:"代码演示",tabindex:"-1"},[o("代码演示 "),s("a",{class:"header-anchor",href:"#代码演示","aria-label":'Permalink to "代码演示"'},"​")],-1),D=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{type:"primary",onclick:"message.info('这是一条提示')"},"点击触发全局提示")],-1),F=a('
      xml
      <r-button type="primary" onclick="message.info('这是一条提示')">点击触发全局提示</r-button>

      属性

      类型type

      不同的提示类型

      ',4),i=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.info('这是一条提示')"},"信息提示")],-1),y=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.warning('这是一条提示')"},"警告提示")],-1),d=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.error('这是一条提示')"},"错误提示")],-1),g=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.success('这是一条提示')"},"成功提示")],-1),m=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.toast('这是一条提示')"},"toast提示")],-1),u=a(`
      html
      <r-button onclick="message.info('这是一条提示')">信息提示</r-button>
       <r-button onclick="message.warning('这是一条提示')">警告提示</r-button>
       <r-button onclick="message.error('这是一条提示')">错误提示</r-button>
       <r-button onclick="message.success('这是一条提示')">成功提示</r-button>
      diff --git a/assets/src_ranui_message_index.md.fa827f88.lean.js b/assets/src_ranui_message_index.md.2e7373db.lean.js
      similarity index 96%
      rename from assets/src_ranui_message_index.md.fa827f88.lean.js
      rename to assets/src_ranui_message_index.md.2e7373db.lean.js
      index 6352a01d2..6bce2bb2e 100644
      --- a/assets/src_ranui_message_index.md.fa827f88.lean.js
      +++ b/assets/src_ranui_message_index.md.2e7373db.lean.js
      @@ -1 +1 @@
      -import{_ as n,o as t,c as l,x as s,a as o,N as a}from"./chunks/framework.6fe2e870.js";const q=JSON.parse('{"title":"message 全局提示","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/message/index.md","lastUpdated":1694276611000}'),e={name:"src/ranui/message/index.md"},p=s("h1",{id:"message-全局提示",tabindex:"-1"},[o("message 全局提示 "),s("a",{class:"header-anchor",href:"#message-全局提示","aria-label":'Permalink to "message 全局提示"'},"​")],-1),c=s("p",null,"全局展示操作反馈信息。",-1),r=s("h2",{id:"代码演示",tabindex:"-1"},[o("代码演示 "),s("a",{class:"header-anchor",href:"#代码演示","aria-label":'Permalink to "代码演示"'},"​")],-1),D=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{type:"primary",onclick:"message.info('这是一条提示')"},"点击触发全局提示")],-1),F=a("",4),i=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.info('这是一条提示')"},"信息提示")],-1),y=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.warning('这是一条提示')"},"警告提示")],-1),d=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.error('这是一条提示')"},"错误提示")],-1),g=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.success('这是一条提示')"},"成功提示")],-1),m=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.toast('这是一条提示')"},"toast提示")],-1),u=a("",16),h=[p,c,r,D,F,i,y,d,g,m,u];function _(b,E,A,C,k,x){return t(),l("div",null,h)}const f=n(e,[["render",_]]);export{q as __pageData,f as default};
      +import{_ as n,o as t,c as l,x as s,a as o,N as a}from"./chunks/framework.6fe2e870.js";const q=JSON.parse('{"title":"message 全局提示","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/message/index.md","lastUpdated":1694419899000}'),e={name:"src/ranui/message/index.md"},p=s("h1",{id:"message-全局提示",tabindex:"-1"},[o("message 全局提示 "),s("a",{class:"header-anchor",href:"#message-全局提示","aria-label":'Permalink to "message 全局提示"'},"​")],-1),c=s("p",null,"全局展示操作反馈信息。",-1),r=s("h2",{id:"代码演示",tabindex:"-1"},[o("代码演示 "),s("a",{class:"header-anchor",href:"#代码演示","aria-label":'Permalink to "代码演示"'},"​")],-1),D=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{type:"primary",onclick:"message.info('这是一条提示')"},"点击触发全局提示")],-1),F=a("",4),i=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.info('这是一条提示')"},"信息提示")],-1),y=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.warning('这是一条提示')"},"警告提示")],-1),d=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.error('这是一条提示')"},"错误提示")],-1),g=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.success('这是一条提示')"},"成功提示")],-1),m=s("div",{style:{display:"inline-block","margin-right":"8px","margin-bottom":"12px"}},[s("r-button",{onclick:"message.toast('这是一条提示')"},"toast提示")],-1),u=a("",16),h=[p,c,r,D,F,i,y,d,g,m,u];function _(b,E,A,C,k,x){return t(),l("div",null,h)}const f=n(e,[["render",_]]);export{q as __pageData,f as default};
      diff --git a/assets/src_ranui_modal_index.md.f410050d.js b/assets/src_ranui_modal_index.md.69c2268b.js
      similarity index 82%
      rename from assets/src_ranui_modal_index.md.f410050d.js
      rename to assets/src_ranui_modal_index.md.69c2268b.js
      index 70fb0ffb7..9b293fec1 100644
      --- a/assets/src_ranui_modal_index.md.f410050d.js
      +++ b/assets/src_ranui_modal_index.md.69c2268b.js
      @@ -1 +1 @@
      -import{_ as e,o as t,c as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/modal/index.md","lastUpdated":1694276611000}'),r={name:"src/ranui/modal/index.md"};function n(o,s,c,d,i,p){return t(),a("div")}const l=e(r,[["render",n]]);export{m as __pageData,l as default};
      +import{_ as e,o as t,c as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/modal/index.md","lastUpdated":1694419899000}'),r={name:"src/ranui/modal/index.md"};function n(o,s,c,d,i,p){return t(),a("div")}const l=e(r,[["render",n]]);export{m as __pageData,l as default};
      diff --git a/assets/src_ranui_modal_index.md.f410050d.lean.js b/assets/src_ranui_modal_index.md.69c2268b.lean.js
      similarity index 82%
      rename from assets/src_ranui_modal_index.md.f410050d.lean.js
      rename to assets/src_ranui_modal_index.md.69c2268b.lean.js
      index 70fb0ffb7..9b293fec1 100644
      --- a/assets/src_ranui_modal_index.md.f410050d.lean.js
      +++ b/assets/src_ranui_modal_index.md.69c2268b.lean.js
      @@ -1 +1 @@
      -import{_ as e,o as t,c as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/modal/index.md","lastUpdated":1694276611000}'),r={name:"src/ranui/modal/index.md"};function n(o,s,c,d,i,p){return t(),a("div")}const l=e(r,[["render",n]]);export{m as __pageData,l as default};
      +import{_ as e,o as t,c as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/modal/index.md","lastUpdated":1694419899000}'),r={name:"src/ranui/modal/index.md"};function n(o,s,c,d,i,p){return t(),a("div")}const l=e(r,[["render",n]]);export{m as __pageData,l as default};
      diff --git a/assets/src_ranui_preview_index.md.9a6fa2f6.js b/assets/src_ranui_preview_index.md.fa3a8f12.js
      similarity index 99%
      rename from assets/src_ranui_preview_index.md.9a6fa2f6.js
      rename to assets/src_ranui_preview_index.md.fa3a8f12.js
      index de0836333..11bde954a 100644
      --- a/assets/src_ranui_preview_index.md.9a6fa2f6.js
      +++ b/assets/src_ranui_preview_index.md.fa3a8f12.js
      @@ -1,4 +1,4 @@
      -import{_ as n,o as l,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"preview 文件预览","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/preview/index.md","lastUpdated":1694276611000}'),p={name:"src/ranui/preview/index.md"},e=a('

      preview 文件预览

      支持docxpptxpdf,xlsx文件的预览

      代码演示

      ',3),t=s("div",{style:{width:"100px","margin-top":"10px"}},[s("r-preview",{id:"preview"}),s("r-button",{type:"primary",onclick:"uploadFile('preview')"},"choose file to preview")],-1),c=a(`
      html
      <r-preview id="preview"></r-preview>
      +import{_ as n,o as l,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"preview 文件预览","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/preview/index.md","lastUpdated":1694419899000}'),p={name:"src/ranui/preview/index.md"},e=a('

      preview 文件预览

      支持docxpptxpdf,xlsx文件的预览

      代码演示

      ',3),t=s("div",{style:{width:"100px","margin-top":"10px"}},[s("r-preview",{id:"preview"}),s("r-button",{type:"primary",onclick:"uploadFile('preview')"},"choose file to preview")],-1),c=a(`
      html
      <r-preview id="preview"></r-preview>
       <r-button type="primary" onclick="uploadFile()"
         >choose file to preview</r-button
       >
      diff --git a/assets/src_ranui_preview_index.md.9a6fa2f6.lean.js b/assets/src_ranui_preview_index.md.fa3a8f12.lean.js
      similarity index 89%
      rename from assets/src_ranui_preview_index.md.9a6fa2f6.lean.js
      rename to assets/src_ranui_preview_index.md.fa3a8f12.lean.js
      index 3c6a46b7c..b08652b8a 100644
      --- a/assets/src_ranui_preview_index.md.9a6fa2f6.lean.js
      +++ b/assets/src_ranui_preview_index.md.fa3a8f12.lean.js
      @@ -1 +1 @@
      -import{_ as n,o as l,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"preview 文件预览","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/preview/index.md","lastUpdated":1694276611000}'),p={name:"src/ranui/preview/index.md"},e=a("",3),t=s("div",{style:{width:"100px","margin-top":"10px"}},[s("r-preview",{id:"preview"}),s("r-button",{type:"primary",onclick:"uploadFile('preview')"},"choose file to preview")],-1),c=a("",8),r=[e,t,c];function F(y,D,i,d,A,u){return l(),o("div",null,r)}const h=n(p,[["render",F]]);export{E as __pageData,h as default};
      +import{_ as n,o as l,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"preview 文件预览","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/preview/index.md","lastUpdated":1694419899000}'),p={name:"src/ranui/preview/index.md"},e=a("",3),t=s("div",{style:{width:"100px","margin-top":"10px"}},[s("r-preview",{id:"preview"}),s("r-button",{type:"primary",onclick:"uploadFile('preview')"},"choose file to preview")],-1),c=a("",8),r=[e,t,c];function F(y,D,i,d,A,u){return l(),o("div",null,r)}const h=n(p,[["render",F]]);export{E as __pageData,h as default};
      diff --git a/assets/src_ranui_skeleton_index.md.472eb512.js b/assets/src_ranui_skeleton_index.md.51e3be48.js
      similarity index 95%
      rename from assets/src_ranui_skeleton_index.md.472eb512.js
      rename to assets/src_ranui_skeleton_index.md.51e3be48.js
      index cbcad4d57..ac770c3f0 100644
      --- a/assets/src_ranui_skeleton_index.md.472eb512.js
      +++ b/assets/src_ranui_skeleton_index.md.51e3be48.js
      @@ -1 +1 @@
      -import{_ as e,o as t,c as s,N as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"skeleton 骨架屏","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/skeleton/index.md","lastUpdated":1694276611000}'),n={name:"src/ranui/skeleton/index.md"},o=a('

      skeleton 骨架屏

      在需要等待加载内容的位置提供一个占位图形组合。

      代码演示

      骨架长度跟随父级元素的长度

      xml
      <r-skeleton ></r-skeleton>
      ',9),l=[o];function r(i,p,c,d,_,k){return t(),s("div",null,l)}const x=e(n,[["render",r]]);export{m as __pageData,x as default}; +import{_ as e,o as t,c as s,N as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"skeleton 骨架屏","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/skeleton/index.md","lastUpdated":1694419899000}'),n={name:"src/ranui/skeleton/index.md"},o=a('

      skeleton 骨架屏

      在需要等待加载内容的位置提供一个占位图形组合。

      代码演示

      骨架长度跟随父级元素的长度

      xml
      <r-skeleton ></r-skeleton>
      ',9),l=[o];function r(i,p,c,d,_,k){return t(),s("div",null,l)}const x=e(n,[["render",r]]);export{m as __pageData,x as default}; diff --git a/assets/src_ranui_skeleton_index.md.472eb512.lean.js b/assets/src_ranui_skeleton_index.md.51e3be48.lean.js similarity index 84% rename from assets/src_ranui_skeleton_index.md.472eb512.lean.js rename to assets/src_ranui_skeleton_index.md.51e3be48.lean.js index 99014109e..cf3d40325 100644 --- a/assets/src_ranui_skeleton_index.md.472eb512.lean.js +++ b/assets/src_ranui_skeleton_index.md.51e3be48.lean.js @@ -1 +1 @@ -import{_ as e,o as t,c as s,N as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"skeleton 骨架屏","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/skeleton/index.md","lastUpdated":1694276611000}'),n={name:"src/ranui/skeleton/index.md"},o=a("",9),l=[o];function r(i,p,c,d,_,k){return t(),s("div",null,l)}const x=e(n,[["render",r]]);export{m as __pageData,x as default}; +import{_ as e,o as t,c as s,N as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"skeleton 骨架屏","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/skeleton/index.md","lastUpdated":1694419899000}'),n={name:"src/ranui/skeleton/index.md"},o=a("",9),l=[o];function r(i,p,c,d,_,k){return t(),s("div",null,l)}const x=e(n,[["render",r]]);export{m as __pageData,x as default}; diff --git a/assets/src_ranui_tab_index.md.087770cd.js b/assets/src_ranui_tab_index.md.ec0122dc.js similarity index 99% rename from assets/src_ranui_tab_index.md.087770cd.js rename to assets/src_ranui_tab_index.md.ec0122dc.js index 423bef8de..40e8b68cd 100644 --- a/assets/src_ranui_tab_index.md.087770cd.js +++ b/assets/src_ranui_tab_index.md.ec0122dc.js @@ -1,4 +1,4 @@ -import{_ as l,o as n,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const q=JSON.parse('{"title":"Tab 图标","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/tab/index.md","lastUpdated":1694276611000}'),p={name:"src/ranui/tab/index.md"},t=a(`

      Tab 图标

      标签页,其中r-tab需要和r-tabs搭配使用

      代码演示

      111112222233333
      xml
      <r-tabs >
      +import{_ as l,o as n,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const q=JSON.parse('{"title":"Tab 图标","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/tab/index.md","lastUpdated":1694419899000}'),p={name:"src/ranui/tab/index.md"},t=a(`

      Tab 图标

      标签页,其中r-tab需要和r-tabs搭配使用

      代码演示

      111112222233333
      xml
      <r-tabs >
             <r-tab label="tab1">11111</r-tab>
             <r-tab label="tab2">22222</r-tab>
             <r-tab label="tab3">33333</r-tab>
      diff --git a/assets/src_ranui_tab_index.md.087770cd.lean.js b/assets/src_ranui_tab_index.md.ec0122dc.lean.js
      similarity index 94%
      rename from assets/src_ranui_tab_index.md.087770cd.lean.js
      rename to assets/src_ranui_tab_index.md.ec0122dc.lean.js
      index 170d128fa..c6cd31339 100644
      --- a/assets/src_ranui_tab_index.md.087770cd.lean.js
      +++ b/assets/src_ranui_tab_index.md.ec0122dc.lean.js
      @@ -1 +1 @@
      -import{_ as l,o as n,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const q=JSON.parse('{"title":"Tab 图标","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/tab/index.md","lastUpdated":1694276611000}'),p={name:"src/ranui/tab/index.md"},t=a("",13),e=s("div",{style:{width:"100%"}},[s("r-tabs",{active:"1"},[s("r-tab",{label:"tab1"},"11111"),s("r-tab",{label:"tab2"},"22222"),s("r-tab",{label:"tab3"},"33333")])],-1),r=a("",2),c=s("div",{style:{width:"100%"}},[s("r-tabs",{active:"c"},[s("r-tab",{label:"tab1",ranKey:"a"},"11111"),s("r-tab",{label:"tab2",ranKey:"b"},"22222"),s("r-tab",{label:"tab3",ranKey:"c"},"33333"),s("r-tab",{label:"tab4"},"4")])],-1),D=a("",3),F=s("div",{style:{width:"100%"}},[s("r-tabs",{active:"c"},[s("r-tab",{label:"tab1",ranKey:"a",disabled:""},"11111"),s("r-tab",{label:"tab2",ranKey:"b"},"22222"),s("r-tab",{label:"tab3",ranKey:"c"},"33333"),s("r-tab",{label:"tab4"},"4")])],-1),y=a("",20),b=[t,e,r,c,D,F,y];function i(d,E,u,B,g,A){return n(),o("div",null,b)}const h=l(p,[["render",i]]);export{q as __pageData,h as default};
      +import{_ as l,o as n,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const q=JSON.parse('{"title":"Tab 图标","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/tab/index.md","lastUpdated":1694419899000}'),p={name:"src/ranui/tab/index.md"},t=a("",13),e=s("div",{style:{width:"100%"}},[s("r-tabs",{active:"1"},[s("r-tab",{label:"tab1"},"11111"),s("r-tab",{label:"tab2"},"22222"),s("r-tab",{label:"tab3"},"33333")])],-1),r=a("",2),c=s("div",{style:{width:"100%"}},[s("r-tabs",{active:"c"},[s("r-tab",{label:"tab1",ranKey:"a"},"11111"),s("r-tab",{label:"tab2",ranKey:"b"},"22222"),s("r-tab",{label:"tab3",ranKey:"c"},"33333"),s("r-tab",{label:"tab4"},"4")])],-1),D=a("",3),F=s("div",{style:{width:"100%"}},[s("r-tabs",{active:"c"},[s("r-tab",{label:"tab1",ranKey:"a",disabled:""},"11111"),s("r-tab",{label:"tab2",ranKey:"b"},"22222"),s("r-tab",{label:"tab3",ranKey:"c"},"33333"),s("r-tab",{label:"tab4"},"4")])],-1),y=a("",20),b=[t,e,r,c,D,F,y];function i(d,E,u,B,g,A){return n(),o("div",null,b)}const h=l(p,[["render",i]]);export{q as __pageData,h as default};
      diff --git a/assets/src_ranui_tabs_index.md.eff91b6b.js b/assets/src_ranui_tabs_index.md.301a2c3b.js
      similarity index 99%
      rename from assets/src_ranui_tabs_index.md.eff91b6b.js
      rename to assets/src_ranui_tabs_index.md.301a2c3b.js
      index 7c80133b2..ff0e62d9f 100644
      --- a/assets/src_ranui_tabs_index.md.eff91b6b.js
      +++ b/assets/src_ranui_tabs_index.md.301a2c3b.js
      @@ -1,4 +1,4 @@
      -import{_ as l,o as n,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const B=JSON.parse('{"title":"Tab","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/tabs/index.md","lastUpdated":1694276611000}'),p={name:"src/ranui/tabs/index.md"},t=a(`

      Tab

      代码展示

      tab1tab2tab3
      xml
      <r-tabs>
      +import{_ as l,o as n,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const B=JSON.parse('{"title":"Tab","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/tabs/index.md","lastUpdated":1694419899000}'),p={name:"src/ranui/tabs/index.md"},t=a(`

      Tab

      代码展示

      tab1tab2tab3
      xml
      <r-tabs>
           <r-tab label="tab1">tab1</r-tab>
           <r-tab label="tab2">tab2</r-tab>
           <r-tab label="tab3">tab3</r-tab>
      diff --git a/assets/src_ranui_tabs_index.md.eff91b6b.lean.js b/assets/src_ranui_tabs_index.md.301a2c3b.lean.js
      similarity index 91%
      rename from assets/src_ranui_tabs_index.md.eff91b6b.lean.js
      rename to assets/src_ranui_tabs_index.md.301a2c3b.lean.js
      index d10bb61a3..07af12a71 100644
      --- a/assets/src_ranui_tabs_index.md.eff91b6b.lean.js
      +++ b/assets/src_ranui_tabs_index.md.301a2c3b.lean.js
      @@ -1 +1 @@
      -import{_ as l,o as n,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const B=JSON.parse('{"title":"Tab","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/tabs/index.md","lastUpdated":1694276611000}'),p={name:"src/ranui/tabs/index.md"},t=a("",16),e=s("r-tabs",{active:"B"},[s("r-tab",{label:"tab1","r-key":"A"},"tab1"),s("r-tab",{label:"tab2","r-key":"B"},"tab2"),s("r-tab",{label:"tab3","r-key":"C"},"tab3")],-1),c=a("",6),r=s("r-tabs",null,[s("r-tab",{icon:"home",iconSize:"22"},"tab1"),s("r-tab",{icon:"message",iconSize:"22"},"tab2"),s("r-tab",{icon:"user",iconSize:"22"},"tab3")],-1),D=a("",7),F=[t,e,c,r,D];function y(b,i,d,u,h,E){return n(),o("div",null,F)}const q=l(p,[["render",y]]);export{B as __pageData,q as default};
      +import{_ as l,o as n,c as o,N as a,x as s}from"./chunks/framework.6fe2e870.js";const B=JSON.parse('{"title":"Tab","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranui/tabs/index.md","lastUpdated":1694419899000}'),p={name:"src/ranui/tabs/index.md"},t=a("",16),e=s("r-tabs",{active:"B"},[s("r-tab",{label:"tab1","r-key":"A"},"tab1"),s("r-tab",{label:"tab2","r-key":"B"},"tab2"),s("r-tab",{label:"tab3","r-key":"C"},"tab3")],-1),c=a("",6),r=s("r-tabs",null,[s("r-tab",{icon:"home",iconSize:"22"},"tab1"),s("r-tab",{icon:"message",iconSize:"22"},"tab2"),s("r-tab",{icon:"user",iconSize:"22"},"tab3")],-1),D=a("",7),F=[t,e,c,r,D];function y(b,i,d,u,h,E){return n(),o("div",null,F)}const q=l(p,[["render",y]]);export{B as __pageData,q as default};
      diff --git a/assets/src_ranuts_binaryTree_index.md.9df1bc99.js b/assets/src_ranuts_binaryTree_index.md.493e0878.js
      similarity index 99%
      rename from assets/src_ranuts_binaryTree_index.md.9df1bc99.js
      rename to assets/src_ranuts_binaryTree_index.md.493e0878.js
      index 067d46815..d0eccc697 100644
      --- a/assets/src_ranuts_binaryTree_index.md.9df1bc99.js
      +++ b/assets/src_ranuts_binaryTree_index.md.493e0878.js
      @@ -1 +1 @@
      -import{_ as a,o as e,c as r,N as i}from"./chunks/framework.6fe2e870.js";const t="/ran/assets/balanceTree.3d01d012.png",m=JSON.parse('{"title":"二叉树的定义","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/binaryTree/index.md","lastUpdated":1694276611000}'),h={name:"src/ranuts/binaryTree/index.md"},l=i('

      二叉树的定义

      在计算机科学中,二叉树(Binary tree)是每个节点最多只有两个分支(即不存在分支度大于 2 的节点)的树结构。通常分支被称作“左子树”或“右子树”。二叉树的分支具有左右次序,不能随意颠倒[1]。。

      二叉树的性质

      • 在二叉树的第 i 层上最多有 2^(i-1)个结点(i>=1)
      • 深度为 h 的二叉树,最多有 2^h-1 个结点,最少有 h 个结点(h>=1)
      • 包含 n 个结点的二叉树的高度至少为(log2n)+1
      • 非空的二叉树,分支度为 0 的总数为 n0,分支度为 2 的总数为 n2,则 n0=n2+1
      • 二叉树的总结点数 n = n1 + n2 + n0
      • 总连线数等于总节点数减一(B = n - 1)
      • 总连线数等于分支度为 2 的节点的两倍加上分支度为 1 的节点(B = n2 _ 2 + n1 _ 1)

      二叉树的类型

      满二叉树

      一棵深度为 k 且有 2k-1 个节点的二叉树称为满二叉树。 除最后一层无任何子节点外,每一层上的所有结点都有两个子结点的二叉树[2]

      完全二叉树

      一棵深度为 k 的有 n 个结点的二叉树,对树中的结点按从上至下、从左到右的顺序进行编号,如果编号为 i(1≤i≤n)的结点与满二叉树中编号为 i 的结点在二叉树中的位置相同,则这棵二叉树称为完全二叉树。

      二叉搜索树

      二叉搜索树(BST)又称二叉查找树或二叉排序树。它或者是一棵空树,或者是具有下列性质的二叉树: 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值; 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值; 它的左、右子树也分别为二叉排序树。

      平衡二叉树

      平衡二叉树(AVL)一定是二叉搜索树,且左子树和右子树的高度差的绝对值不超过 1。 平衡二叉树

      B 树

      B 树属于多叉树又名平衡多路查找树(查找路径不只两个)

      B+树

      B+树是 B 树的变体,也是一种多路搜索树。

      B*树

      B* 树是 B+树的变体,在 B+树的非根和非叶子结点再增加指向兄弟的指针;B* 树定义了非叶子结点关键字个数至少为(2/3)M,即块的最低使用率为 2/3(代替 B+树的 1/2)。B 树分配新结点的概率比 B+树要低,空间使用率更高;

      红黑树

      红黑树是一种平衡二叉查找树的变体,它的左右子树高差有可能大于 1,所以红黑树不是严格意义上的平衡二叉树(AVL),但对它进行平衡的代价较低, 其平均统计性能要强于 AVL 。

      遍历

      前序遍历

      后序遍历

      中序遍历

      层序遍历

      常见算法题

      镜像二叉树

      重建二叉树

      二叉树深度

      二叉树节点总数

      判断二叉树子结构

      输入两棵二叉树 A 和 B,判断 B 是不是 A 的子结构。(ps:约定空树不是任意一个树的子结构)

      参考文档

      1. 维基百科二叉树
      2. 百度百科满二叉树
      ',36),o=[l];function n(d,s,c,u,b,q){return e(),r("div",null,o)}const f=a(h,[["render",n]]);export{m as __pageData,f as default}; +import{_ as a,o as e,c as r,N as i}from"./chunks/framework.6fe2e870.js";const t="/ran/assets/balanceTree.3d01d012.png",m=JSON.parse('{"title":"二叉树的定义","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/binaryTree/index.md","lastUpdated":1694419899000}'),h={name:"src/ranuts/binaryTree/index.md"},l=i('

      二叉树的定义

      在计算机科学中,二叉树(Binary tree)是每个节点最多只有两个分支(即不存在分支度大于 2 的节点)的树结构。通常分支被称作“左子树”或“右子树”。二叉树的分支具有左右次序,不能随意颠倒[1]。。

      二叉树的性质

      • 在二叉树的第 i 层上最多有 2^(i-1)个结点(i>=1)
      • 深度为 h 的二叉树,最多有 2^h-1 个结点,最少有 h 个结点(h>=1)
      • 包含 n 个结点的二叉树的高度至少为(log2n)+1
      • 非空的二叉树,分支度为 0 的总数为 n0,分支度为 2 的总数为 n2,则 n0=n2+1
      • 二叉树的总结点数 n = n1 + n2 + n0
      • 总连线数等于总节点数减一(B = n - 1)
      • 总连线数等于分支度为 2 的节点的两倍加上分支度为 1 的节点(B = n2 _ 2 + n1 _ 1)

      二叉树的类型

      满二叉树

      一棵深度为 k 且有 2k-1 个节点的二叉树称为满二叉树。 除最后一层无任何子节点外,每一层上的所有结点都有两个子结点的二叉树[2]

      完全二叉树

      一棵深度为 k 的有 n 个结点的二叉树,对树中的结点按从上至下、从左到右的顺序进行编号,如果编号为 i(1≤i≤n)的结点与满二叉树中编号为 i 的结点在二叉树中的位置相同,则这棵二叉树称为完全二叉树。

      二叉搜索树

      二叉搜索树(BST)又称二叉查找树或二叉排序树。它或者是一棵空树,或者是具有下列性质的二叉树: 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值; 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值; 它的左、右子树也分别为二叉排序树。

      平衡二叉树

      平衡二叉树(AVL)一定是二叉搜索树,且左子树和右子树的高度差的绝对值不超过 1。 平衡二叉树

      B 树

      B 树属于多叉树又名平衡多路查找树(查找路径不只两个)

      B+树

      B+树是 B 树的变体,也是一种多路搜索树。

      B*树

      B* 树是 B+树的变体,在 B+树的非根和非叶子结点再增加指向兄弟的指针;B* 树定义了非叶子结点关键字个数至少为(2/3)M,即块的最低使用率为 2/3(代替 B+树的 1/2)。B 树分配新结点的概率比 B+树要低,空间使用率更高;

      红黑树

      红黑树是一种平衡二叉查找树的变体,它的左右子树高差有可能大于 1,所以红黑树不是严格意义上的平衡二叉树(AVL),但对它进行平衡的代价较低, 其平均统计性能要强于 AVL 。

      遍历

      前序遍历

      后序遍历

      中序遍历

      层序遍历

      常见算法题

      镜像二叉树

      重建二叉树

      二叉树深度

      二叉树节点总数

      判断二叉树子结构

      输入两棵二叉树 A 和 B,判断 B 是不是 A 的子结构。(ps:约定空树不是任意一个树的子结构)

      参考文档

      1. 维基百科二叉树
      2. 百度百科满二叉树
      ',36),o=[l];function n(d,s,c,u,b,q){return e(),r("div",null,o)}const f=a(h,[["render",n]]);export{m as __pageData,f as default}; diff --git a/assets/src_ranuts_binaryTree_index.md.9df1bc99.lean.js b/assets/src_ranuts_binaryTree_index.md.493e0878.lean.js similarity index 86% rename from assets/src_ranuts_binaryTree_index.md.9df1bc99.lean.js rename to assets/src_ranuts_binaryTree_index.md.493e0878.lean.js index e0c69479f..55d8337d3 100644 --- a/assets/src_ranuts_binaryTree_index.md.9df1bc99.lean.js +++ b/assets/src_ranuts_binaryTree_index.md.493e0878.lean.js @@ -1 +1 @@ -import{_ as a,o as e,c as r,N as i}from"./chunks/framework.6fe2e870.js";const t="/ran/assets/balanceTree.3d01d012.png",m=JSON.parse('{"title":"二叉树的定义","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/binaryTree/index.md","lastUpdated":1694276611000}'),h={name:"src/ranuts/binaryTree/index.md"},l=i("",36),o=[l];function n(d,s,c,u,b,q){return e(),r("div",null,o)}const f=a(h,[["render",n]]);export{m as __pageData,f as default}; +import{_ as a,o as e,c as r,N as i}from"./chunks/framework.6fe2e870.js";const t="/ran/assets/balanceTree.3d01d012.png",m=JSON.parse('{"title":"二叉树的定义","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/binaryTree/index.md","lastUpdated":1694419899000}'),h={name:"src/ranuts/binaryTree/index.md"},l=i("",36),o=[l];function n(d,s,c,u,b,q){return e(),r("div",null,o)}const f=a(h,[["render",n]]);export{m as __pageData,f as default}; diff --git a/assets/src_ranuts_bundler_index.md.605ec91e.js b/assets/src_ranuts_bundler_index.md.bf6a07c5.js similarity index 96% rename from assets/src_ranuts_bundler_index.md.605ec91e.js rename to assets/src_ranuts_bundler_index.md.bf6a07c5.js index b0cd2ee86..dc1a24d0b 100644 --- a/assets/src_ranuts_bundler_index.md.605ec91e.js +++ b/assets/src_ranuts_bundler_index.md.bf6a07c5.js @@ -1,4 +1,4 @@ -import{_ as n,o as s,c as a,N as e}from"./chunks/framework.6fe2e870.js";const l="/ran/assets/bundle.9ebe4911.png",m=JSON.parse('{"title":"Bundler","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/bundler/index.md","lastUpdated":1694276611000}'),t={name:"src/ranuts/bundler/index.md"},p=e(`

      Bundler

      Bundler的使用: 传入 options 参数

      function build(options: Options):Promise<Build> {
      +import{_ as n,o as s,c as a,N as e}from"./chunks/framework.6fe2e870.js";const l="/ran/assets/bundle.9ebe4911.png",m=JSON.parse('{"title":"Bundler","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/bundler/index.md","lastUpdated":1694419899000}'),t={name:"src/ranuts/bundler/index.md"},p=e(`

      Bundler

      Bundler的使用: 传入 options 参数

      function build(options: Options):Promise<Build> {
         const bundle = new Bundle({
           entry: options.input
         });
      diff --git a/assets/src_ranuts_bundler_index.md.605ec91e.lean.js b/assets/src_ranuts_bundler_index.md.bf6a07c5.lean.js
      similarity index 85%
      rename from assets/src_ranuts_bundler_index.md.605ec91e.lean.js
      rename to assets/src_ranuts_bundler_index.md.bf6a07c5.lean.js
      index cc69917ce..e73bd0251 100644
      --- a/assets/src_ranuts_bundler_index.md.605ec91e.lean.js
      +++ b/assets/src_ranuts_bundler_index.md.bf6a07c5.lean.js
      @@ -1 +1 @@
      -import{_ as n,o as s,c as a,N as e}from"./chunks/framework.6fe2e870.js";const l="/ran/assets/bundle.9ebe4911.png",m=JSON.parse('{"title":"Bundler","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/bundler/index.md","lastUpdated":1694276611000}'),t={name:"src/ranuts/bundler/index.md"},p=e("",5),o=[p];function r(c,d,i,b,u,_){return s(),a("div",null,o)}const y=n(t,[["render",r]]);export{m as __pageData,y as default};
      +import{_ as n,o as s,c as a,N as e}from"./chunks/framework.6fe2e870.js";const l="/ran/assets/bundle.9ebe4911.png",m=JSON.parse('{"title":"Bundler","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/bundler/index.md","lastUpdated":1694419899000}'),t={name:"src/ranuts/bundler/index.md"},p=e("",5),o=[p];function r(c,d,i,b,u,_){return s(),a("div",null,o)}const y=n(t,[["render",r]]);export{m as __pageData,y as default};
      diff --git a/assets/src_ranuts_file_index.md.177e1adc.js b/assets/src_ranuts_file_index.md.3ff8632d.js
      similarity index 96%
      rename from assets/src_ranuts_file_index.md.177e1adc.js
      rename to assets/src_ranuts_file_index.md.3ff8632d.js
      index e1224e831..74d156035 100644
      --- a/assets/src_ranuts_file_index.md.177e1adc.js
      +++ b/assets/src_ranuts_file_index.md.3ff8632d.js
      @@ -1 +1 @@
      -import{_ as t,o as e,c as a,N as d}from"./chunks/framework.6fe2e870.js";const b=JSON.parse('{"title":"文件操作函数","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/file/index.md","lastUpdated":1694276611000}'),r={name:"src/ranuts/file/index.md"},o=d('

      文件操作函数

      watchFile

      观察一个文件是否改变

      API

      Return

      • Promise
      参数说明类型描述
      status文件是否被改变booleantrue 文件改变 false 文件没变

      Options

      参数说明类型默认值
      path文件路径,需要监听的文件stringundefined
      interval监听文件改变的时间,单位毫秒。number20
      ',9),h=[o];function i(n,l,s,c,u,_){return e(),a("div",null,h)}const f=t(r,[["render",i]]);export{b as __pageData,f as default}; +import{_ as t,o as e,c as a,N as d}from"./chunks/framework.6fe2e870.js";const b=JSON.parse('{"title":"文件操作函数","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/file/index.md","lastUpdated":1694419899000}'),r={name:"src/ranuts/file/index.md"},o=d('

      文件操作函数

      watchFile

      观察一个文件是否改变

      API

      Return

      • Promise
      参数说明类型描述
      status文件是否被改变booleantrue 文件改变 false 文件没变

      Options

      参数说明类型默认值
      path文件路径,需要监听的文件stringundefined
      interval监听文件改变的时间,单位毫秒。number20
      ',9),h=[o];function i(n,l,s,c,u,_){return e(),a("div",null,h)}const f=t(r,[["render",i]]);export{b as __pageData,f as default}; diff --git a/assets/src_ranuts_file_index.md.177e1adc.lean.js b/assets/src_ranuts_file_index.md.3ff8632d.lean.js similarity index 84% rename from assets/src_ranuts_file_index.md.177e1adc.lean.js rename to assets/src_ranuts_file_index.md.3ff8632d.lean.js index 9a759dce7..fecb160d2 100644 --- a/assets/src_ranuts_file_index.md.177e1adc.lean.js +++ b/assets/src_ranuts_file_index.md.3ff8632d.lean.js @@ -1 +1 @@ -import{_ as t,o as e,c as a,N as d}from"./chunks/framework.6fe2e870.js";const b=JSON.parse('{"title":"文件操作函数","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/file/index.md","lastUpdated":1694276611000}'),r={name:"src/ranuts/file/index.md"},o=d("",9),h=[o];function i(n,l,s,c,u,_){return e(),a("div",null,h)}const f=t(r,[["render",i]]);export{b as __pageData,f as default}; +import{_ as t,o as e,c as a,N as d}from"./chunks/framework.6fe2e870.js";const b=JSON.parse('{"title":"文件操作函数","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/file/index.md","lastUpdated":1694419899000}'),r={name:"src/ranuts/file/index.md"},o=d("",9),h=[o];function i(n,l,s,c,u,_){return e(),a("div",null,h)}const f=t(r,[["render",i]]);export{b as __pageData,f as default}; diff --git a/assets/src_ranuts_index.md.238fdd9e.js b/assets/src_ranuts_index.md.f4ae71e9.js similarity index 81% rename from assets/src_ranuts_index.md.238fdd9e.js rename to assets/src_ranuts_index.md.f4ae71e9.js index 3a4a79a6b..2ecb1c08f 100644 --- a/assets/src_ranuts_index.md.238fdd9e.js +++ b/assets/src_ranuts_index.md.f4ae71e9.js @@ -1 +1 @@ -import{_ as e,o as t,c as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/index.md","lastUpdated":1694276611000}'),r={name:"src/ranuts/index.md"};function s(n,c,o,d,i,p){return t(),a("div")}const f=e(r,[["render",s]]);export{m as __pageData,f as default}; +import{_ as e,o as t,c as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/index.md","lastUpdated":1694419899000}'),r={name:"src/ranuts/index.md"};function s(n,c,o,d,i,p){return t(),a("div")}const f=e(r,[["render",s]]);export{m as __pageData,f as default}; diff --git a/assets/src_ranuts_index.md.238fdd9e.lean.js b/assets/src_ranuts_index.md.f4ae71e9.lean.js similarity index 81% rename from assets/src_ranuts_index.md.238fdd9e.lean.js rename to assets/src_ranuts_index.md.f4ae71e9.lean.js index 3a4a79a6b..2ecb1c08f 100644 --- a/assets/src_ranuts_index.md.238fdd9e.lean.js +++ b/assets/src_ranuts_index.md.f4ae71e9.lean.js @@ -1 +1 @@ -import{_ as e,o as t,c as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/index.md","lastUpdated":1694276611000}'),r={name:"src/ranuts/index.md"};function s(n,c,o,d,i,p){return t(),a("div")}const f=e(r,[["render",s]]);export{m as __pageData,f as default}; +import{_ as e,o as t,c as a}from"./chunks/framework.6fe2e870.js";const m=JSON.parse('{"title":"","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/index.md","lastUpdated":1694419899000}'),r={name:"src/ranuts/index.md"};function s(n,c,o,d,i,p){return t(),a("div")}const f=e(r,[["render",s]]);export{m as __pageData,f as default}; diff --git a/assets/src_ranuts_sort_bubble_index.md.3b551a0c.js b/assets/src_ranuts_sort_bubble_index.md.d2e4bc60.js similarity index 99% rename from assets/src_ranuts_sort_bubble_index.md.3b551a0c.js rename to assets/src_ranuts_sort_bubble_index.md.d2e4bc60.js index f10ebd1a8..e2dc775fc 100644 --- a/assets/src_ranuts_sort_bubble_index.md.3b551a0c.js +++ b/assets/src_ranuts_sort_bubble_index.md.d2e4bc60.js @@ -1,4 +1,4 @@ -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/bubble.287f9a70.gif",E=JSON.parse('{"title":"冒泡排序(Bubble Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/bubble/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/bubble/index.md"},e=l('

      冒泡排序(Bubble Sort)

      冒泡排序是一种简单的排序算法。它重复地走访过要排序的数列,一次比较两个元素,如果它们的顺序错误就把它们交换过来。走访数列的工作是重复地进行直到没有再需要交换,也就是说该数列已经排序完成。这个算法的名字由来是因为越小的元素会经由交换慢慢“浮”到数列的顶端。

      算法描述

      • 比较相邻的元素。如果第一个比第二个大,就交换它们两个;
      • 对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对,这样在最后的元素应该会是最大的数;
      • 针对所有的元素重复以上的步骤,除了最后一个;
      • 重复步骤 1~3,直到排序完成。

      动图演示

      冒泡排序

      代码演示

      js
      function bubbleSort(arr) {
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/bubble.287f9a70.gif",E=JSON.parse('{"title":"冒泡排序(Bubble Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/bubble/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/bubble/index.md"},e=l('

      冒泡排序(Bubble Sort)

      冒泡排序是一种简单的排序算法。它重复地走访过要排序的数列,一次比较两个元素,如果它们的顺序错误就把它们交换过来。走访数列的工作是重复地进行直到没有再需要交换,也就是说该数列已经排序完成。这个算法的名字由来是因为越小的元素会经由交换慢慢“浮”到数列的顶端。

      算法描述

      • 比较相邻的元素。如果第一个比第二个大,就交换它们两个;
      • 对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对,这样在最后的元素应该会是最大的数;
      • 针对所有的元素重复以上的步骤,除了最后一个;
      • 重复步骤 1~3,直到排序完成。

      动图演示

      冒泡排序

      代码演示

      js
      function bubbleSort(arr) {
         var len = arr.length
         for (var i = 0; i < len - 1; i++) {
           for (var j = 0; j < len - 1 - i; j++) {
      diff --git a/assets/src_ranuts_sort_bubble_index.md.3b551a0c.lean.js b/assets/src_ranuts_sort_bubble_index.md.d2e4bc60.lean.js
      similarity index 86%
      rename from assets/src_ranuts_sort_bubble_index.md.3b551a0c.lean.js
      rename to assets/src_ranuts_sort_bubble_index.md.d2e4bc60.lean.js
      index d170d3d53..5942c44da 100644
      --- a/assets/src_ranuts_sort_bubble_index.md.3b551a0c.lean.js
      +++ b/assets/src_ranuts_sort_bubble_index.md.d2e4bc60.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/bubble.287f9a70.gif",E=JSON.parse('{"title":"冒泡排序(Bubble Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/bubble/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/bubble/index.md"},e=l("",8),t=[e];function r(c,y,F,D,i,B){return a(),n("div",null,t)}const b=s(p,[["render",r]]);export{E as __pageData,b as default};
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/bubble.287f9a70.gif",E=JSON.parse('{"title":"冒泡排序(Bubble Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/bubble/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/bubble/index.md"},e=l("",8),t=[e];function r(c,y,F,D,i,B){return a(),n("div",null,t)}const b=s(p,[["render",r]]);export{E as __pageData,b as default};
      diff --git a/assets/src_ranuts_sort_bucket_index.md.755536c3.js b/assets/src_ranuts_sort_bucket_index.md.39a1b5ca.js
      similarity index 99%
      rename from assets/src_ranuts_sort_bucket_index.md.755536c3.js
      rename to assets/src_ranuts_sort_bucket_index.md.39a1b5ca.js
      index 083892d4b..b4ec310f5 100644
      --- a/assets/src_ranuts_sort_bucket_index.md.755536c3.js
      +++ b/assets/src_ranuts_sort_bucket_index.md.39a1b5ca.js
      @@ -1,4 +1,4 @@
      -import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"桶排序 (Bucket Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/bucket/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/bucket/index.md"},o=l(`

      桶排序 (Bucket Sort)

      高效与否的关键在于这个分桶函数。将数据分到有限数量的桶里,每个桶再分别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排)。

      算法描述

      • 设置一个定量的数组当作空桶;
      • 遍历输入数据,并且把数据一个一个放到对应的桶里去;
      • 对每个不是空的桶进行排序;
      • 从不是空的桶里把排好序的数据拼接起来。

      代码演示

      ts
      const count = (list: Array<number>, max: number = 100): Array<number> => {
      +import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"桶排序 (Bucket Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/bucket/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/bucket/index.md"},o=l(`

      桶排序 (Bucket Sort)

      高效与否的关键在于这个分桶函数。将数据分到有限数量的桶里,每个桶再分别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排)。

      算法描述

      • 设置一个定量的数组当作空桶;
      • 遍历输入数据,并且把数据一个一个放到对应的桶里去;
      • 对每个不是空的桶进行排序;
      • 从不是空的桶里把排好序的数据拼接起来。

      代码演示

      ts
      const count = (list: Array<number>, max: number = 100): Array<number> => {
         const countList = new Array(max + 1)
         for (let i = 0; i < list.length; i++) {
           if (!countList[list[i]]) {
      diff --git a/assets/src_ranuts_sort_bucket_index.md.755536c3.lean.js b/assets/src_ranuts_sort_bucket_index.md.39a1b5ca.lean.js
      similarity index 85%
      rename from assets/src_ranuts_sort_bucket_index.md.755536c3.lean.js
      rename to assets/src_ranuts_sort_bucket_index.md.39a1b5ca.lean.js
      index cf655c363..d393ab2cb 100644
      --- a/assets/src_ranuts_sort_bucket_index.md.755536c3.lean.js
      +++ b/assets/src_ranuts_sort_bucket_index.md.39a1b5ca.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"桶排序 (Bucket Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/bucket/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/bucket/index.md"},o=l("",8),t=[o];function e(c,r,y,F,D,B){return n(),a("div",null,t)}const E=s(p,[["render",e]]);export{A as __pageData,E as default};
      +import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"桶排序 (Bucket Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/bucket/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/bucket/index.md"},o=l("",8),t=[o];function e(c,r,y,F,D,B){return n(),a("div",null,t)}const E=s(p,[["render",e]]);export{A as __pageData,E as default};
      diff --git a/assets/src_ranuts_sort_count_index.md.52dd4e7c.js b/assets/src_ranuts_sort_count_index.md.bcf7daeb.js
      similarity index 99%
      rename from assets/src_ranuts_sort_count_index.md.52dd4e7c.js
      rename to assets/src_ranuts_sort_count_index.md.bcf7daeb.js
      index e41f38456..b62008a3b 100644
      --- a/assets/src_ranuts_sort_count_index.md.52dd4e7c.js
      +++ b/assets/src_ranuts_sort_count_index.md.bcf7daeb.js
      @@ -1,4 +1,4 @@
      -import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/count.9e7ca98e.gif",E=JSON.parse('{"title":"计数排序( Count Sort )","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/count/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/count/index.md"},t=l('

      计数排序( Count Sort )

      计数排序(counting sort)就是一种牺牲内存空间来换取低时间复杂度的排序算法,同时它也是一种不基于比较的算法。这里的不基于比较指的是数组元素之间不存在比较大小的排序算法,我们知道,用分治法来解决排序问题最快也只能使算法的时间复杂度接近 Θ(nlogn),即基于比较的时间复杂度存在下界 Ω(nlog⁡n),而不基于比较的排序算法可以突破这一下界。

      算法描述

      • 找出待排序的数组中最大和最小的元素;
      • 统计数组中每个值为 i 的元素出现的次数,存入数组 C 的第 i 项;
      • 对所有的计数累加(从 C 中的第一个元素开始,每一项和前一项相加);
      • 反向填充目标数组:将每个元素 i 放在新数组的第 C(i)项,每放一个元素就将 C(i)减去 1。

      动图演示

      计数排序

      代码演示

      ts
      const getMax = (list: Array<number>) => {
      +import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/count.9e7ca98e.gif",E=JSON.parse('{"title":"计数排序( Count Sort )","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/count/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/count/index.md"},t=l('

      计数排序( Count Sort )

      计数排序(counting sort)就是一种牺牲内存空间来换取低时间复杂度的排序算法,同时它也是一种不基于比较的算法。这里的不基于比较指的是数组元素之间不存在比较大小的排序算法,我们知道,用分治法来解决排序问题最快也只能使算法的时间复杂度接近 Θ(nlogn),即基于比较的时间复杂度存在下界 Ω(nlog⁡n),而不基于比较的排序算法可以突破这一下界。

      算法描述

      • 找出待排序的数组中最大和最小的元素;
      • 统计数组中每个值为 i 的元素出现的次数,存入数组 C 的第 i 项;
      • 对所有的计数累加(从 C 中的第一个元素开始,每一项和前一项相加);
      • 反向填充目标数组:将每个元素 i 放在新数组的第 C(i)项,每放一个元素就将 C(i)减去 1。

      动图演示

      计数排序

      代码演示

      ts
      const getMax = (list: Array<number>) => {
         let max = list[0]
         for (let i = 0; i < list.length; i++) {
           if (max < list[i]) {
      diff --git a/assets/src_ranuts_sort_count_index.md.52dd4e7c.lean.js b/assets/src_ranuts_sort_count_index.md.bcf7daeb.lean.js
      similarity index 86%
      rename from assets/src_ranuts_sort_count_index.md.52dd4e7c.lean.js
      rename to assets/src_ranuts_sort_count_index.md.bcf7daeb.lean.js
      index 51ca3d867..2a5d6b8f7 100644
      --- a/assets/src_ranuts_sort_count_index.md.52dd4e7c.lean.js
      +++ b/assets/src_ranuts_sort_count_index.md.bcf7daeb.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/count.9e7ca98e.gif",E=JSON.parse('{"title":"计数排序( Count Sort )","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/count/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/count/index.md"},t=l("",10),e=[t];function c(r,y,F,D,i,B){return n(),a("div",null,e)}const f=s(p,[["render",c]]);export{E as __pageData,f as default};
      +import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/count.9e7ca98e.gif",E=JSON.parse('{"title":"计数排序( Count Sort )","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/count/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/count/index.md"},t=l("",10),e=[t];function c(r,y,F,D,i,B){return n(),a("div",null,e)}const f=s(p,[["render",c]]);export{E as __pageData,f as default};
      diff --git a/assets/src_ranuts_sort_heap_index.md.c3678584.js b/assets/src_ranuts_sort_heap_index.md.ed19bcdb.js
      similarity index 99%
      rename from assets/src_ranuts_sort_heap_index.md.c3678584.js
      rename to assets/src_ranuts_sort_heap_index.md.ed19bcdb.js
      index 19aa0ff5a..022aa3068 100644
      --- a/assets/src_ranuts_sort_heap_index.md.c3678584.js
      +++ b/assets/src_ranuts_sort_heap_index.md.ed19bcdb.js
      @@ -1,4 +1,4 @@
      -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/heap.0c214e72.gif",E=JSON.parse('{"title":"堆排序(Heap Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/heap/index.md","lastUpdated":1694276611000}'),o={name:"src/ranuts/sort/heap/index.md"},t=l('

      堆排序(Heap Sort)

      堆排序(Heapsort)是指利用堆这种数据结构所设计的一种排序算法。堆积是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父节点。

      算法描述

      • 将初始待排序关键字序列(R1,R2….Rn)构建成大顶堆,此堆为初始的无序区;
      • 将堆顶元素 R[1]与最后一个元素 R[n]交换,此时得到新的无序区(R1,R2,……Rn-1)和新的有序区(Rn),且满足 R[1,2…n-1]<=R[n];
      • 由于交换后新的堆顶 R[1]可能违反堆的性质,因此需要对当前无序区(R1,R2,……Rn-1)调整为新堆,然后再次将 R[1]与无序区最后一个元素交换,得到新的无序区(R1,R2….Rn-2)和新的有序区(Rn-1,Rn)。不断重复此过程直到有序区的元素个数为 n-1,则整个排序过程完成。
      • 升序用大根堆,降序用小根堆

      动图演示

      堆排序

      代码演示

      ts
      class Heap {
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/heap.0c214e72.gif",E=JSON.parse('{"title":"堆排序(Heap Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/heap/index.md","lastUpdated":1694419899000}'),o={name:"src/ranuts/sort/heap/index.md"},t=l('

      堆排序(Heap Sort)

      堆排序(Heapsort)是指利用堆这种数据结构所设计的一种排序算法。堆积是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父节点。

      算法描述

      • 将初始待排序关键字序列(R1,R2….Rn)构建成大顶堆,此堆为初始的无序区;
      • 将堆顶元素 R[1]与最后一个元素 R[n]交换,此时得到新的无序区(R1,R2,……Rn-1)和新的有序区(Rn),且满足 R[1,2…n-1]<=R[n];
      • 由于交换后新的堆顶 R[1]可能违反堆的性质,因此需要对当前无序区(R1,R2,……Rn-1)调整为新堆,然后再次将 R[1]与无序区最后一个元素交换,得到新的无序区(R1,R2….Rn-2)和新的有序区(Rn-1,Rn)。不断重复此过程直到有序区的元素个数为 n-1,则整个排序过程完成。
      • 升序用大根堆,降序用小根堆

      动图演示

      堆排序

      代码演示

      ts
      class Heap {
         arr: Array<number>
         size: number
         constructor(arr: Array<number>) {
      diff --git a/assets/src_ranuts_sort_heap_index.md.c3678584.lean.js b/assets/src_ranuts_sort_heap_index.md.ed19bcdb.lean.js
      similarity index 86%
      rename from assets/src_ranuts_sort_heap_index.md.c3678584.lean.js
      rename to assets/src_ranuts_sort_heap_index.md.ed19bcdb.lean.js
      index 8f75e55ae..90976a6cc 100644
      --- a/assets/src_ranuts_sort_heap_index.md.c3678584.lean.js
      +++ b/assets/src_ranuts_sort_heap_index.md.ed19bcdb.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/heap.0c214e72.gif",E=JSON.parse('{"title":"堆排序(Heap Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/heap/index.md","lastUpdated":1694276611000}'),o={name:"src/ranuts/sort/heap/index.md"},t=l("",8),e=[t];function r(c,y,F,D,B,i){return a(),n("div",null,e)}const h=s(o,[["render",r]]);export{E as __pageData,h as default};
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const p="/ran/assets/heap.0c214e72.gif",E=JSON.parse('{"title":"堆排序(Heap Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/heap/index.md","lastUpdated":1694419899000}'),o={name:"src/ranuts/sort/heap/index.md"},t=l("",8),e=[t];function r(c,y,F,D,B,i){return a(),n("div",null,e)}const h=s(o,[["render",r]]);export{E as __pageData,h as default};
      diff --git a/assets/src_ranuts_sort_index.md.b7c48f70.js b/assets/src_ranuts_sort_index.md.fb0f8a4a.js
      similarity index 96%
      rename from assets/src_ranuts_sort_index.md.b7c48f70.js
      rename to assets/src_ranuts_sort_index.md.fb0f8a4a.js
      index 1a8cc21a3..ef3c4d670 100644
      --- a/assets/src_ranuts_sort_index.md.b7c48f70.js
      +++ b/assets/src_ranuts_sort_index.md.fb0f8a4a.js
      @@ -1 +1 @@
      -import{_ as t,o as a,c as s,N as e}from"./chunks/framework.6fe2e870.js";const r="/ran/assets/sort.0e02a465.png",n="/ran/assets/complexity.aded73a0.png",b=JSON.parse('{"title":"十大经典排序","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/index.md","lastUpdated":1694276611000}'),o={name:"src/ranuts/sort/index.md"},i=e('

      十大经典排序

      十种常见排序算法可以分为两大类:

      • 比较类排序:通过比较来决定元素间的相对次序,由于其时间复杂度不能突破 O(nlogn),因此也称为非线性时间比较类排序。
      • 非比较类排序:不通过比较来决定元素间的相对次序,它可以突破基于比较排序的时间下界,以线性时间运行,因此也称为线性时间非比较类排序。 排序分类 0.2 算法复杂度

      算法复杂度 0.3 相关概念

      • 稳定:如果 a 原本在 b 前面,而 a=b,排序之后 a 仍然在 b 的前面。
      • 不稳定:如果 a 原本在 b 的前面,而 a=b,排序之后 a 可能会出现在 b 的后面。
      • 时间复杂度:对排序数据的总的操作次数。反映当 n 变化时,操作次数呈现什么规律。
      • 空间复杂度:是指算法在计算机内执行时所需存储空间的度量,它也是数据规模 n 的函数。
      ',5),l=[i];function _(c,d,p,m,u,g){return a(),s("div",null,l)}const f=t(o,[["render",_]]);export{b as __pageData,f as default}; +import{_ as t,o as a,c as s,N as e}from"./chunks/framework.6fe2e870.js";const r="/ran/assets/sort.0e02a465.png",n="/ran/assets/complexity.aded73a0.png",b=JSON.parse('{"title":"十大经典排序","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/index.md","lastUpdated":1694419899000}'),o={name:"src/ranuts/sort/index.md"},i=e('

      十大经典排序

      十种常见排序算法可以分为两大类:

      • 比较类排序:通过比较来决定元素间的相对次序,由于其时间复杂度不能突破 O(nlogn),因此也称为非线性时间比较类排序。
      • 非比较类排序:不通过比较来决定元素间的相对次序,它可以突破基于比较排序的时间下界,以线性时间运行,因此也称为线性时间非比较类排序。 排序分类 0.2 算法复杂度

      算法复杂度 0.3 相关概念

      • 稳定:如果 a 原本在 b 前面,而 a=b,排序之后 a 仍然在 b 的前面。
      • 不稳定:如果 a 原本在 b 的前面,而 a=b,排序之后 a 可能会出现在 b 的后面。
      • 时间复杂度:对排序数据的总的操作次数。反映当 n 变化时,操作次数呈现什么规律。
      • 空间复杂度:是指算法在计算机内执行时所需存储空间的度量,它也是数据规模 n 的函数。
      ',5),l=[i];function _(c,d,p,m,u,g){return a(),s("div",null,l)}const f=t(o,[["render",_]]);export{b as __pageData,f as default}; diff --git a/assets/src_ranuts_sort_index.md.b7c48f70.lean.js b/assets/src_ranuts_sort_index.md.fb0f8a4a.lean.js similarity index 86% rename from assets/src_ranuts_sort_index.md.b7c48f70.lean.js rename to assets/src_ranuts_sort_index.md.fb0f8a4a.lean.js index bc4e93729..5d1032845 100644 --- a/assets/src_ranuts_sort_index.md.b7c48f70.lean.js +++ b/assets/src_ranuts_sort_index.md.fb0f8a4a.lean.js @@ -1 +1 @@ -import{_ as t,o as a,c as s,N as e}from"./chunks/framework.6fe2e870.js";const r="/ran/assets/sort.0e02a465.png",n="/ran/assets/complexity.aded73a0.png",b=JSON.parse('{"title":"十大经典排序","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/index.md","lastUpdated":1694276611000}'),o={name:"src/ranuts/sort/index.md"},i=e("",5),l=[i];function _(c,d,p,m,u,g){return a(),s("div",null,l)}const f=t(o,[["render",_]]);export{b as __pageData,f as default}; +import{_ as t,o as a,c as s,N as e}from"./chunks/framework.6fe2e870.js";const r="/ran/assets/sort.0e02a465.png",n="/ran/assets/complexity.aded73a0.png",b=JSON.parse('{"title":"十大经典排序","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/index.md","lastUpdated":1694419899000}'),o={name:"src/ranuts/sort/index.md"},i=e("",5),l=[i];function _(c,d,p,m,u,g){return a(),s("div",null,l)}const f=t(o,[["render",_]]);export{b as __pageData,f as default}; diff --git a/assets/src_ranuts_sort_insert_index.md.de43fe03.js b/assets/src_ranuts_sort_insert_index.md.032011a9.js similarity index 99% rename from assets/src_ranuts_sort_insert_index.md.de43fe03.js rename to assets/src_ranuts_sort_insert_index.md.032011a9.js index ade44178f..66c5d2805 100644 --- a/assets/src_ranuts_sort_insert_index.md.de43fe03.js +++ b/assets/src_ranuts_sort_insert_index.md.032011a9.js @@ -1,4 +1,4 @@ -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/insert.4cd822c8.gif",A=JSON.parse('{"title":"插入排序(Insert Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/insert/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/insert/index.md"},t=l('

      插入排序(Insert Sort)

      表现稳定的排序算法,因为无论什么数据进去都是 O(n2)的时间复杂度,所以用到它的时候,数据规模越小越好。优点是不占用额外的内存空间。工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入。

      算法描述

      • 从第一个元素开始,该元素可以认为已经被排序;
      • 取出下一个元素,在已经排序的元素序列中从后向前扫描;
      • 如果该元素(已排序)大于新元素,将该元素移到下一位置;
      • 重复步骤 3,直到找到已排序的元素小于或者等于新元素的位置;
      • 将新元素插入到该位置后;
      • 重复步骤 2~5。

      动图演示

      插入排序

      代码演示

      ts
      const insert = (list: Array<string>): Array<string> => {
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/insert.4cd822c8.gif",A=JSON.parse('{"title":"插入排序(Insert Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/insert/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/insert/index.md"},t=l('

      插入排序(Insert Sort)

      表现稳定的排序算法,因为无论什么数据进去都是 O(n2)的时间复杂度,所以用到它的时候,数据规模越小越好。优点是不占用额外的内存空间。工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入。

      算法描述

      • 从第一个元素开始,该元素可以认为已经被排序;
      • 取出下一个元素,在已经排序的元素序列中从后向前扫描;
      • 如果该元素(已排序)大于新元素,将该元素移到下一位置;
      • 重复步骤 3,直到找到已排序的元素小于或者等于新元素的位置;
      • 将新元素插入到该位置后;
      • 重复步骤 2~5。

      动图演示

      插入排序

      代码演示

      ts
      const insert = (list: Array<string>): Array<string> => {
         const { length } = list
         for (let i = 1; i < length; i++) {
           let preIndex = i - 1
      diff --git a/assets/src_ranuts_sort_insert_index.md.de43fe03.lean.js b/assets/src_ranuts_sort_insert_index.md.032011a9.lean.js
      similarity index 86%
      rename from assets/src_ranuts_sort_insert_index.md.de43fe03.lean.js
      rename to assets/src_ranuts_sort_insert_index.md.032011a9.lean.js
      index 75043e428..816123869 100644
      --- a/assets/src_ranuts_sort_insert_index.md.de43fe03.lean.js
      +++ b/assets/src_ranuts_sort_insert_index.md.032011a9.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/insert.4cd822c8.gif",A=JSON.parse('{"title":"插入排序(Insert Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/insert/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/insert/index.md"},t=l("",10),e=[t];function r(c,y,F,D,i,B){return a(),n("div",null,e)}const E=s(p,[["render",r]]);export{A as __pageData,E as default};
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/insert.4cd822c8.gif",A=JSON.parse('{"title":"插入排序(Insert Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/insert/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/insert/index.md"},t=l("",10),e=[t];function r(c,y,F,D,i,B){return a(),n("div",null,e)}const E=s(p,[["render",r]]);export{A as __pageData,E as default};
      diff --git a/assets/src_ranuts_sort_merge_index.md.46e8a482.js b/assets/src_ranuts_sort_merge_index.md.d6dbf578.js
      similarity index 99%
      rename from assets/src_ranuts_sort_merge_index.md.46e8a482.js
      rename to assets/src_ranuts_sort_merge_index.md.d6dbf578.js
      index 2d383fd1c..c5e0fbdaa 100644
      --- a/assets/src_ranuts_sort_merge_index.md.46e8a482.js
      +++ b/assets/src_ranuts_sort_merge_index.md.d6dbf578.js
      @@ -1,4 +1,4 @@
      -import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/merge.7613b4a8.gif",E=JSON.parse('{"title":"归并排序(Merge Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/merge/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/merge/index.md"},t=l('

      归并排序(Merge Sort)

      归并排序是建立在归并操作上的一种有效的排序算法。该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为 2-路归并。

      算法描述

      • 把长度为 n 的输入序列分成两个长度为 n/2 的子序列;
      • 对这两个子序列分别采用归并排序;
      • 将两个排序好的子序列合并成一个最终的排序序列。

      动图演示

      归并排序

      代码演示

      ts
      const combine = (left: Array<number>, right: Array<number>) => {
      +import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/merge.7613b4a8.gif",E=JSON.parse('{"title":"归并排序(Merge Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/merge/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/merge/index.md"},t=l('

      归并排序(Merge Sort)

      归并排序是建立在归并操作上的一种有效的排序算法。该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为 2-路归并。

      算法描述

      • 把长度为 n 的输入序列分成两个长度为 n/2 的子序列;
      • 对这两个子序列分别采用归并排序;
      • 将两个排序好的子序列合并成一个最终的排序序列。

      动图演示

      归并排序

      代码演示

      ts
      const combine = (left: Array<number>, right: Array<number>) => {
         const list: Array<number> = []
         while (left.length > 0 && right.length > 0) {
           if (left[0] <= right[0]) {
      diff --git a/assets/src_ranuts_sort_merge_index.md.46e8a482.lean.js b/assets/src_ranuts_sort_merge_index.md.d6dbf578.lean.js
      similarity index 86%
      rename from assets/src_ranuts_sort_merge_index.md.46e8a482.lean.js
      rename to assets/src_ranuts_sort_merge_index.md.d6dbf578.lean.js
      index afdd210c7..92b82f2f3 100644
      --- a/assets/src_ranuts_sort_merge_index.md.46e8a482.lean.js
      +++ b/assets/src_ranuts_sort_merge_index.md.d6dbf578.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/merge.7613b4a8.gif",E=JSON.parse('{"title":"归并排序(Merge Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/merge/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/merge/index.md"},t=l("",10),e=[t];function c(r,y,F,D,i,B){return n(),a("div",null,e)}const f=s(p,[["render",c]]);export{E as __pageData,f as default};
      +import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/merge.7613b4a8.gif",E=JSON.parse('{"title":"归并排序(Merge Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/merge/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/merge/index.md"},t=l("",10),e=[t];function c(r,y,F,D,i,B){return n(),a("div",null,e)}const f=s(p,[["render",c]]);export{E as __pageData,f as default};
      diff --git a/assets/src_ranuts_sort_quick_index.md.9ea0bd8c.js b/assets/src_ranuts_sort_quick_index.md.3d358bda.js
      similarity index 99%
      rename from assets/src_ranuts_sort_quick_index.md.9ea0bd8c.js
      rename to assets/src_ranuts_sort_quick_index.md.3d358bda.js
      index d35f28d41..904aef841 100644
      --- a/assets/src_ranuts_sort_quick_index.md.9ea0bd8c.js
      +++ b/assets/src_ranuts_sort_quick_index.md.3d358bda.js
      @@ -1,4 +1,4 @@
      -import{_ as s,o as l,c as n,N as a}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/quick.e1c33863.gif",f=JSON.parse('{"title":"快速排序(Quick Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/quick/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/quick/index.md"},t=a('

      快速排序(Quick Sort)

      快速排序的基本思想:通过一趟排序将待排记录分隔成独立的两部分,其中一部分记录的关键字均比另一部分的关键字小,则可分别对这两部分记录继续进行排序,以达到整个序列有序。

      算法描述

      快速排序使用分治法来把一个串(list)分为两个子串(sub-lists)。具体算法描述如下:

      • 从数列中挑出一个元素,称为 “基准”(pivot);
      • 重新排序数列,所有元素比基准值小的摆放在基准前面,所有元素比基准值大的摆在基准的后面(相同的数可以到任一边)。在这个分区退出之后,该基准就处于数列的中间位置。这个称为分区(partition)操作;
      • 递归地(recursive)把小于基准值元素的子数列和大于基准值元素的子数列排序。

      动图演示

      快速排序

      代码演示

      ts
      /**
      +import{_ as s,o as l,c as n,N as a}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/quick.e1c33863.gif",f=JSON.parse('{"title":"快速排序(Quick Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/quick/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/quick/index.md"},t=a('

      快速排序(Quick Sort)

      快速排序的基本思想:通过一趟排序将待排记录分隔成独立的两部分,其中一部分记录的关键字均比另一部分的关键字小,则可分别对这两部分记录继续进行排序,以达到整个序列有序。

      算法描述

      快速排序使用分治法来把一个串(list)分为两个子串(sub-lists)。具体算法描述如下:

      • 从数列中挑出一个元素,称为 “基准”(pivot);
      • 重新排序数列,所有元素比基准值小的摆放在基准前面,所有元素比基准值大的摆在基准的后面(相同的数可以到任一边)。在这个分区退出之后,该基准就处于数列的中间位置。这个称为分区(partition)操作;
      • 递归地(recursive)把小于基准值元素的子数列和大于基准值元素的子数列排序。

      动图演示

      快速排序

      代码演示

      ts
      /**
        * @description: 设置基准值pivot
        * @param {Array} list
        * @param {number} left
      diff --git a/assets/src_ranuts_sort_quick_index.md.9ea0bd8c.lean.js b/assets/src_ranuts_sort_quick_index.md.3d358bda.lean.js
      similarity index 86%
      rename from assets/src_ranuts_sort_quick_index.md.9ea0bd8c.lean.js
      rename to assets/src_ranuts_sort_quick_index.md.3d358bda.lean.js
      index 3fa7cce49..982d085d8 100644
      --- a/assets/src_ranuts_sort_quick_index.md.9ea0bd8c.lean.js
      +++ b/assets/src_ranuts_sort_quick_index.md.3d358bda.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as l,c as n,N as a}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/quick.e1c33863.gif",f=JSON.parse('{"title":"快速排序(Quick Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/quick/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/quick/index.md"},t=a("",9),e=[t];function c(r,y,i,F,D,B){return l(),n("div",null,e)}const A=s(p,[["render",c]]);export{f as __pageData,A as default};
      +import{_ as s,o as l,c as n,N as a}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/quick.e1c33863.gif",f=JSON.parse('{"title":"快速排序(Quick Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/quick/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/quick/index.md"},t=a("",9),e=[t];function c(r,y,i,F,D,B){return l(),n("div",null,e)}const A=s(p,[["render",c]]);export{f as __pageData,A as default};
      diff --git a/assets/src_ranuts_sort_radix_index.md.45ded274.js b/assets/src_ranuts_sort_radix_index.md.55bb0018.js
      similarity index 99%
      rename from assets/src_ranuts_sort_radix_index.md.45ded274.js
      rename to assets/src_ranuts_sort_radix_index.md.55bb0018.js
      index 38bbbdac7..71286f9ef 100644
      --- a/assets/src_ranuts_sort_radix_index.md.45ded274.js
      +++ b/assets/src_ranuts_sort_radix_index.md.55bb0018.js
      @@ -1,4 +1,4 @@
      -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/radix.101a451a.gif",E=JSON.parse('{"title":"基数排序(Radix Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/radix/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/radix/index.md"},t=l('

      基数排序(Radix Sort)

      基数排序是按照低位先排序,然后收集;再按照高位排序,然后再收集;依次类推,直到最高位。有时候有些属性是有优先级顺序的,先按低优先级排序,再按高优先级排序。最后的次序就是高优先级高的在前,高优先级相同的低优先级高的在前。桶排序扩展,类似于指定桶排序按位数排序规则,同时能利用计数排序适用于小范围数的特点。

      算法描述

      • 取得数组中的最大数,并取得位数;
      • arr 为原始数组,从最低位开始取每个位组成 radix 数组;
      • 对 radix 进行计数排序(利用计数排序适用于小范围数的特点);

      动图演示

      基数排序

      代码演示

      ts
      const getMax = (list: Array<number>) => {
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/radix.101a451a.gif",E=JSON.parse('{"title":"基数排序(Radix Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/radix/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/radix/index.md"},t=l('

      基数排序(Radix Sort)

      基数排序是按照低位先排序,然后收集;再按照高位排序,然后再收集;依次类推,直到最高位。有时候有些属性是有优先级顺序的,先按低优先级排序,再按高优先级排序。最后的次序就是高优先级高的在前,高优先级相同的低优先级高的在前。桶排序扩展,类似于指定桶排序按位数排序规则,同时能利用计数排序适用于小范围数的特点。

      算法描述

      • 取得数组中的最大数,并取得位数;
      • arr 为原始数组,从最低位开始取每个位组成 radix 数组;
      • 对 radix 进行计数排序(利用计数排序适用于小范围数的特点);

      动图演示

      基数排序

      代码演示

      ts
      const getMax = (list: Array<number>) => {
         let max = list[0]
         for (let i = 0; i < list.length; i++) {
           if (max < list[i]) {
      diff --git a/assets/src_ranuts_sort_radix_index.md.45ded274.lean.js b/assets/src_ranuts_sort_radix_index.md.55bb0018.lean.js
      similarity index 86%
      rename from assets/src_ranuts_sort_radix_index.md.45ded274.lean.js
      rename to assets/src_ranuts_sort_radix_index.md.55bb0018.lean.js
      index 16a34e4ff..1c7b5b46e 100644
      --- a/assets/src_ranuts_sort_radix_index.md.45ded274.lean.js
      +++ b/assets/src_ranuts_sort_radix_index.md.55bb0018.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/radix.101a451a.gif",E=JSON.parse('{"title":"基数排序(Radix Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/radix/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/radix/index.md"},t=l("",11),e=[t];function c(r,y,F,D,i,B){return a(),n("div",null,e)}const d=s(p,[["render",c]]);export{E as __pageData,d as default};
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/radix.101a451a.gif",E=JSON.parse('{"title":"基数排序(Radix Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/radix/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/radix/index.md"},t=l("",11),e=[t];function c(r,y,F,D,i,B){return a(),n("div",null,e)}const d=s(p,[["render",c]]);export{E as __pageData,d as default};
      diff --git a/assets/src_ranuts_sort_select_index.md.f91454eb.js b/assets/src_ranuts_sort_select_index.md.cb9012a3.js
      similarity index 99%
      rename from assets/src_ranuts_sort_select_index.md.f91454eb.js
      rename to assets/src_ranuts_sort_select_index.md.cb9012a3.js
      index 428a9b969..c561a8fb8 100644
      --- a/assets/src_ranuts_sort_select_index.md.f91454eb.js
      +++ b/assets/src_ranuts_sort_select_index.md.cb9012a3.js
      @@ -1,4 +1,4 @@
      -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/select.5aa51d89.gif",E=JSON.parse('{"title":"选择排序(Selection Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/select/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/select/index.md"},e=l('

      选择排序(Selection Sort)

      选择排序(Selection-sort)是一种简单直观的排序算法。它的工作原理:首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。以此类推,直到所有元素均排序完毕。

      算法描述

      n 个记录的直接选择排序可经过 n-1 趟直接选择排序得到有序结果。具体算法描述如下:

      • 初始状态:无序区为 R[1..n],有序区为空;
      • 第 i 趟排序(i=1,2,3…n-1)开始时,当前有序区和无序区分别为 R[1..i-1]和 R(i..n)。该趟排序从当前无序区中-选出关键字最小的记录 R[k],将它与无序区的第 1 个记录 R 交换,使 R[1..i]和 R[i+1..n)分别变为记录个数增加 1 个的新有序区和记录个数减少 1 个的新无序区;
      • n-1 趟结束,数组有序化了。

      动图演示

      选择排序

      代码实现

      js
      function selectionSort(arr) {
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/select.5aa51d89.gif",E=JSON.parse('{"title":"选择排序(Selection Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/select/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/select/index.md"},e=l('

      选择排序(Selection Sort)

      选择排序(Selection-sort)是一种简单直观的排序算法。它的工作原理:首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。以此类推,直到所有元素均排序完毕。

      算法描述

      n 个记录的直接选择排序可经过 n-1 趟直接选择排序得到有序结果。具体算法描述如下:

      • 初始状态:无序区为 R[1..n],有序区为空;
      • 第 i 趟排序(i=1,2,3…n-1)开始时,当前有序区和无序区分别为 R[1..i-1]和 R(i..n)。该趟排序从当前无序区中-选出关键字最小的记录 R[k],将它与无序区的第 1 个记录 R 交换,使 R[1..i]和 R[i+1..n)分别变为记录个数增加 1 个的新有序区和记录个数减少 1 个的新无序区;
      • n-1 趟结束,数组有序化了。

      动图演示

      选择排序

      代码实现

      js
      function selectionSort(arr) {
         var len = arr.length
         var minIndex, temp
         for (var i = 0; i < len - 1; i++) {
      diff --git a/assets/src_ranuts_sort_select_index.md.f91454eb.lean.js b/assets/src_ranuts_sort_select_index.md.cb9012a3.lean.js
      similarity index 86%
      rename from assets/src_ranuts_sort_select_index.md.f91454eb.lean.js
      rename to assets/src_ranuts_sort_select_index.md.cb9012a3.lean.js
      index fffde6444..399e9a312 100644
      --- a/assets/src_ranuts_sort_select_index.md.f91454eb.lean.js
      +++ b/assets/src_ranuts_sort_select_index.md.cb9012a3.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/select.5aa51d89.gif",E=JSON.parse('{"title":"选择排序(Selection Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/select/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/select/index.md"},e=l("",11),t=[e];function r(c,y,F,i,D,B){return a(),n("div",null,t)}const d=s(p,[["render",r]]);export{E as __pageData,d as default};
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/select.5aa51d89.gif",E=JSON.parse('{"title":"选择排序(Selection Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/select/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/select/index.md"},e=l("",11),t=[e];function r(c,y,F,i,D,B){return a(),n("div",null,t)}const d=s(p,[["render",r]]);export{E as __pageData,d as default};
      diff --git a/assets/src_ranuts_sort_shell_index.md.fb9ca14b.js b/assets/src_ranuts_sort_shell_index.md.11a9ac10.js
      similarity index 99%
      rename from assets/src_ranuts_sort_shell_index.md.fb9ca14b.js
      rename to assets/src_ranuts_sort_shell_index.md.11a9ac10.js
      index 25412dc38..34e74bca5 100644
      --- a/assets/src_ranuts_sort_shell_index.md.fb9ca14b.js
      +++ b/assets/src_ranuts_sort_shell_index.md.11a9ac10.js
      @@ -1,4 +1,4 @@
      -import{_ as s,o as a,c as l,N as n}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/shell.7f63988c.gif",E=JSON.parse('{"title":"希尔排序(Shell Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/shell/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/shell/index.md"},t=n('

      希尔排序(Shell Sort)

      1959 年 Shell 发明,第一个突破 O(n2)的排序算法,是简单插入排序的改进版。它与插入排序的不同之处在于,它会优先比较距离较远的元素。希尔排序又叫缩小增量排序。

      算法描述

      先将整个待排序的记录序列分割成为若干子序列分别进行直接插入排序,具体算法描述:

      • 选择一个增量序列 t1,t2,…,tk,其中 ti>tj,tk=1;
      • 按增量序列个数 k,对序列进行 k 趟排序;
      • 每趟排序,根据对应的增量 ti,将待排序列分割成若干长度为 m 的子序列,分别对各子表进行直接插入排序。仅增量因子为 1 时,整个序列作为一个表来处理,表长度即为整个序列的长度。

      动图演示

      希尔排序

      代码实现

      js
      /**
      +import{_ as s,o as a,c as l,N as n}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/shell.7f63988c.gif",E=JSON.parse('{"title":"希尔排序(Shell Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/shell/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/shell/index.md"},t=n('

      希尔排序(Shell Sort)

      1959 年 Shell 发明,第一个突破 O(n2)的排序算法,是简单插入排序的改进版。它与插入排序的不同之处在于,它会优先比较距离较远的元素。希尔排序又叫缩小增量排序。

      算法描述

      先将整个待排序的记录序列分割成为若干子序列分别进行直接插入排序,具体算法描述:

      • 选择一个增量序列 t1,t2,…,tk,其中 ti>tj,tk=1;
      • 按增量序列个数 k,对序列进行 k 趟排序;
      • 每趟排序,根据对应的增量 ti,将待排序列分割成若干长度为 m 的子序列,分别对各子表进行直接插入排序。仅增量因子为 1 时,整个序列作为一个表来处理,表长度即为整个序列的长度。

      动图演示

      希尔排序

      代码实现

      js
      /**
        * @description: 希尔排序,是简单插入排序的改进版。它与插入排序的不同之处在于,它会优先比较距离较远的元素。希尔排序又叫缩小增量排序。
        * @param {Array} list
        * @return {Array}
      diff --git a/assets/src_ranuts_sort_shell_index.md.fb9ca14b.lean.js b/assets/src_ranuts_sort_shell_index.md.11a9ac10.lean.js
      similarity index 86%
      rename from assets/src_ranuts_sort_shell_index.md.fb9ca14b.lean.js
      rename to assets/src_ranuts_sort_shell_index.md.11a9ac10.lean.js
      index 296a23719..4152ec4a6 100644
      --- a/assets/src_ranuts_sort_shell_index.md.fb9ca14b.lean.js
      +++ b/assets/src_ranuts_sort_shell_index.md.11a9ac10.lean.js
      @@ -1 +1 @@
      -import{_ as s,o as a,c as l,N as n}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/shell.7f63988c.gif",E=JSON.parse('{"title":"希尔排序(Shell Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/shell/index.md","lastUpdated":1694276611000}'),p={name:"src/ranuts/sort/shell/index.md"},t=n("",11),e=[t];function c(r,y,F,D,i,B){return a(),l("div",null,e)}const h=s(p,[["render",c]]);export{E as __pageData,h as default};
      +import{_ as s,o as a,c as l,N as n}from"./chunks/framework.6fe2e870.js";const o="/ran/assets/shell.7f63988c.gif",E=JSON.parse('{"title":"希尔排序(Shell Sort)","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/sort/shell/index.md","lastUpdated":1694419899000}'),p={name:"src/ranuts/sort/shell/index.md"},t=n("",11),e=[t];function c(r,y,F,D,i,B){return a(),l("div",null,e)}const h=s(p,[["render",c]]);export{E as __pageData,h as default};
      diff --git a/assets/src_ranuts_utils_index.md.0a4014cb.js b/assets/src_ranuts_utils_index.md.5d834957.js
      similarity index 97%
      rename from assets/src_ranuts_utils_index.md.0a4014cb.js
      rename to assets/src_ranuts_utils_index.md.5d834957.js
      index fcddc8783..f17eb807a 100644
      --- a/assets/src_ranuts_utils_index.md.0a4014cb.js
      +++ b/assets/src_ranuts_utils_index.md.5d834957.js
      @@ -1 +1 @@
      -import{_ as t,o as e,c as d,N as a}from"./chunks/framework.6fe2e870.js";const p=JSON.parse('{"title":"通用函数","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/utils/index.md","lastUpdated":1694276611000}'),r={name:"src/ranuts/utils/index.md"},o=a('

      通用函数

      filterObj

      过滤对象的属性,去除对象中在 list 数组里面有的属性,返回一个新对象,一般是用于去除空字符和 null

      API

      Return

      参数说明类型
      run启动轮询(...params: TParams) => void
      runAsync启动轮询(...params: TParams) => Promise<TData>
      cancel停止轮询() => void

      Options

      参数说明类型默认值
      pollingInterval轮询间隔,单位为毫秒。如果值大于 0,则启动轮询模式。number0
      pollingWhenHidden在页面隐藏时,是否继续轮询。如果设置为 false,在页面隐藏时会暂时停止轮询,页面重新显示时继续上次轮询。booleantrue
      pollingErrorRetryCount轮询错误重试次数。如果设置为 -1,则无限次number-1
      ',8),n=[o];function i(l,c,s,h,u,_){return e(),d("div",null,n)}const m=t(r,[["render",i]]);export{p as __pageData,m as default}; +import{_ as t,o as e,c as d,N as a}from"./chunks/framework.6fe2e870.js";const p=JSON.parse('{"title":"通用函数","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/utils/index.md","lastUpdated":1694419899000}'),r={name:"src/ranuts/utils/index.md"},o=a('

      通用函数

      filterObj

      过滤对象的属性,去除对象中在 list 数组里面有的属性,返回一个新对象,一般是用于去除空字符和 null

      API

      Return

      参数说明类型
      run启动轮询(...params: TParams) => void
      runAsync启动轮询(...params: TParams) => Promise<TData>
      cancel停止轮询() => void

      Options

      参数说明类型默认值
      pollingInterval轮询间隔,单位为毫秒。如果值大于 0,则启动轮询模式。number0
      pollingWhenHidden在页面隐藏时,是否继续轮询。如果设置为 false,在页面隐藏时会暂时停止轮询,页面重新显示时继续上次轮询。booleantrue
      pollingErrorRetryCount轮询错误重试次数。如果设置为 -1,则无限次number-1
      ',8),n=[o];function i(l,c,s,h,u,_){return e(),d("div",null,n)}const m=t(r,[["render",i]]);export{p as __pageData,m as default}; diff --git a/assets/src_ranuts_utils_index.md.0a4014cb.lean.js b/assets/src_ranuts_utils_index.md.5d834957.lean.js similarity index 84% rename from assets/src_ranuts_utils_index.md.0a4014cb.lean.js rename to assets/src_ranuts_utils_index.md.5d834957.lean.js index 99bb1d02d..bf6158ce8 100644 --- a/assets/src_ranuts_utils_index.md.0a4014cb.lean.js +++ b/assets/src_ranuts_utils_index.md.5d834957.lean.js @@ -1 +1 @@ -import{_ as t,o as e,c as d,N as a}from"./chunks/framework.6fe2e870.js";const p=JSON.parse('{"title":"通用函数","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/utils/index.md","lastUpdated":1694276611000}'),r={name:"src/ranuts/utils/index.md"},o=a("",8),n=[o];function i(l,c,s,h,u,_){return e(),d("div",null,n)}const m=t(r,[["render",i]]);export{p as __pageData,m as default}; +import{_ as t,o as e,c as d,N as a}from"./chunks/framework.6fe2e870.js";const p=JSON.parse('{"title":"通用函数","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/utils/index.md","lastUpdated":1694419899000}'),r={name:"src/ranuts/utils/index.md"},o=a("",8),n=[o];function i(l,c,s,h,u,_){return e(),d("div",null,n)}const m=t(r,[["render",i]]);export{p as __pageData,m as default}; diff --git a/assets/src_ranuts_utils_task.md.71af1297.js b/assets/src_ranuts_utils_task.md.5c6e5a88.js similarity index 99% rename from assets/src_ranuts_utils_task.md.71af1297.js rename to assets/src_ranuts_utils_task.md.5c6e5a88.js index 30c0a0e1e..4767d35fe 100644 --- a/assets/src_ranuts_utils_task.md.71af1297.js +++ b/assets/src_ranuts_utils_task.md.5c6e5a88.js @@ -1 +1 @@ -import{_ as e,o as a,c as t,N as o}from"./chunks/framework.6fe2e870.js";const k=JSON.parse('{"title":"统计执行时间","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/utils/task.md","lastUpdated":1694276611000}'),s={name:"src/ranuts/utils/task.md"},n=o('

      统计执行时间

      有的时候,我们需要统计一个函数的执行时间,用于分析性能。因此封装了startTasktaskEnd函数。同时介绍其他三种统计方法

      1. new Date().getTime(),
      2. console.time()console.timeEnd(),
      3. performance.now()

      一.startTask,taskEnd

      1.startTask

      任务开始之前执行

      Return

      参数说明类型
      taskId任务标识unique symbol

      2.taskEnd

      任务结束的时候执行,需要传入startTask返回的任务标识

      Options

      参数说明类型默认值
      taskId任务标识unique symbol 无默认值,参数必传,否则无法识别是哪个任务

      Return

      参数说明类型
      timetask执行的时间number

      3.使用例子

      js
      const taskId = startTask()\n\n// do something\n\nconst time = taskEnd(taskId)\n\nconsole.log('task 执行花费的时间', time)

      二.new Date().getTime()

      new Date().getTime() 返回一个数值,表示从 1970 年 1 月 1 日 0 时 0 分 0 秒(UTC,即协调世界时)距离该日期对象所代表时间的毫秒数。用来计算 JS 执行时间会有两个问题:

      1. 某些情况下,毫秒级精度可能不够。
      2. new Date() 解析的时间在不同浏览器,或者不同设备上可能并不一致。MDN 说明

        由于浏览器之间的差异与不一致性,强烈不推荐使用 Date 构造函数来解析日期字符串 (或使用与其等价的 Date.parse)。对 RFC 2822 格式的日期仅有约定俗成的支持。对 ISO 8601 格式的支持中,仅有日期的串 (例如 "1970-01-01") 会被处理为 UTC 而不是本地时间,与其他格式的串的处理不同。

      三.console.time(), console.timeEnd()

      启动一个计时器来跟踪某一个操作的占用时长。每一个计时器必须拥有唯一的名字,页面中最多能同时运行 10,000 个计时器。当以此计时器名字为参数调用 console.timeEnd() 时,浏览器将以毫秒为单位,输出对应计时器所经过的时间。比起new Date().getTime(),统计时间更加精确,可以统计到 0.001 毫秒(比如:0.134ms)

      四.performance.now()

      performance.now()返回的时间精度最高可达微秒级,且不会受到系统时间的影响(系统时钟可能会被手动调整或被 NTP 等软件篡改)。另外,performance.timing.navigationStart + performance.now() 约等于 Date.now()。因此对于统计 JS 执行耗时方面,更推荐使用performance.now()

      注意:为了提供对定时攻击和指纹的保护,performance.now() 的精度可能会根据浏览器的设置而被舍弃。 在 Firefox 中,privacy.reduceTimerPrecision 偏好是默认启用的,默认值为 1ms。可以启用 privacy.resistFingerprinting 这将精度改为 100ms 或privacy.resistFingerprinting.reduceTimerPrecision.microseconds 的值,以较大者为准。

      ',24),r=[n];function c(d,l,i,p,h,m){return a(),t("div",null,r)}const b=e(s,[["render",c]]);export{k as __pageData,b as default}; +import{_ as e,o as a,c as t,N as o}from"./chunks/framework.6fe2e870.js";const k=JSON.parse('{"title":"统计执行时间","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/utils/task.md","lastUpdated":1694419899000}'),s={name:"src/ranuts/utils/task.md"},n=o('

      统计执行时间

      有的时候,我们需要统计一个函数的执行时间,用于分析性能。因此封装了startTasktaskEnd函数。同时介绍其他三种统计方法

      1. new Date().getTime(),
      2. console.time()console.timeEnd(),
      3. performance.now()

      一.startTask,taskEnd

      1.startTask

      任务开始之前执行

      Return

      参数说明类型
      taskId任务标识unique symbol

      2.taskEnd

      任务结束的时候执行,需要传入startTask返回的任务标识

      Options

      参数说明类型默认值
      taskId任务标识unique symbol 无默认值,参数必传,否则无法识别是哪个任务

      Return

      参数说明类型
      timetask执行的时间number

      3.使用例子

      js
      const taskId = startTask()\n\n// do something\n\nconst time = taskEnd(taskId)\n\nconsole.log('task 执行花费的时间', time)

      二.new Date().getTime()

      new Date().getTime() 返回一个数值,表示从 1970 年 1 月 1 日 0 时 0 分 0 秒(UTC,即协调世界时)距离该日期对象所代表时间的毫秒数。用来计算 JS 执行时间会有两个问题:

      1. 某些情况下,毫秒级精度可能不够。
      2. new Date() 解析的时间在不同浏览器,或者不同设备上可能并不一致。MDN 说明

        由于浏览器之间的差异与不一致性,强烈不推荐使用 Date 构造函数来解析日期字符串 (或使用与其等价的 Date.parse)。对 RFC 2822 格式的日期仅有约定俗成的支持。对 ISO 8601 格式的支持中,仅有日期的串 (例如 "1970-01-01") 会被处理为 UTC 而不是本地时间,与其他格式的串的处理不同。

      三.console.time(), console.timeEnd()

      启动一个计时器来跟踪某一个操作的占用时长。每一个计时器必须拥有唯一的名字,页面中最多能同时运行 10,000 个计时器。当以此计时器名字为参数调用 console.timeEnd() 时,浏览器将以毫秒为单位,输出对应计时器所经过的时间。比起new Date().getTime(),统计时间更加精确,可以统计到 0.001 毫秒(比如:0.134ms)

      四.performance.now()

      performance.now()返回的时间精度最高可达微秒级,且不会受到系统时间的影响(系统时钟可能会被手动调整或被 NTP 等软件篡改)。另外,performance.timing.navigationStart + performance.now() 约等于 Date.now()。因此对于统计 JS 执行耗时方面,更推荐使用performance.now()

      注意:为了提供对定时攻击和指纹的保护,performance.now() 的精度可能会根据浏览器的设置而被舍弃。 在 Firefox 中,privacy.reduceTimerPrecision 偏好是默认启用的,默认值为 1ms。可以启用 privacy.resistFingerprinting 这将精度改为 100ms 或privacy.resistFingerprinting.reduceTimerPrecision.microseconds 的值,以较大者为准。

      ',24),r=[n];function c(d,l,i,p,h,m){return a(),t("div",null,r)}const b=e(s,[["render",c]]);export{k as __pageData,b as default}; diff --git a/assets/src_ranuts_utils_task.md.71af1297.lean.js b/assets/src_ranuts_utils_task.md.5c6e5a88.lean.js similarity index 84% rename from assets/src_ranuts_utils_task.md.71af1297.lean.js rename to assets/src_ranuts_utils_task.md.5c6e5a88.lean.js index 5af208fdb..b8114a33f 100644 --- a/assets/src_ranuts_utils_task.md.71af1297.lean.js +++ b/assets/src_ranuts_utils_task.md.5c6e5a88.lean.js @@ -1 +1 @@ -import{_ as e,o as a,c as t,N as o}from"./chunks/framework.6fe2e870.js";const k=JSON.parse('{"title":"统计执行时间","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/utils/task.md","lastUpdated":1694276611000}'),s={name:"src/ranuts/utils/task.md"},n=o("",24),r=[n];function c(d,l,i,p,h,m){return a(),t("div",null,r)}const b=e(s,[["render",c]]);export{k as __pageData,b as default}; +import{_ as e,o as a,c as t,N as o}from"./chunks/framework.6fe2e870.js";const k=JSON.parse('{"title":"统计执行时间","description":"","frontmatter":{},"headers":[],"relativePath":"src/ranuts/utils/task.md","lastUpdated":1694419899000}'),s={name:"src/ranuts/utils/task.md"},n=o("",24),r=[n];function c(d,l,i,p,h,m){return a(),t("div",null,r)}const b=e(s,[["render",c]]);export{k as __pageData,b as default}; diff --git "a/assets/src_types_TS\347\261\273\345\236\213.md.c98631f5.js" "b/assets/src_types_TS\347\261\273\345\236\213.md.7ed2f3b0.js" similarity index 99% rename from "assets/src_types_TS\347\261\273\345\236\213.md.c98631f5.js" rename to "assets/src_types_TS\347\261\273\345\236\213.md.7ed2f3b0.js" index 93e759ca2..31d3115fe 100644 --- "a/assets/src_types_TS\347\261\273\345\236\213.md.c98631f5.js" +++ "b/assets/src_types_TS\347\261\273\345\236\213.md.7ed2f3b0.js" @@ -1,4 +1,4 @@ -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"TypeScript 类型系统中的类型","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/TS类型.md","lastUpdated":1694276611000}'),p={name:"src/types/TS类型.md"},o=l(`

      TypeScript 类型系统中的类型

      1. 基本类型: number、boolean、string、object、bigint、symbol、undefined、null
      2. 复合类型: class、Array、元组(Tuple)、接口(Interface)、枚举(Enum)
      3. 特殊的类型:void、never、any、unknown

      Tuple

      元组(Tuple)就是元素个数和类型固定的数组类型:

      ts
      type Tuple = [number, string]

      Interface

      接口(Interface)可以描述函数、对象、构造器的结构:

      • 对象
      ts
      interface IPerson {
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"TypeScript 类型系统中的类型","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/TS类型.md","lastUpdated":1694419899000}'),p={name:"src/types/TS类型.md"},o=l(`

      TypeScript 类型系统中的类型

      1. 基本类型: number、boolean、string、object、bigint、symbol、undefined、null
      2. 复合类型: class、Array、元组(Tuple)、接口(Interface)、枚举(Enum)
      3. 特殊的类型:void、never、any、unknown

      Tuple

      元组(Tuple)就是元素个数和类型固定的数组类型:

      ts
      type Tuple = [number, string]

      Interface

      接口(Interface)可以描述函数、对象、构造器的结构:

      • 对象
      ts
      interface IPerson {
         name: string
         age: number
       }
      diff --git "a/assets/src_types_TS\347\261\273\345\236\213.md.c98631f5.lean.js" "b/assets/src_types_TS\347\261\273\345\236\213.md.7ed2f3b0.lean.js"
      similarity index 84%
      rename from "assets/src_types_TS\347\261\273\345\236\213.md.c98631f5.lean.js"
      rename to "assets/src_types_TS\347\261\273\345\236\213.md.7ed2f3b0.lean.js"
      index 66930b6d7..6e79b136c 100644
      --- "a/assets/src_types_TS\347\261\273\345\236\213.md.c98631f5.lean.js"
      +++ "b/assets/src_types_TS\347\261\273\345\236\213.md.7ed2f3b0.lean.js"
      @@ -1 +1 @@
      -import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"TypeScript 类型系统中的类型","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/TS类型.md","lastUpdated":1694276611000}'),p={name:"src/types/TS类型.md"},o=l("",33),e=[o];function t(c,r,y,D,F,B){return a(),n("div",null,e)}const A=s(p,[["render",t]]);export{E as __pageData,A as default};
      +import{_ as s,o as a,c as n,N as l}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"TypeScript 类型系统中的类型","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/TS类型.md","lastUpdated":1694419899000}'),p={name:"src/types/TS类型.md"},o=l("",33),e=[o];function t(c,r,y,D,F,B){return a(),n("div",null,e)}const A=s(p,[["render",t]]);export{E as __pageData,A as default};
      diff --git "a/assets/src_types_\346\250\241\345\274\217\345\214\271\351\205\215.md.78b179d7.js" "b/assets/src_types_\346\250\241\345\274\217\345\214\271\351\205\215.md.c289dbd1.js"
      similarity index 99%
      rename from "assets/src_types_\346\250\241\345\274\217\345\214\271\351\205\215.md.78b179d7.js"
      rename to "assets/src_types_\346\250\241\345\274\217\345\214\271\351\205\215.md.c289dbd1.js"
      index 1522f2e36..25aec7e12 100644
      --- "a/assets/src_types_\346\250\241\345\274\217\345\214\271\351\205\215.md.78b179d7.js"
      +++ "b/assets/src_types_\346\250\241\345\274\217\345\214\271\351\205\215.md.c289dbd1.js"
      @@ -1,4 +1,4 @@
      -import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"模式匹配","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/模式匹配.md","lastUpdated":1694276611000}'),p={name:"src/types/模式匹配.md"},o=l(`

      模式匹配

      Typescript 的类型也同样可以做模式匹配。

      比如这样一个 Promise 类型:

      ts
      type p = Promise<'value'>

      我们想提取 value 的类型,可以这样做:

      ts
      type GetPromiseValue<T> = T extends Promise<infer value> ? value : never

      通过 extends 对传入的类型参数 P 做模式匹配,其中值的类型是需要提取的,通过 infer 声明一个局部变量 Value 来保存,如果匹配,就返回匹配到的 Value,否则就返回 never 代表没匹配到。

      ts
      type result = GetPromiseValue<Promise<'name'>> // name

      数组类型

      数组类型想提取第一个元素的类型怎么做呢?

      ts
      type arr = [1, 2, 3]

      用它来匹配一个模式类型,提取第一个元素的类型到通过 infer 声明的局部变量里返回。

      ts
      type GetArrayFirstItem<T extends unknown[]> = T extends [
      +import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"模式匹配","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/模式匹配.md","lastUpdated":1694419899000}'),p={name:"src/types/模式匹配.md"},o=l(`

      模式匹配

      Typescript 的类型也同样可以做模式匹配。

      比如这样一个 Promise 类型:

      ts
      type p = Promise<'value'>

      我们想提取 value 的类型,可以这样做:

      ts
      type GetPromiseValue<T> = T extends Promise<infer value> ? value : never

      通过 extends 对传入的类型参数 P 做模式匹配,其中值的类型是需要提取的,通过 infer 声明一个局部变量 Value 来保存,如果匹配,就返回匹配到的 Value,否则就返回 never 代表没匹配到。

      ts
      type result = GetPromiseValue<Promise<'name'>> // name

      数组类型

      数组类型想提取第一个元素的类型怎么做呢?

      ts
      type arr = [1, 2, 3]

      用它来匹配一个模式类型,提取第一个元素的类型到通过 infer 声明的局部变量里返回。

      ts
      type GetArrayFirstItem<T extends unknown[]> = T extends [
         infer value,
         ...unknown[],
       ]
      diff --git "a/assets/src_types_\346\250\241\345\274\217\345\214\271\351\205\215.md.78b179d7.lean.js" "b/assets/src_types_\346\250\241\345\274\217\345\214\271\351\205\215.md.c289dbd1.lean.js"
      similarity index 84%
      rename from "assets/src_types_\346\250\241\345\274\217\345\214\271\351\205\215.md.78b179d7.lean.js"
      rename to "assets/src_types_\346\250\241\345\274\217\345\214\271\351\205\215.md.c289dbd1.lean.js"
      index 3e3901689..9862a428f 100644
      --- "a/assets/src_types_\346\250\241\345\274\217\345\214\271\351\205\215.md.78b179d7.lean.js"
      +++ "b/assets/src_types_\346\250\241\345\274\217\345\214\271\351\205\215.md.c289dbd1.lean.js"
      @@ -1 +1 @@
      -import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"模式匹配","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/模式匹配.md","lastUpdated":1694276611000}'),p={name:"src/types/模式匹配.md"},o=l("",79),e=[o];function t(r,c,B,y,F,D){return n(),a("div",null,e)}const E=s(p,[["render",t]]);export{A as __pageData,E as default};
      +import{_ as s,o as n,c as a,N as l}from"./chunks/framework.6fe2e870.js";const A=JSON.parse('{"title":"模式匹配","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/模式匹配.md","lastUpdated":1694419899000}'),p={name:"src/types/模式匹配.md"},o=l("",79),e=[o];function t(r,c,B,y,F,D){return n(),a("div",null,e)}const E=s(p,[["render",t]]);export{A as __pageData,E as default};
      diff --git "a/assets/src_types_\347\261\273\345\236\213\350\277\220\347\256\227.md.78b341b5.js" "b/assets/src_types_\347\261\273\345\236\213\350\277\220\347\256\227.md.bb1f0e32.js"
      similarity index 99%
      rename from "assets/src_types_\347\261\273\345\236\213\350\277\220\347\256\227.md.78b341b5.js"
      rename to "assets/src_types_\347\261\273\345\236\213\350\277\220\347\256\227.md.bb1f0e32.js"
      index 72fe0d59e..3e0bae4b9 100644
      --- "a/assets/src_types_\347\261\273\345\236\213\350\277\220\347\256\227.md.78b341b5.js"
      +++ "b/assets/src_types_\347\261\273\345\236\213\350\277\220\347\256\227.md.bb1f0e32.js"
      @@ -1,4 +1,4 @@
      -import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"TypeScript 类型系统中的类型运算","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/类型运算.md","lastUpdated":1694276611000}'),l={name:"src/types/类型运算.md"},o=p(`

      TypeScript 类型系统中的类型运算

      条件:extends ? :

      TypeScript 里的条件判断是 extends ? :,叫做条件类型(Conditional Type)比如:

      ts
      type isTwo<T> = T extends 2 ? true : false
      +import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"TypeScript 类型系统中的类型运算","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/类型运算.md","lastUpdated":1694419899000}'),l={name:"src/types/类型运算.md"},o=p(`

      TypeScript 类型系统中的类型运算

      条件:extends ? :

      TypeScript 里的条件判断是 extends ? :,叫做条件类型(Conditional Type)比如:

      ts
      type isTwo<T> = T extends 2 ? true : false
       
       type res = isTwo<1> // true
       type res2 = isTwo<2> // false

      这种类型也叫做高级类型。

      高级类型的特点是传入类型参数,经过一系列类型运算逻辑后,返回新的类型。

      推导:infer

      如何提取类型的一部分呢?答案是 infer。

      比如提取元组类型的第一个元素:

      ts
      type FirstTupleItem<Tuple extends unknown[]> = Tuple extends [infer T, ...inter R] ? T : never;
      diff --git "a/assets/src_types_\347\261\273\345\236\213\350\277\220\347\256\227.md.78b341b5.lean.js" "b/assets/src_types_\347\261\273\345\236\213\350\277\220\347\256\227.md.bb1f0e32.lean.js"
      similarity index 70%
      rename from "assets/src_types_\347\261\273\345\236\213\350\277\220\347\256\227.md.78b341b5.lean.js"
      rename to "assets/src_types_\347\261\273\345\236\213\350\277\220\347\256\227.md.bb1f0e32.lean.js"
      index 4a010428d..9fae5e705 100644
      --- "a/assets/src_types_\347\261\273\345\236\213\350\277\220\347\256\227.md.78b341b5.lean.js"
      +++ "b/assets/src_types_\347\261\273\345\236\213\350\277\220\347\256\227.md.bb1f0e32.lean.js"
      @@ -1 +1 @@
      -import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"TypeScript 类型系统中的类型运算","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/类型运算.md","lastUpdated":1694276611000}'),l={name:"src/types/类型运算.md"},o=p("",35),e=[o];function t(c,r,y,B,F,D){return a(),n("div",null,e)}const A=s(l,[["render",t]]);export{E as __pageData,A as default};
      +import{_ as s,o as a,c as n,N as p}from"./chunks/framework.6fe2e870.js";const E=JSON.parse('{"title":"TypeScript 类型系统中的类型运算","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/类型运算.md","lastUpdated":1694419899000}'),l={name:"src/types/类型运算.md"},o=p("",35),e=[o];function t(c,r,y,B,F,D){return a(),n("div",null,e)}const A=s(l,[["render",t]]);export{E as __pageData,A as default};
      diff --git "a/assets/src_types_\351\253\230\347\272\247\347\261\273\345\236\213.md.1ffcb487.js" "b/assets/src_types_\351\253\230\347\272\247\347\261\273\345\236\213.md.3ad1299c.js"
      similarity index 98%
      rename from "assets/src_types_\351\253\230\347\272\247\347\261\273\345\236\213.md.1ffcb487.js"
      rename to "assets/src_types_\351\253\230\347\272\247\347\261\273\345\236\213.md.3ad1299c.js"
      index 8c33c7924..2688531f9 100644
      --- "a/assets/src_types_\351\253\230\347\272\247\347\261\273\345\236\213.md.1ffcb487.js"
      +++ "b/assets/src_types_\351\253\230\347\272\247\347\261\273\345\236\213.md.3ad1299c.js"
      @@ -1 +1 @@
      -import{_ as a,o as e,c as r,N as t}from"./chunks/framework.6fe2e870.js";const b=JSON.parse('{"title":"TypeScript 内置的高级类型","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/高级类型.md","lastUpdated":1694276611000}'),i={name:"src/types/高级类型.md"},o=t('

      TypeScript 内置的高级类型

      Parameters

      Parameters 用于提取函数类型的参数类型。

      ReturnType

      ReturnType 用于提取函数类型的返回值类型。

      ConstructorParameters

      构造器类型和函数类型的区别就是可以被 new。

      Parameters 用于提取函数参数的类型,而 ConstructorParameters 用于提取构造器参数的类型。

      InstanceType

      提取了构造器参数的类型,自然也可以提取构造器返回值的类型,就是 InstanceType。

      ThisParameterType

      OmitThisParameter

      Partial

      Required

      Readonly

      Pick

      Record

      Exclude

      Extract

      Omit

      Awaited

      NonNullable

      Uppercase

      Lowercase

      Capitalize

      Uncapitalize

      总结

      比如用模式匹配可以实现:Parameters、ReturnType、ConstructorParameters、InstanceType、ThisParameterType。

      用模式匹配 + 重新构造可以实现:OmitThisParameter

      用重新构造可以实现:Partial、Required、Readonly、Pick、Record

      用模式匹配 + 递归可以实现: Awaited

      用联合类型在分布式条件类型的特性可以实现: Exclude

      此外还有 NonNullable 和四个编译器内部实现的类型:Uppercase、Lowercase、Capitalize、Uncapitalize。

      ',33),l=[o];function n(h,c,s,d,p,u){return e(),r("div",null,l)}const q=a(i,[["render",n]]);export{b as __pageData,q as default}; +import{_ as a,o as e,c as r,N as t}from"./chunks/framework.6fe2e870.js";const b=JSON.parse('{"title":"TypeScript 内置的高级类型","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/高级类型.md","lastUpdated":1694419899000}'),i={name:"src/types/高级类型.md"},o=t('

      TypeScript 内置的高级类型

      Parameters

      Parameters 用于提取函数类型的参数类型。

      ReturnType

      ReturnType 用于提取函数类型的返回值类型。

      ConstructorParameters

      构造器类型和函数类型的区别就是可以被 new。

      Parameters 用于提取函数参数的类型,而 ConstructorParameters 用于提取构造器参数的类型。

      InstanceType

      提取了构造器参数的类型,自然也可以提取构造器返回值的类型,就是 InstanceType。

      ThisParameterType

      OmitThisParameter

      Partial

      Required

      Readonly

      Pick

      Record

      Exclude

      Extract

      Omit

      Awaited

      NonNullable

      Uppercase

      Lowercase

      Capitalize

      Uncapitalize

      总结

      比如用模式匹配可以实现:Parameters、ReturnType、ConstructorParameters、InstanceType、ThisParameterType。

      用模式匹配 + 重新构造可以实现:OmitThisParameter

      用重新构造可以实现:Partial、Required、Readonly、Pick、Record

      用模式匹配 + 递归可以实现: Awaited

      用联合类型在分布式条件类型的特性可以实现: Exclude

      此外还有 NonNullable 和四个编译器内部实现的类型:Uppercase、Lowercase、Capitalize、Uncapitalize。

      ',33),l=[o];function n(h,c,s,d,p,u){return e(),r("div",null,l)}const q=a(i,[["render",n]]);export{b as __pageData,q as default}; diff --git "a/assets/src_types_\351\253\230\347\272\247\347\261\273\345\236\213.md.1ffcb487.lean.js" "b/assets/src_types_\351\253\230\347\272\247\347\261\273\345\236\213.md.3ad1299c.lean.js" similarity index 85% rename from "assets/src_types_\351\253\230\347\272\247\347\261\273\345\236\213.md.1ffcb487.lean.js" rename to "assets/src_types_\351\253\230\347\272\247\347\261\273\345\236\213.md.3ad1299c.lean.js" index d86cc91a1..f73b30b10 100644 --- "a/assets/src_types_\351\253\230\347\272\247\347\261\273\345\236\213.md.1ffcb487.lean.js" +++ "b/assets/src_types_\351\253\230\347\272\247\347\261\273\345\236\213.md.3ad1299c.lean.js" @@ -1 +1 @@ -import{_ as a,o as e,c as r,N as t}from"./chunks/framework.6fe2e870.js";const b=JSON.parse('{"title":"TypeScript 内置的高级类型","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/高级类型.md","lastUpdated":1694276611000}'),i={name:"src/types/高级类型.md"},o=t("",33),l=[o];function n(h,c,s,d,p,u){return e(),r("div",null,l)}const q=a(i,[["render",n]]);export{b as __pageData,q as default}; +import{_ as a,o as e,c as r,N as t}from"./chunks/framework.6fe2e870.js";const b=JSON.parse('{"title":"TypeScript 内置的高级类型","description":"","frontmatter":{},"headers":[],"relativePath":"src/types/高级类型.md","lastUpdated":1694419899000}'),i={name:"src/types/高级类型.md"},o=t("",33),l=[o];function n(h,c,s,d,p,u){return e(),r("div",null,l)}const q=a(i,[["render",n]]);export{b as __pageData,q as default}; diff --git a/hashmap.json b/hashmap.json index 290de0a89..492977447 100644 --- a/hashmap.json +++ b/hashmap.json @@ -1 +1 @@ -{"index.md":"0d318df1","src_article_bundle.md":"7a4db991","src_article_babel.md":"31e21b4d","src_article_astparse_tokenizer.md":"ccc361f6","src_article_typescript_index.md":"9450f83c","src_article_imagemin.md":"05c1ce1a","src_article_javascript_domload.md":"c8afd20d","src_article_typescript_calculate.md":"f31139e2","src_ranui_message_index.md":"fa827f88","src_article_designmode.md":"fdb686ab","src_ranuts_sort_insert_index.md":"de43fe03","src_ranuts_sort_quick_index.md":"9ea0bd8c","src_ranuts_sort_radix_index.md":"45ded274","src_ranuts_sort_select_index.md":"f91454eb","src_ranuts_sort_shell_index.md":"fb9ca14b","src_ranuts_utils_index.md":"0a4014cb","src_ranuts_utils_task.md":"71af1297","src_types_ts类型.md":"c98631f5","src_types_类型运算.md":"78b341b5","src_types_高级类型.md":"1ffcb487","src_article_typescript_reconstruction.md":"a186fc17","src_article_typescript_recursion.md":"9db97e45","src_article_typescript_uniontype.md":"0ba9cf91","src_ranui_button_index.md":"32a9d03f","src_ranui_icon_index.md":"11f7a3d7","src_ranui_image_index.md":"08e89593","src_ranui_index.md":"0d4c041d","src_ranui_input_index.md":"24d6061b","src_ranui_modal_index.md":"f410050d","src_ranui_preview_index.md":"9a6fa2f6","src_ranui_skeleton_index.md":"472eb512","src_ranui_tab_index.md":"087770cd","src_ranui_tabs_index.md":"eff91b6b","src_ranuts_binarytree_index.md":"9df1bc99","src_ranuts_bundler_index.md":"605ec91e","src_ranuts_file_index.md":"177e1adc","src_ranuts_index.md":"238fdd9e","src_ranuts_sort_bubble_index.md":"3b551a0c","src_ranuts_sort_bucket_index.md":"755536c3","src_ranuts_sort_count_index.md":"52dd4e7c","src_ranuts_sort_heap_index.md":"c3678584","src_ranuts_sort_index.md":"b7c48f70","src_types_模式匹配.md":"78b179d7","src_article_functionalprogramming.md":"e9f945b3","src_ranuts_sort_merge_index.md":"46e8a482","src_article_typescript_pattern.md":"e19bc8e4"} +{"src_article_babel.md":"90287ca6","src_article_astparse_tokenizer.md":"99a1983d","src_article_bundle.md":"1794ffc6","src_ranuts_sort_insert_index.md":"032011a9","src_article_javascript_domload.md":"f4dd6368","src_article_typescript_uniontype.md":"2b84a746","src_ranui_button_index.md":"eb788bcc","src_ranui_icon_index.md":"e1b014ce","src_ranui_image_index.md":"c0b5dda1","src_ranui_input_index.md":"755c9dc7","src_ranui_index.md":"8ed6a12e","src_ranui_message_index.md":"2e7373db","src_article_typescript_calculate.md":"9ee3c861","src_article_typescript_reconstruction.md":"21d449a5","src_ranuts_sort_heap_index.md":"ed19bcdb","src_ranuts_sort_merge_index.md":"d6dbf578","src_article_typescript_index.md":"8330b87a","src_article_typescript_pattern.md":"ed4c198d","src_ranuts_sort_radix_index.md":"55bb0018","src_ranuts_utils_index.md":"5d834957","src_ranuts_utils_task.md":"5c6e5a88","src_article_typescript_recursion.md":"061a1d67","src_ranui_preview_index.md":"fa3a8f12","src_ranui_skeleton_index.md":"51e3be48","src_ranui_tab_index.md":"ec0122dc","src_ranuts_index.md":"f4ae71e9","src_ranui_tabs_index.md":"301a2c3b","src_ranuts_sort_index.md":"fb0f8a4a","src_ranuts_binarytree_index.md":"493e0878","src_ranuts_file_index.md":"3ff8632d","src_ranuts_sort_bubble_index.md":"d2e4bc60","src_article_imagemin.md":"32ccabea","index.md":"4abc46fe","src_ranuts_sort_count_index.md":"bcf7daeb","src_ranuts_sort_quick_index.md":"3d358bda","src_ranuts_sort_select_index.md":"cb9012a3","src_ranuts_sort_shell_index.md":"11a9ac10","src_types_ts类型.md":"7ed2f3b0","src_types_类型运算.md":"bb1f0e32","src_types_高级类型.md":"3ad1299c","src_article_functionalprogramming.md":"0ef4df92","src_ranui_modal_index.md":"69c2268b","src_types_模式匹配.md":"c289dbd1","src_article_designmode.md":"c8355f94","src_ranuts_bundler_index.md":"bf6a07c5","src_ranuts_sort_bucket_index.md":"39a1b5ca"} diff --git a/index.html b/index.html index 31dc350ef..b523cf4d1 100644 --- a/index.html +++ b/index.html @@ -37,7 +37,7 @@ - + @@ -50,7 +50,7 @@
      Skip to content

      ran

      风起于青萍之末

      A ship in harbor is safe, but that is not what ships are built for.

      logo
      ⚡️

      记录

      每当遇到问题或者可以改进的事情的时候,会把想法先写下来,如果这是你遇到的问题,那么很可能有一群人也遇到同样的困难。

      🖖

      解决

      过一段时间后,再回来看看这些问题是否还有意义。然后再研究这个问题,确定是否存在解决方案。

      🛠️

      改进

      通常会发现一个工具是为了解决一个问题,我会去考虑如何改进或者简化这个工具。创造一个更好的轮子。

      Released under the MIT License.

      - diff --git a/src/article/astParse/tokenizer.html b/src/article/astParse/tokenizer.html index 8ad13c39e..d48972397 100644 --- a/src/article/astParse/tokenizer.html +++ b/src/article/astParse/tokenizer.html @@ -37,7 +37,7 @@ - + @@ -240,8 +240,8 @@ { type: 'RightParen', value: ')', start: 17, end: 18 }, { type: 'LeftCurly', value: '{', start: 19, end: 20 }, { type: 'RightCurly', value: '}', start: 20, end: 21 }, -]

      一个简易版本的分词器已经被我们开发出来了,不过目前的分词器还比较简陋,仅仅支持有限的语法,不过在明确了核心的开发步骤之后,后面继续完善的过程就比较简单了。

      四.编写语法分析器(Parser)

      在解析出词法 token 之后,我们就可以进入语法分析阶段了。在这个阶段,我们会依次遍历 token ,对代码进行语法结构层面的分析,最后的目标是生成 AST 数据结构。至于代码的 AST 结构到底是什么样子,你可以去 AST Explorer 网站进行在线预览:

      接下来,我们要做的就是将 token 数组转换为上图所示的 AST 数据。

      开发步骤主要分为:

      • 初始化类型声明

      Released under the MIT License.

      - diff --git a/src/article/babel.html b/src/article/babel.html index c8368f0b9..7907d7f7b 100644 --- a/src/article/babel.html +++ b/src/article/babel.html @@ -37,7 +37,7 @@ - + @@ -49,8 +49,8 @@ -
      Skip to content
      On this page

      Babel

      babel 核心库主要是:

      • @babel/parser 对源码进行 parse,可以通过 plugins、sourceType 等来指定 parse 语法,功能是把源码转成 AST。
      • @babel/traverse 通过 visitor 函数对遍历到的 ast 进行处理,分为 enter 和 exit 两个阶段,具体操作 AST 使用 path 的 api,还可以通过 state 来在遍历过程中传递一些数据
      • @babel/types 用于创建、判断 AST 节点,提供了 xxx、isXxx、assertXxx 的 api
      • @babel/template 当需要批量创建 AST 的时候可以使用 @babel/template 来简化 AST 创建逻辑。
      • @babel/code-frame 可以创建友好的报错信息
      • @babel/generator 打印 AST 成目标代码字符串,支持 comments、minified、sourceMaps 等选项。
      • @babel/core 基于上面的包来完成 babel 的编译流程,并应用 plugin 和 preset。

      Released under the MIT License.

      - diff --git a/src/article/bundle.html b/src/article/bundle.html index 3da765909..8fd75b53e 100644 --- a/src/article/bundle.html +++ b/src/article/bundle.html @@ -37,7 +37,7 @@ - + @@ -49,8 +49,8 @@ -
      Skip to content
      On this page

      Bundle

      Bundle 的本质就是输入,转换,输出。在机器上直接运行的代码,往往都难以维护和理解,我们需要将开发者方便理解和维护的代码,通过打包等工具转换成方便机器或者程序使用的代码。对于 web 前端来说,打包工具,至少需要以下功能:

      • 编译能力
      • 插件机制
      • HMR
      • cli 和命令行能力

      Released under the MIT License.

      - diff --git a/src/article/designMode.html b/src/article/designMode.html index a495686b2..3329969bb 100644 --- a/src/article/designMode.html +++ b/src/article/designMode.html @@ -37,7 +37,7 @@ - + @@ -800,8 +800,8 @@ Visitor.push(a,1,2,3,4); Visitor.push(a,4,5,6); Visitor.pop(a); -Visitor.splice(a,2);

      访问者模式解决了数据与数据的操作方法之间的耦合,让数据的操作方法独立于数据,使其可以自由演变。因此,访问者模式更适合于那些数据稳定、但数据的操作方法易变的环境下。

      优点:

      • 增加新的访问操作很方便。使用访问者模式,增加新的访问操作就意味着增加一个新的具体访问者类,实现简单,无须修改源代码,符合“开闭原则”。
      • 将有关元素对象的访问行为集中到一个访问者对象中,而不是分散在一个个的元素类中。类的职责更加清晰,有利于对象结构中元素对象的复用,相同的对象结构可以供多个不同的访问者访问。
      • 让用户能够在不修改现有元素类层次结构的情况下,定义作用于该层次结构的操作。

      缺点:

      • 增加新的元素类很困难。在访问者模式中,每增加一个新的元素类都意味着要在抽象访问者角色中增加一个新的抽象操作,并在每一个具体访问者类中增加相应的具体操作,这违背了“开闭原则”的要求。
      • 破坏封装。访问者模式要求访问者对象访问并调用每一个元素对象的操作,这意味着元素对象有时候必须暴露一些自己的内部操作和内部状态,否则无法供访问者访问。

      总结

      系统地学习设计模式后,你可以在过往的开发经历中发现,设计模式是无处不在的。在学习设计模式之前的很多时候我们是凭借过往经验和智慧来完善系统的设计,而这些经验很多和某个设计模式的思想不谋而合。

      还有一些地方没有完全理解,文中有误之处还望不吝指出。

      参考资料

      Released under the MIT License.

      - diff --git a/src/article/functionalProgramming.html b/src/article/functionalProgramming.html index 139997cb2..33ba0408c 100644 --- a/src/article/functionalProgramming.html +++ b/src/article/functionalProgramming.html @@ -37,7 +37,7 @@ - + @@ -386,8 +386,8 @@ } let r = readFile('package.json') //这里可以用map去处理内容 .flatMap(print) - .join()

      参考资料

      Released under the MIT License.

      - diff --git a/src/article/imagemin.html b/src/article/imagemin.html index 413995eb0..26219a193 100644 --- a/src/article/imagemin.html +++ b/src/article/imagemin.html @@ -37,7 +37,7 @@ - + @@ -49,8 +49,8 @@ -
      Skip to content
      On this page

      imagemin 图片压缩源码分析

      Released under the MIT License.

      - diff --git a/src/article/javascript/domLoad.html b/src/article/javascript/domLoad.html index ee7be22c6..570873c7d 100644 --- a/src/article/javascript/domLoad.html +++ b/src/article/javascript/domLoad.html @@ -37,7 +37,7 @@ - + @@ -51,8 +51,8 @@
      Skip to content
      On this page

      页面加载完成后事件

      window.onload

      DOMContentLoaded

      js
      document.addEventListener('DOMContentLoaded', fun)

      <body onload="fun()">

      readyState

      js
      document.readyState
       
      -document.onreadystatechange

      一个文档的 readyState 可以是以下之一:

      • loading / 加载 。document 仍在加载。
      • interactive / 互动。文档已经完成加载,文档已被解析,但是诸如图像,样式表和框架之类的子资源仍在加载。
      • complete / 完成。T 文档和所有子资源已完成加载。状态表示 load 事件即将被触发。

      Released under the MIT License.

      - diff --git a/src/article/typescript/calculate.html b/src/article/typescript/calculate.html index c808ee114..d97b63708 100644 --- a/src/article/typescript/calculate.html +++ b/src/article/typescript/calculate.html @@ -37,7 +37,7 @@ - + @@ -108,8 +108,8 @@ Num > -type Fibonacci<Num extends number> = FibonacciLoop<[1], [], [], Num>

      类型参数 PrevArr 是代表之前的累加值的数组。类型参数 CurrentArr 是代表当前数值的数组。

      类型参数 IndexArr 用于记录 index,每次递归加一,默认值是 [],代表从 0 开始。

      类型参数 Num 代表求数列的第几个数。

      判断当前 index 也就是 IndexArr['length'] 是否到了 Num,到了就返回当前的数值 CurrentArr['length']。

      否则求出当前 index 对应的数值,用之前的数加上当前的数 [...PrevArr, ... CurrentArr]。

      然后继续递归,index + 1,也就是 [...IndexArr, unknown]。

      这就是递归计算 Fibinacci 数列的数的过程。

      可以正确的算出第 8 个数是 21:

      Released under the MIT License.

      - diff --git a/src/article/typescript/index.html b/src/article/typescript/index.html index 77a3aeca8..bd84e44b2 100644 --- a/src/article/typescript/index.html +++ b/src/article/typescript/index.html @@ -37,7 +37,7 @@ - + @@ -230,8 +230,8 @@ ? R : never

      类型参数 U 是要转换的联合类型。

      U extends U 是为了触发联合类型的 distributive 的性质,让每个类型单独传入做计算,最后合并。

      利用 U 做为参数构造个函数,通过模式匹配取参数的类型。

      结果就是交叉类型

      函数参数的逆变性质一般就联合类型转交叉类型会用,记住就行。

      GetOptional

      如何提取索引类型中的可选索引呢?

      这也要利用可选索引的特性:可选索引的值为 undefined 和值类型的联合类型。

      过滤可选索引,就要构造一个新的索引类型,过程中做过滤:

      ts
      type GetOptional<Obj extends Record<string, any>> = {
         [Key in keyof Obj as {} extends Pick<Obj, Key> ? Key : never]: Obj[Key]
      -}

      类型参数 Obj 为待处理的索引类型,类型约束为索引为 string、值为任意类型的索引类型 Record<string, any>。

      用映射类型的语法重新构造索引类型,索引是之前的索引也就是 Key in keyof Obj,但要做一些过滤,也就是 as 之后的部分。

      过滤的方式就是单独取出该索引之后,判断空对象是否是其子类型。

      这里的 Pick 是 ts 提供的内置高级类型,就是取出某个 Key 构造新的索引类型:

      ts
      type Pick<T, K extends keyof T> = { [P in K]: T[P] }

      比如单独取出 age 构造的新的索引类型是这样的:

      可选的意思是这个索引可能没有,没有的时候,那 Pick<Obj, Key> 就是空的,所以 {} extends Pick<Obj, Key> 就能过滤出可选索引。

      值的类型依然是之前的,也就是 Obj[Key]。

      这样,就能过滤出所有可选索引,构造成新的索引类型:

      总结

      • any 类型与任何类型的交叉都是 any,也就是 1 & any 结果是 any,可以用这个特性判断 any 类型。
      • 联合类型作为类型参数出现在条件类型左侧时,会分散成单个类型传入,最后合并。
      • never 作为类型参数出现在条件类型左侧时,会直接返回 never。
      • any 作为类型参数出现在条件类型左侧时,会直接返回 trueType 和 falseType 的联合类型。
      • 元组类型也是数组类型,但 length 是数字字面量,而数组的 length 是 number。可以用来判断元组类型。
      • 函数参数处会发生逆变,可以用来实现联合类型转交叉类型。
      • 可选索引的索引可能没有,那 Pick 出来的就可能是 {},可以用来过滤可选索引,反过来也可以过滤非可选索引。
      • 索引类型的索引为字符串字面量类型,而可索引签名不是,可以用这个特性过滤掉可索引签名。
      • keyof 只能拿到 class 的 public 的索引,可以用来过滤出 public 的属性。
      • 默认推导出来的不是字面量类型,加上 as const 可以推导出字面量类型,但带有 readonly 修饰,这样模式匹配的时候也得加上 readonly 才行。

      Released under the MIT License.

      - diff --git a/src/article/typescript/pattern.html b/src/article/typescript/pattern.html index 6c0a2f91e..0a74373b7 100644 --- a/src/article/typescript/pattern.html +++ b/src/article/typescript/pattern.html @@ -37,7 +37,7 @@ - + @@ -169,8 +169,8 @@ : never : never

      类型参数 Props 为待处理的类型。

      通过 keyof Props 取出 Props 的所有索引构成的联合类型,判断下 ref 是否在其中,也就是 'ref' extends keyof Props。

      为什么要做这个判断,上面注释里写了:

      在 ts3.0 里面如果没有对应的索引,Obj[Key] 返回的是 {} 而不是 never,所以这样做下兼容处理。

      如果有 ref 这个索引的话,就通过 infer 提取 Value 的类型返回,否则返回 never。

      ts
      type GetPropsRefResult = GetPropsRef<{ ref: 1; name: 'str' }>
       // type GetPropsRefResult = 1

      当 ref 为 undefined 时:

      ts
      type GetPropsRefResult = GetPropsRef<{ ref: undefined; name: 'str' }>
      -// type GetPropsRefResult = undefined

    Released under the MIT License.

- diff --git a/src/article/typescript/reconstruction.html b/src/article/typescript/reconstruction.html index ea08b8aec..b299dd6f7 100644 --- a/src/article/typescript/reconstruction.html +++ b/src/article/typescript/reconstruction.html @@ -37,7 +37,7 @@ - + @@ -109,8 +109,8 @@ [Key in keyof T]-?: T[Key] }

给索引类型 T 的索引去掉 ? 的修饰 ,其余保持不变。

FilterByValueType

可以在构造新索引类型的时候根据值的类型做下过滤:

ts
type FilterByValueType<Obj extends Record<string, any>, ValueType> = {
   [Key in keyof Obj as Obj[Key] extends ValueType ? Key : never]: Obj[Key]
-}

类型参数 Obj 为要处理的索引类型,通过 extends 约束为索引为 string,值为任意类型的索引类型 Record<string, any>。

类型参数 ValueType 为要过滤出的值的类型。

构造新的索引类型,索引为 Obj 的索引,也就是 Key in keyof Obj,但要做一些变换,也就是 as 之后的部分。

如果原来索引的值 Obj[Key] 是 ValueType 类型,索引依然为之前的索引 Key,否则索引设置为 never,never 的索引会在生成新的索引类型时被去掉。

值保持不变,依然为原来索引的值,也就是 Obj[Key]。

这样就达到了过滤索引类型的索引,产生新的索引类型的目的:

- diff --git a/src/article/typescript/recursion.html b/src/article/typescript/recursion.html index 3f26a4095..65362277b 100644 --- a/src/article/typescript/recursion.html +++ b/src/article/typescript/recursion.html @@ -37,7 +37,7 @@ - + @@ -145,8 +145,8 @@ : DeepReadonly<Obj[Key]> : Obj[Key] } - : never - diff --git a/src/article/typescript/unionType.html b/src/article/typescript/unionType.html index 052f2b3c6..1796d923b 100644 --- a/src/article/typescript/unionType.html +++ b/src/article/typescript/unionType.html @@ -37,7 +37,7 @@ - + @@ -78,8 +78,8 @@ | `${A}${B}` | `${B}${A}`

然后构造出来的字符串再和其他字符串组合。

所以全组合的高级类型就是这样:

ts
type AllCombinations<A extends string, B extends string = A> = A extends A
   ? Combination<A, AllCombinations<Exclude<B, A>>>
-  : never

类型参数 A、B 是待组合的两个联合类型,B 默认是 A 也就是同一个。

A extends A 的意义就是让联合类型每个类型单独传入做处理,上面我们刚学会。

A 的处理就是 A 和 B 中去掉 A 以后的所有类型组合,也就是 Combination<A, B 去掉 A 以后的所有组合>。

而 B 去掉 A 以后的所有组合就是 AllCombinations<Exclude<B, A>>,所以全组合就是 Combination<A, AllCombinations<Exclude<B, A>>>。

总结

联合类型中的每个类型都是相互独立的,TypeScript 对它做了特殊处理,也就是遇到字符串类型、条件类型的时候会把每个类型单独传入做计算,最后把每个类型的计算结果合并成联合类型。

条件类型左边是联合类型的时候就会触法这种处理,叫做分布式条件类型。

有两点特别要注意:

我们后面做了一些案例,发现联合类型的这种 distributive 的特性确实能简化类型编程,但是也增加了认知成本,不过这也是不可避免的事。

- diff --git a/src/ranui/button/index.html b/src/ranui/button/index.html index 0f3c104d0..45eecc4e4 100644 --- a/src/ranui/button/index.html +++ b/src/ranui/button/index.html @@ -37,7 +37,7 @@ - + @@ -57,8 +57,8 @@ <r-button type="text" disabled>文本按钮</r-button> <r-button disabled>默认按钮</r-button>

图标icon

当需要在 Button 内嵌入 Icon 时,可以设置 icon 属性,或者直接在 Button 内使用 Icon 组件。

如果想控制 Icon 具体的位置,只能直接使用 Icon 组件,而非 icon 属性。

默认按钮
主要按钮
xml
<r-button type="default" icon="user">默认按钮</r-button>
 <r-button type="primary" icon="home">主要按钮</r-button>

特效 effect

如果需要纯净的 Button ,可以加上 effect = false ,屏蔽点击时候的水波纹特效

默认按钮主要按钮
xml
<r-button type="default" icon="user">默认按钮</r-button>
-<r-button type="primary" icon="home">主要按钮</r-button>
- diff --git a/src/ranui/icon/index.html b/src/ranui/icon/index.html index 745869f11..ff89f6d80 100644 --- a/src/ranui/icon/index.html +++ b/src/ranui/icon/index.html @@ -37,7 +37,7 @@ - + @@ -60,8 +60,8 @@ <r-icon name="lock" size="50" color="#F44336"></r-icon> <r-icon name="lock" size="50" color="#3F51B5"></r-icon>

旋转spin

设置 spin 开启旋转,传入数字控制旋转的速度,数字越小旋转越快

html
<r-icon name="loading" size="50" color="#1E90FF" spin="0.7"></r-icon>
 <r-icon name="loading" size="50" color="#1E90FF" spin></r-icon>
-<r-icon name="loading" size="50" color="#1E90FF" spin="5"></r-icon>

图标列表

- diff --git a/src/ranui/image/index.html b/src/ranui/image/index.html index f5b339678..77f0f28fd 100644 --- a/src/ranui/image/index.html +++ b/src/ranui/image/index.html @@ -37,7 +37,7 @@ - + @@ -49,8 +49,8 @@ -
Skip to content
On this page

Image 图片

代码演示

xml
<r-img src="" fallback=""></r-img>

属性

图片加载地址src

图片的地址

图片加载失败fallback

src配置的图片加载失败,兜底的图片地址,下面是默认加载失败图片

Released under the MIT License.

- diff --git a/src/ranui/index.html b/src/ranui/index.html index 0932410ee..afc54a44f 100644 --- a/src/ranui/index.html +++ b/src/ranui/index.html @@ -37,7 +37,7 @@ - + @@ -139,8 +139,8 @@ function change(e) { console.log('e--->', e) } -</script>

请注意,使用on属性来定义事件处理程序有一些限制和缺点。例如,你不能使用事件捕获或事件委托,而且每个事件类型都需要一个单独的属性。这也是为什么现代的web开发推荐使用addEventListener方法的原因。

Compatibility 兼容性

Contributors 贡献者

Other 相关资源

  1. 优秀的组件设计
  2. 在线生成 CSS 渐变色
  3. 优秀设计作品,有 psd 和 sketch
  4. 3D UI 设计,类似于 3D 版的 figma
  5. 设计规范
  6. 优秀设计作品
  7. element UI 中文网
  8. Ant design 中文网
  9. 在线绘制 CSS 动画
  10. tailwindcss 组件库
  11. animate css 非常优秀的 css 动画
  12. can i use 检测兼容性 API 网站
  13. figma

协议和标准

  1. RFCs
  2. ECMA
  3. w3c
- diff --git a/src/ranui/input/index.html b/src/ranui/input/index.html index aae8ba1bb..02612124c 100644 --- a/src/ranui/input/index.html +++ b/src/ranui/input/index.html @@ -37,7 +37,7 @@ - + @@ -59,8 +59,8 @@ const func = (e) => { console.log(e) } -input.addEventListener('input', func)

事件的e参数结构 input方法

- diff --git a/src/ranui/message/index.html b/src/ranui/message/index.html index bd9e9c7f6..793c17547 100644 --- a/src/ranui/message/index.html +++ b/src/ranui/message/index.html @@ -37,7 +37,7 @@ - + @@ -53,8 +53,8 @@ <r-button onclick="message.warning('这是一条提示')">警告提示</r-button> <r-button onclick="message.error('这是一条提示')">错误提示</r-button> <r-button onclick="message.success('这是一条提示')">成功提示</r-button> -<r-button onclick="message.toast('这是一条提示')">toast提示</r-button>

方法

组件提供了一些静态方法,使用方式和参数如下:

  1. 可以只传一个参数,提示的内容,默认提示 3000 毫秒

message.info('这是一条提示')

message.warning('这是一条提示')

message.error('这是一条提示')

message.success('这是一条提示')

message.toast('这是一条提示')"

  1. 也可以传一个对象,设置提示内容,关闭延时,关闭时触发的回调函数

message.info({content:'这是一条提示', duration: 2000, close: () => {}})

message.warning({content:'这是一条提示', duration: 2000, close: () => {}})

message.error({content:'这是一条提示', duration: 2000, close: () => {}})

message.success({content:'这是一条提示', duration: 2000, close: () => {}})

message.toast({content:'这是一条提示', duration: 2000, close: () => {}})

参数说明类型
content提示内容string
duration自动关闭的延时,单位毫秒。默认 3000 毫秒number
close关闭时触发的回调函数() => void
- diff --git a/src/ranui/modal/index.html b/src/ranui/modal/index.html index 4f2c90d82..ae69da6af 100644 --- a/src/ranui/modal/index.html +++ b/src/ranui/modal/index.html @@ -37,7 +37,7 @@ - + @@ -49,8 +49,8 @@ -
Skip to content
On this page

Released under the MIT License.

- diff --git a/src/ranui/preview/index.html b/src/ranui/preview/index.html index f948b0cb5..8c1620f63 100644 --- a/src/ranui/preview/index.html +++ b/src/ranui/preview/index.html @@ -37,7 +37,7 @@ - + @@ -70,8 +70,8 @@ } } } -</script>

属性

资源地址src

src 地址即可打开弹窗,没有src就不展示

html
<r-preview src=""></r-preview>

是否可关闭closeable

closeable 默认为 true ,可以关闭,设置成 false 时, 表示不可关闭,将不会展示右上角的关闭按钮

html
<r-preview closeable="false"></r-preview>
- diff --git a/src/ranui/skeleton/index.html b/src/ranui/skeleton/index.html index 0d362caa9..fbf51e6bf 100644 --- a/src/ranui/skeleton/index.html +++ b/src/ranui/skeleton/index.html @@ -37,7 +37,7 @@ - + @@ -49,8 +49,8 @@ -
Skip to content
On this page

skeleton 骨架屏

在需要等待加载内容的位置提供一个占位图形组合。

代码演示

骨架长度跟随父级元素的长度

xml
<r-skeleton ></r-skeleton>

Released under the MIT License.

- diff --git a/src/ranui/tab/index.html b/src/ranui/tab/index.html index 7714e3537..de2640777 100644 --- a/src/ranui/tab/index.html +++ b/src/ranui/tab/index.html @@ -37,7 +37,7 @@ - + @@ -91,8 +91,8 @@ <r-tab label="tab1">11111</r-tab> <r-tab label="tab2">22222</r-tab> <r-tab label="tab3">33333</r-tab> - </r-tabs> - diff --git a/src/ranui/tabs/index.html b/src/ranui/tabs/index.html index c61282dd1..8468d293f 100644 --- a/src/ranui/tabs/index.html +++ b/src/ranui/tabs/index.html @@ -37,7 +37,7 @@ - + @@ -73,8 +73,8 @@ <r-tab icon="home" iconSize="22">tab1</r-tab> <r-tab icon="message" iconSize="22">tab2</r-tab> <r-tab icon="user" iconSize="22">tab3</r-tab> -</r-tabs>

风格type

风格有 text,clean,

对齐align

事件event

onchange

切换完成时触发。

- diff --git a/src/ranuts/binaryTree/index.html b/src/ranuts/binaryTree/index.html index c80407d2b..264fcc7b5 100644 --- a/src/ranuts/binaryTree/index.html +++ b/src/ranuts/binaryTree/index.html @@ -37,7 +37,7 @@ - + @@ -49,8 +49,8 @@ -
Skip to content
On this page

二叉树的定义

在计算机科学中,二叉树(Binary tree)是每个节点最多只有两个分支(即不存在分支度大于 2 的节点)的树结构。通常分支被称作“左子树”或“右子树”。二叉树的分支具有左右次序,不能随意颠倒[1]。。

二叉树的性质

  • 在二叉树的第 i 层上最多有 2^(i-1)个结点(i>=1)
  • 深度为 h 的二叉树,最多有 2^h-1 个结点,最少有 h 个结点(h>=1)
  • 包含 n 个结点的二叉树的高度至少为(log2n)+1
  • 非空的二叉树,分支度为 0 的总数为 n0,分支度为 2 的总数为 n2,则 n0=n2+1
  • 二叉树的总结点数 n = n1 + n2 + n0
  • 总连线数等于总节点数减一(B = n - 1)
  • 总连线数等于分支度为 2 的节点的两倍加上分支度为 1 的节点(B = n2 _ 2 + n1 _ 1)

二叉树的类型

满二叉树

一棵深度为 k 且有 2k-1 个节点的二叉树称为满二叉树。 除最后一层无任何子节点外,每一层上的所有结点都有两个子结点的二叉树[2]

完全二叉树

一棵深度为 k 的有 n 个结点的二叉树,对树中的结点按从上至下、从左到右的顺序进行编号,如果编号为 i(1≤i≤n)的结点与满二叉树中编号为 i 的结点在二叉树中的位置相同,则这棵二叉树称为完全二叉树。

二叉搜索树

二叉搜索树(BST)又称二叉查找树或二叉排序树。它或者是一棵空树,或者是具有下列性质的二叉树: 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值; 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值; 它的左、右子树也分别为二叉排序树。

平衡二叉树

平衡二叉树(AVL)一定是二叉搜索树,且左子树和右子树的高度差的绝对值不超过 1。 平衡二叉树

B 树

B 树属于多叉树又名平衡多路查找树(查找路径不只两个)

B+树

B+树是 B 树的变体,也是一种多路搜索树。

B*树

B* 树是 B+树的变体,在 B+树的非根和非叶子结点再增加指向兄弟的指针;B* 树定义了非叶子结点关键字个数至少为(2/3)M,即块的最低使用率为 2/3(代替 B+树的 1/2)。B 树分配新结点的概率比 B+树要低,空间使用率更高;

红黑树

红黑树是一种平衡二叉查找树的变体,它的左右子树高差有可能大于 1,所以红黑树不是严格意义上的平衡二叉树(AVL),但对它进行平衡的代价较低, 其平均统计性能要强于 AVL 。

遍历

前序遍历

后序遍历

中序遍历

层序遍历

常见算法题

镜像二叉树

重建二叉树

二叉树深度

二叉树节点总数

判断二叉树子结构

输入两棵二叉树 A 和 B,判断 B 是不是 A 的子结构。(ps:约定空树不是任意一个树的子结构)

参考文档

  1. 维基百科二叉树
  2. 百度百科满二叉树

Released under the MIT License.

- diff --git a/src/ranuts/bundler/index.html b/src/ranuts/bundler/index.html index 3b30f5ab5..93243b08d 100644 --- a/src/ranuts/bundler/index.html +++ b/src/ranuts/bundler/index.html @@ -37,7 +37,7 @@ - + @@ -58,8 +58,8 @@ generate: () => bundle.render() }; }); -}

架构图

- diff --git a/src/ranuts/file/index.html b/src/ranuts/file/index.html index 52e57774f..cf4230f98 100644 --- a/src/ranuts/file/index.html +++ b/src/ranuts/file/index.html @@ -37,7 +37,7 @@ - + @@ -49,8 +49,8 @@ -
Skip to content
On this page

文件操作函数

watchFile

观察一个文件是否改变

API

Return

  • Promise
参数说明类型描述
status文件是否被改变booleantrue 文件改变 false 文件没变

Options

参数说明类型默认值
path文件路径,需要监听的文件stringundefined
interval监听文件改变的时间,单位毫秒。number20

Released under the MIT License.

- diff --git a/src/ranuts/index.html b/src/ranuts/index.html index 02d21b6ad..e8a1ecfce 100644 --- a/src/ranuts/index.html +++ b/src/ranuts/index.html @@ -37,7 +37,7 @@ - + @@ -49,8 +49,8 @@ -
Skip to content
On this page

Released under the MIT License.

- diff --git a/src/ranuts/sort/bubble/index.html b/src/ranuts/sort/bubble/index.html index f0e7c9bd6..89b562e84 100644 --- a/src/ranuts/sort/bubble/index.html +++ b/src/ranuts/sort/bubble/index.html @@ -37,7 +37,7 @@ - + @@ -62,8 +62,8 @@ } } return arr -} - diff --git a/src/ranuts/sort/bucket/index.html b/src/ranuts/sort/bucket/index.html index f9841f9f7..c4d0cef8f 100644 --- a/src/ranuts/sort/bucket/index.html +++ b/src/ranuts/sort/bucket/index.html @@ -37,7 +37,7 @@ - + @@ -110,8 +110,8 @@ list = list.concat(count(buckets[i])) } return list -}

算法分析

桶排序最好情况下使用线性时间 O(n),桶排序的时间复杂度,取决与对各个桶之间数据进行排序的时间复杂度,因为其它部分的时间复杂度都为 O(n)。很显然,桶划分的越小,各个桶之间的数据越少,排序所用的时间也会越少。但相应的空间消耗就会增大。

- diff --git a/src/ranuts/sort/count/index.html b/src/ranuts/sort/count/index.html index 8ac6b6059..14391cbe9 100644 --- a/src/ranuts/sort/count/index.html +++ b/src/ranuts/sort/count/index.html @@ -37,7 +37,7 @@ - + @@ -81,8 +81,8 @@ } } return list -}

算法分析

计数排序是一个稳定的排序算法。当输入的元素是 n 个 0 到 k 之间的整数时,时间复杂度是 O(n+k),空间复杂度也是 O(n+k),其排序速度快于任何比较排序算法。当 k 不是很大并且序列比较集中时,计数排序是一个很有效的排序算法。

- diff --git a/src/ranuts/sort/heap/index.html b/src/ranuts/sort/heap/index.html index d13d850cc..03e6bb1c3 100644 --- a/src/ranuts/sort/heap/index.html +++ b/src/ranuts/sort/heap/index.html @@ -37,7 +37,7 @@ - + @@ -97,8 +97,8 @@ const heap = (list: Array<number>): Array<number> => { const { arr } = new Heap(list) return arr -} - diff --git a/src/ranuts/sort/index.html b/src/ranuts/sort/index.html index 3265de39e..d02a21a38 100644 --- a/src/ranuts/sort/index.html +++ b/src/ranuts/sort/index.html @@ -37,7 +37,7 @@ - + @@ -49,8 +49,8 @@ -
Skip to content
On this page

十大经典排序

十种常见排序算法可以分为两大类:

  • 比较类排序:通过比较来决定元素间的相对次序,由于其时间复杂度不能突破 O(nlogn),因此也称为非线性时间比较类排序。
  • 非比较类排序:不通过比较来决定元素间的相对次序,它可以突破基于比较排序的时间下界,以线性时间运行,因此也称为线性时间非比较类排序。 排序分类 0.2 算法复杂度

算法复杂度 0.3 相关概念

  • 稳定:如果 a 原本在 b 前面,而 a=b,排序之后 a 仍然在 b 的前面。
  • 不稳定:如果 a 原本在 b 的前面,而 a=b,排序之后 a 可能会出现在 b 的后面。
  • 时间复杂度:对排序数据的总的操作次数。反映当 n 变化时,操作次数呈现什么规律。
  • 空间复杂度:是指算法在计算机内执行时所需存储空间的度量,它也是数据规模 n 的函数。

Released under the MIT License.

- diff --git a/src/ranuts/sort/insert/index.html b/src/ranuts/sort/insert/index.html index 9ebb2b031..affca068b 100644 --- a/src/ranuts/sort/insert/index.html +++ b/src/ranuts/sort/insert/index.html @@ -37,7 +37,7 @@ - + @@ -61,8 +61,8 @@ list[preIndex + 1] = current } return list -}

算法分析

插入排序在实现上,通常采用 in-place 排序(即只需用到 O(1)的额外空间的排序),因而在从后向前扫描过程中,需要反复把已排序元素逐步向后挪位,为最新元素提供插入空间。

- diff --git a/src/ranuts/sort/merge/index.html b/src/ranuts/sort/merge/index.html index f45891be6..11e03137d 100644 --- a/src/ranuts/sort/merge/index.html +++ b/src/ranuts/sort/merge/index.html @@ -37,7 +37,7 @@ - + @@ -81,8 +81,8 @@ const left = list.slice(0, middle) const right = list.slice(middle) return combine(merge(left), merge(right)) -}

算法分析

归并排序是一种稳定的排序方法。和选择排序一样,归并排序的性能不受输入数据的影响,但表现比选择排序好的多,因为始终都是 O(nlogn)的时间复杂度。代价是需要额外的内存空间。

- diff --git a/src/ranuts/sort/quick/index.html b/src/ranuts/sort/quick/index.html index 507efc15b..1eebee087 100644 --- a/src/ranuts/sort/quick/index.html +++ b/src/ranuts/sort/quick/index.html @@ -37,7 +37,7 @@ - + @@ -96,8 +96,8 @@ const quick = (list: Array<number>): Array<number> => { const { length } = list return combine(list, 0, length - 1) -} - diff --git a/src/ranuts/sort/radix/index.html b/src/ranuts/sort/radix/index.html index 003a55824..670cc28dd 100644 --- a/src/ranuts/sort/radix/index.html +++ b/src/ranuts/sort/radix/index.html @@ -37,7 +37,7 @@ - + @@ -84,8 +84,8 @@ list = list.concat(count(buckets[i])) } return list -}

算法分析

基数排序基于分配排序,所以是稳定的。但基数排序的性能比桶排序要略差,每一次关键字的桶分配都需要 O(n)的时间复杂度,而且分配之后得到新的关键字序列又需要 O(n)的时间复杂度。假如待排数据可以分为 d 个关键字,则基数排序的时间复杂度将是 O(d*2n) ,当然 d 要远远小于 n,因此基本上还是线性级别的。

基数排序的空间复杂度为 O(n+k),其中 k 为桶的数量。一般来说 n>>k,因此额外空间需要大概 n 个左右。

- diff --git a/src/ranuts/sort/select/index.html b/src/ranuts/sort/select/index.html index f54b1f19e..05628993a 100644 --- a/src/ranuts/sort/select/index.html +++ b/src/ranuts/sort/select/index.html @@ -37,7 +37,7 @@ - + @@ -65,8 +65,8 @@ arr[minIndex] = temp } return arr -}

算法分析

表现最稳定的排序算法之一,因为无论什么数据进去都是 O(n2)的时间复杂度,所以用到它的时候,数据规模越小越好。唯一的好处可能就是不占用额外的内存空间了吧。理论上讲,选择排序可能也是平时排序一般人想到的最多的排序方法了吧。

- diff --git a/src/ranuts/sort/shell/index.html b/src/ranuts/sort/shell/index.html index 939183bbd..6084aeb7c 100644 --- a/src/ranuts/sort/shell/index.html +++ b/src/ranuts/sort/shell/index.html @@ -37,7 +37,7 @@ - + @@ -68,8 +68,8 @@ } } return list -}

算法分析

希尔排序的核心在于间隔序列的设定。既可以提前设定好间隔序列,也可以动态的定义间隔序列。动态定义间隔序列的算法是《算法(第 4 版)》的合著者 Robert Sedgewick 提出的。

- diff --git a/src/ranuts/utils/index.html b/src/ranuts/utils/index.html index 08a55ad56..4503edf1c 100644 --- a/src/ranuts/utils/index.html +++ b/src/ranuts/utils/index.html @@ -37,7 +37,7 @@ - + @@ -49,8 +49,8 @@ -
Skip to content
On this page

通用函数

filterObj

过滤对象的属性,去除对象中在 list 数组里面有的属性,返回一个新对象,一般是用于去除空字符和 null

API

Return

参数说明类型
run启动轮询(...params: TParams) => void
runAsync启动轮询(...params: TParams) => Promise<TData>
cancel停止轮询() => void

Options

参数说明类型默认值
pollingInterval轮询间隔,单位为毫秒。如果值大于 0,则启动轮询模式。number0
pollingWhenHidden在页面隐藏时,是否继续轮询。如果设置为 false,在页面隐藏时会暂时停止轮询,页面重新显示时继续上次轮询。booleantrue
pollingErrorRetryCount轮询错误重试次数。如果设置为 -1,则无限次number-1

Released under the MIT License.

- diff --git a/src/ranuts/utils/task.html b/src/ranuts/utils/task.html index 79d9ef9aa..779165a68 100644 --- a/src/ranuts/utils/task.html +++ b/src/ranuts/utils/task.html @@ -37,7 +37,7 @@ - + @@ -55,8 +55,8 @@ const time = taskEnd(taskId) -console.log('task 执行花费的时间', time)

二.new Date().getTime()

new Date().getTime() 返回一个数值,表示从 1970 年 1 月 1 日 0 时 0 分 0 秒(UTC,即协调世界时)距离该日期对象所代表时间的毫秒数。用来计算 JS 执行时间会有两个问题:

  1. 某些情况下,毫秒级精度可能不够。
  2. new Date() 解析的时间在不同浏览器,或者不同设备上可能并不一致。MDN 说明

    由于浏览器之间的差异与不一致性,强烈不推荐使用 Date 构造函数来解析日期字符串 (或使用与其等价的 Date.parse)。对 RFC 2822 格式的日期仅有约定俗成的支持。对 ISO 8601 格式的支持中,仅有日期的串 (例如 "1970-01-01") 会被处理为 UTC 而不是本地时间,与其他格式的串的处理不同。

三.console.time(), console.timeEnd()

启动一个计时器来跟踪某一个操作的占用时长。每一个计时器必须拥有唯一的名字,页面中最多能同时运行 10,000 个计时器。当以此计时器名字为参数调用 console.timeEnd() 时,浏览器将以毫秒为单位,输出对应计时器所经过的时间。比起new Date().getTime(),统计时间更加精确,可以统计到 0.001 毫秒(比如:0.134ms)

四.performance.now()

performance.now()返回的时间精度最高可达微秒级,且不会受到系统时间的影响(系统时钟可能会被手动调整或被 NTP 等软件篡改)。另外,performance.timing.navigationStart + performance.now() 约等于 Date.now()。因此对于统计 JS 执行耗时方面,更推荐使用performance.now()

注意:为了提供对定时攻击和指纹的保护,performance.now() 的精度可能会根据浏览器的设置而被舍弃。 在 Firefox 中,privacy.reduceTimerPrecision 偏好是默认启用的,默认值为 1ms。可以启用 privacy.resistFingerprinting 这将精度改为 100ms 或privacy.resistFingerprinting.reduceTimerPrecision.microseconds 的值,以较大者为准。

- diff --git "a/src/types/TS\347\261\273\345\236\213.html" "b/src/types/TS\347\261\273\345\236\213.html" index 42000a088..a3f53a37c 100644 --- "a/src/types/TS\347\261\273\345\236\213.html" +++ "b/src/types/TS\347\261\273\345\236\213.html" @@ -37,7 +37,7 @@ - + @@ -96,8 +96,8 @@ age?: number } -type tuple = [string, number?] - diff --git "a/src/types/\346\250\241\345\274\217\345\214\271\351\205\215.html" "b/src/types/\346\250\241\345\274\217\345\214\271\351\205\215.html" index af09b36b1..f1d911ad6 100644 --- "a/src/types/\346\250\241\345\274\217\345\214\271\351\205\215.html" +++ "b/src/types/\346\250\241\345\274\217\345\214\271\351\205\215.html" @@ -37,7 +37,7 @@ - + @@ -90,8 +90,8 @@ interface PersonConstructor { new (name: string): Person }

这里的 PersonConstructor 返回的是 Person 类型的实例对象,这个也可以通过模式匹配取出来。

ts
type GetInstanceType<C extends new (...args: unknown[]) => unknown> =
-  C extends new (...args: unknown[]) => infer T ? T : unknown
- diff --git "a/src/types/\347\261\273\345\236\213\350\277\220\347\256\227.html" "b/src/types/\347\261\273\345\236\213\350\277\220\347\256\227.html" index 08c9497c2..4151fe26e 100644 --- "a/src/types/\347\261\273\345\236\213\350\277\220\347\256\227.html" +++ "b/src/types/\347\261\273\345\236\213\350\277\220\347\256\227.html" @@ -37,7 +37,7 @@ - + @@ -79,8 +79,8 @@ // type res = { // aaa:[1,1,1] // bbb:[2,2,2] -// }

这里的 & string 可能大家会迷惑,解释一下:

因为索引类型(对象、class 等)可以用 string、number 和 symbol 作为 key,这里 keyof T 取出的索引就是 string | number | symbol 的联合类型,和 string 取交叉部分就只剩下 string 了。就像前面所说,交叉类型会把同一类型做合并,不同类型舍弃。

因为 js 处理对象比较多,所以索引类型的映射比较重要。

- diff --git "a/src/types/\351\253\230\347\272\247\347\261\273\345\236\213.html" "b/src/types/\351\253\230\347\272\247\347\261\273\345\236\213.html" index 0a7122325..23f69e2b4 100644 --- "a/src/types/\351\253\230\347\272\247\347\261\273\345\236\213.html" +++ "b/src/types/\351\253\230\347\272\247\347\261\273\345\236\213.html" @@ -37,7 +37,7 @@ - + @@ -49,8 +49,8 @@ -
Skip to content
On this page

TypeScript 内置的高级类型

Parameters

Parameters 用于提取函数类型的参数类型。

ReturnType

ReturnType 用于提取函数类型的返回值类型。

ConstructorParameters

构造器类型和函数类型的区别就是可以被 new。

Parameters 用于提取函数参数的类型,而 ConstructorParameters 用于提取构造器参数的类型。

InstanceType

提取了构造器参数的类型,自然也可以提取构造器返回值的类型,就是 InstanceType。

ThisParameterType

OmitThisParameter

Partial

Required

Readonly

Pick

Record

Exclude

Extract

Omit

Awaited

NonNullable

Uppercase

Lowercase

Capitalize

Uncapitalize

总结

比如用模式匹配可以实现:Parameters、ReturnType、ConstructorParameters、InstanceType、ThisParameterType。

用模式匹配 + 重新构造可以实现:OmitThisParameter

用重新构造可以实现:Partial、Required、Readonly、Pick、Record

用模式匹配 + 递归可以实现: Awaited

用联合类型在分布式条件类型的特性可以实现: Exclude

此外还有 NonNullable 和四个编译器内部实现的类型:Uppercase、Lowercase、Capitalize、Uncapitalize。

Last updated:

Released under the MIT License.

-