记录
每当遇到问题或者可以改进的事情的时候,会把想法先写下来,如果这是你遇到的问题,那么很可能有一群人也遇到同样的困难。
diff --git a/404.html b/404.html index 15eecf1cf..65ab9b2a7 100644 --- a/404.html +++ b/404.html @@ -20,7 +20,7 @@
404
But if you don't change your direction, and if you keep looking, you may end up where you are heading.
abstract syntax tree
)抽象语法树的作用 源码是一串按照语法格式来组织的字符串,人能够认识,但是计算机并不认识,想让计算机认识就要转成一种数据结构,通过不同的对象来保存不同的数据,并且按照依赖关系组织起来,这种数据结构就是抽象语法树(abstract syntax tree
)。
之所以叫“抽象”语法树是因为数据结构中省略掉了一些无具体意义的分隔符比如 ; { }
等。
有了 AST
,计算机就能理解源码字符串的意思,而理解是能够转换的前提,所以编译的第一步需要把源码 parse
成 AST
。
转成 AST
之后就可以通过修改 AST
,分析 AST
的方式来修改和分析代码,比如 babel
就通过这种方式进行代码的转换,比如 rollup
的 Tree Shaking
,就是通过分析 AST
的 导入导出语法,从而分析出没有使用的代码,进行去除。
常见的 AST 节点 AST 是对源码的抽象,字面量、标识符、表达式、语句、模块语法、class 语法都有各自的 AST。
我们分别来了解一下:
Literal
是字面量的意思,比如 let name = 'value'
中,'value'
就是一个字符串字面量 StringLiteral
,相应的还有数字字面量 NumericLiteral
,布尔字面量 BooleanLiteral
,字符串字面量 StringLiteral
,正则表达式字面量 RegExpLiteral
等。
下面这些字面量都有对应的 Literal
节点:
代码中的字面量很多,babel
就是通过 xxLiteral
来抽象这部分内容的。
Identifer
是标识符的意思,变量名、属性名、参数名等各种声明和引用的名字,都是Identifer
。
我们知道, JS
中的标识符只能包含字母或数字或下划线 (“_”)
或美元符号 (“$”)
,且不能以数字开头。这是 Identifier
的词法特点。
尝试分析一下,下面这一段代码里面有多少 Identifier
呢?
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
,计算机就能理解源码字符串的意思,而理解是能够转换的前提,所以编译的第一步需要把源码 parse
成 AST
。
转成 AST
之后就可以通过修改 AST
,分析 AST
的方式来修改和分析代码,比如 babel
就通过这种方式进行代码的转换,比如 rollup
的 Tree 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
呢?
jsconst 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
jsdocument.addEventListener('DOMContentLoaded', fun)
<body onload="fun()">
readyState
jsdocument.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
jsdocument.addEventListener('DOMContentLoaded', fun)
<body onload="fun()">
readyState
jsdocument.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 就是数值。
比如:
tstype 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 就是数值。
比如:
tstype 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 函数既可以做整数加法、又可以做浮点数加法,却需要声明两个函数:
cint 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 函数既可以做整数加法、又可以做浮点数加法,却需要声明两个函数:
cint 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 类型:
tstype p = Promise<'value'>
我们想提取 value 的类型,可以这样做:
tstype 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.数组类型
提取第一个元素
数组类型想提取第一个元素的类型怎么做呢?
tstype arr = [1, 2, 3]
用它来匹配一个模式类型,提取第一个元素的类型到通过 infer 声明的局部变量里返回。
tstype 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 叫做类型别名,其实就是声明一个变量存储某个类型:
tstype ttt = Promise<number>
infer 用于类型的提取,然后存到一个变量里,相当于局部变量:
tstype GetValueType<P> = P extends Promise<infer Value> ? Value : never
类型参数用于接受具体的类型,在类型运算中也相当于局部变量:
tstype isTwo<T> = T extends 2 ? true : false
但是,严格来说这三种也都不叫变量,因为它们不能被重新赋值。
TypeScript 设计可以做类型编程的类型系统的目的就是为了产生各种复杂的类型,那不能修改怎么产生新类型呢?
答案是重新构造。
这就涉及到了第二个类型体操套路:重新构造做变换。
重新构造
TypeScript 的 type、infer、类型参数声明的变量都不能修改,想对类型做各种变换产生新的类型就需要重新构造。
数组、字符串、函数等类型的重新构造比较简单。
索引类型,也就是多个元素的聚合类型的重新构造复杂一些,涉及到了映射类型的语法。
我们先从简单的开始:
数组类型的重新构造
Push
有这样一个元组类型:
tstype tuple = [1, 2, 3]
我想给这个元组类型再添加一些类型,怎么做呢?
TypeScript 类型变量不支持修改,我们可以构造一个新的元组类型:
tstype Push<Arr extends unknown[], Ele> = [...Arr, Ele]
类型参数 Arr 是要修改的数组/元组类型,元素的类型任意,也就是 unknown。
类型参数 Ele 是添加的元素的类型。
返回的是用 Arr 已有的元素加上 Ele 构造的新的元组类型。
tstype 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 叫做类型别名,其实就是声明一个变量存储某个类型:
tstype ttt = Promise<number>
infer 用于类型的提取,然后存到一个变量里,相当于局部变量:
tstype GetValueType<P> = P extends Promise<infer Value> ? Value : never
类型参数用于接受具体的类型,在类型运算中也相当于局部变量:
tstype isTwo<T> = T extends 2 ? true : false
但是,严格来说这三种也都不叫变量,因为它们不能被重新赋值。
TypeScript 设计可以做类型编程的类型系统的目的就是为了产生各种复杂的类型,那不能修改怎么产生新类型呢?
答案是重新构造。
这就涉及到了第二个类型体操套路:重新构造做变换。
重新构造
TypeScript 的 type、infer、类型参数声明的变量都不能修改,想对类型做各种变换产生新的类型就需要重新构造。
数组、字符串、函数等类型的重新构造比较简单。
索引类型,也就是多个元素的聚合类型的重新构造复杂一些,涉及到了映射类型的语法。
我们先从简单的开始:
数组类型的重新构造
Push
有这样一个元组类型:
tstype tuple = [1, 2, 3]
我想给这个元组类型再添加一些类型,怎么做呢?
TypeScript 类型变量不支持修改,我们可以构造一个新的元组类型:
tstype Push<Arr extends unknown[], Ele> = [...Arr, Ele]
类型参数 Arr 是要修改的数组/元组类型,元素的类型任意,也就是 unknown。
类型参数 Ele 是添加的元素的类型。
返回的是用 Arr 已有的元素加上 Ele 构造的新的元组类型。
tstype PushResult = Push<[1, 2, 3], 4>
// type PushResult = [1,2,3,4]
这就是数组/元组的重新构造。
数组和元组的区别:数组类型是指任意多个同一类型的元素构成的,比如 number[]
、Array<number>
,而元组则是数量固定,类型可以不同的元素构成的,比如 [1, true, 'name']
。
Unshift
可以在后面添加,同样也可以在前面添加:
tstype Unshift<Arr extends unknown[], Ele> = [Ele, ...Arr]
Zip
有这样两个元组:
tstype tuple1 = [1, 2]
type tuple2 = ['name', 'value']
我们想把它们合并成这样的元组:
tstype tuple = [[1, 'name'], [2, 'value']]
思路很容易想到,提取元组中的两个元素,构造成新的元组:
tstype 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 类型的高级类型。
tstype ttt = Promise<Promise<Promise<Record<string, any>>>>
这里是 3 层 Promise,value 类型是索引类型。
数量不确定,一涉及到这个就要想到用递归来做,每次只处理一层的提取,然后剩下的到下次递归做,直到结束条件。
所以高级类型是这样的:
tstype 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 类型的高级类型。
tstype ttt = Promise<Promise<Promise<Record<string, any>>>>
这里是 3 层 Promise,value 类型是索引类型。
数量不确定,一涉及到这个就要想到用递归来做,每次只处理一层的提取,然后剩下的到下次递归做,直到结束条件。
所以高级类型是这样的:
tstype 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 会把每一个元素单独传入来做类型运算,最后再合并成联合类型,这种语法叫做分布式条件类型。
比如这样一个联合类型:
tstype Union = 'a' | 'b' | 'c'
我们想把其中的 a 大写,就可以这样写:
tstype UppercaseA<Item extends string> = Item extends 'a' ? Uppercase<Item> : Item
tstype result = UppercaseA<Union>\n// type result = 'A' | 'b' | 'c';
可以看到,我们类型参数 Item 约束为 string,条件类型的判断中也是判断是否是 a,但传入的是联合类型。
这就是 TypeScript 对联合类型在条件类型中使用时的特殊处理:会把联合类型的每一个元素单独传入做类型计算,最后合并。
这和联合类型遇到字符串时的处理一样:
这样确实是简化了类型编程逻辑的,不需要递归提取每个元素再处理。
TypeScript 之所以这样处理联合类型也很容易理解,因为联合类型的每个元素都是互不相关的,不像数组、索引、字符串那样元素之间是有关系的。所以设计成了每一个单独处理,最后合并。
知道了 TypeScript 怎么处理的联合类型,趁热打铁来练习一下:
CamelcaseUnion
Camelcase 我们实现过,就是提取字符串中的字符,首字母大写以后重新构造一个新的。
tstype Camelcase<Str extends string> =\n Str extends `${infer Left}_${infer Right}${infer Rest}`\n ? `${Left}${Uppercase<Right>}${Camelcase<Rest>}`\n : Str
提取 _ 左右的字符,把右边字符大写之后构造成新的字符串,余下的字符串递归处理。
tstype CamelcaseResult = Camelcase<'aa_aa_aa'>\n// type CamelcaseResult = 'aaAaAa'
如果是对字符串数组做 Camelcase,那就要递归处理每一个元素:
tstype 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 内部会把每一个元素传入单独做计算,之后把每个元素的计算结果合并成联合类型。
tstype CamelcaseUnion<Item extends string> =\n Item extends `${infer Left}_${infer Right}${infer Rest}`\n ? `${Left}${Uppercase<Right>}${CamelcaseUnion<Rest>}`\n : Item
这不和单个字符串的处理没区别么?
没错,对联合类型的处理和对单个类型的处理没什么区别,TypeScript 会把每个单独的类型拆开传入。不需要像数组类型那样需要递归提取每个元素做处理。
确实简化了很多,好像都是优点?
也不全是,其实这样处理也增加了一些认知成本,不信我们再来看个例子:
IsUnion
判断联合类型我们会这样写:
tstype IsUnion<A, B = A> = A extends A ? ([B] extends [A] ? false : true) : never
当传入联合类型时,会返回 true:
tstype IsUnionResult = IsUnion<'a' | 'b' 'c'>\n// type IsUnionResult = true
当传入其他类型时,会返回 false:
tstype IsUnionResult = IsUnion<['a' | 'b' 'c']>\n// type IsUnionResult = false
这就是分布式条件类型带来的认知成本。
我们先来看这样一个类型:
tstype 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 类型的判断:
tstype 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 名:
这样使用:
tstype bemResult = BEM<'guang', ['aaa', 'bbb'], ['warning', 'success']>
它的实现就是三部分的合并,但传入的是数组,要递归遍历取出每一个元素来和其他部分组合,这样太麻烦了。
而如果是联合类型就不用递归遍历了,因为联合类型遇到字符串也是会单独每个元素单独传入做处理。
数组转联合类型可以这样写:
tstype union = ['aaa', 'bbb'][number]\n// type union = 'aaa' | 'bbb'
那么 BEM 就可以这样实现:
tstype 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 通过索引访问来变为联合类型。
字符串类型中遇到联合类型的时候,会每个元素单独传入计算,也就是这样的效果:
tstype 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
tstype Combination<A extends string, B extends string> =\n | A\n | B\n | `${A}${B}`\n | `${B}${A}`
然后构造出来的字符串再和其他字符串组合。
所以全组合的高级类型就是这样:
tstype 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 会把每一个元素单独传入来做类型运算,最后再合并成联合类型,这种语法叫做分布式条件类型。
比如这样一个联合类型:
tstype Union = 'a' | 'b' | 'c'
我们想把其中的 a 大写,就可以这样写:
tstype UppercaseA<Item extends string> = Item extends 'a' ? Uppercase<Item> : Item
tstype result = UppercaseA<Union>\n// type result = 'A' | 'b' | 'c';
可以看到,我们类型参数 Item 约束为 string,条件类型的判断中也是判断是否是 a,但传入的是联合类型。
这就是 TypeScript 对联合类型在条件类型中使用时的特殊处理:会把联合类型的每一个元素单独传入做类型计算,最后合并。
这和联合类型遇到字符串时的处理一样:
这样确实是简化了类型编程逻辑的,不需要递归提取每个元素再处理。
TypeScript 之所以这样处理联合类型也很容易理解,因为联合类型的每个元素都是互不相关的,不像数组、索引、字符串那样元素之间是有关系的。所以设计成了每一个单独处理,最后合并。
知道了 TypeScript 怎么处理的联合类型,趁热打铁来练习一下:
CamelcaseUnion
Camelcase 我们实现过,就是提取字符串中的字符,首字母大写以后重新构造一个新的。
tstype Camelcase<Str extends string> =\n Str extends `${infer Left}_${infer Right}${infer Rest}`\n ? `${Left}${Uppercase<Right>}${Camelcase<Rest>}`\n : Str
提取 _ 左右的字符,把右边字符大写之后构造成新的字符串,余下的字符串递归处理。
tstype CamelcaseResult = Camelcase<'aa_aa_aa'>\n// type CamelcaseResult = 'aaAaAa'
如果是对字符串数组做 Camelcase,那就要递归处理每一个元素:
tstype 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 内部会把每一个元素传入单独做计算,之后把每个元素的计算结果合并成联合类型。
tstype CamelcaseUnion<Item extends string> =\n Item extends `${infer Left}_${infer Right}${infer Rest}`\n ? `${Left}${Uppercase<Right>}${CamelcaseUnion<Rest>}`\n : Item
这不和单个字符串的处理没区别么?
没错,对联合类型的处理和对单个类型的处理没什么区别,TypeScript 会把每个单独的类型拆开传入。不需要像数组类型那样需要递归提取每个元素做处理。
确实简化了很多,好像都是优点?
也不全是,其实这样处理也增加了一些认知成本,不信我们再来看个例子:
IsUnion
判断联合类型我们会这样写:
tstype IsUnion<A, B = A> = A extends A ? ([B] extends [A] ? false : true) : never
当传入联合类型时,会返回 true:
tstype IsUnionResult = IsUnion<'a' | 'b' 'c'>\n// type IsUnionResult = true
当传入其他类型时,会返回 false:
tstype IsUnionResult = IsUnion<['a' | 'b' 'c']>\n// type IsUnionResult = false
这就是分布式条件类型带来的认知成本。
我们先来看这样一个类型:
tstype 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 类型的判断:
tstype 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 名:
这样使用:
tstype bemResult = BEM<'guang', ['aaa', 'bbb'], ['warning', 'success']>
它的实现就是三部分的合并,但传入的是数组,要递归遍历取出每一个元素来和其他部分组合,这样太麻烦了。
而如果是联合类型就不用递归遍历了,因为联合类型遇到字符串也是会单独每个元素单独传入做处理。
数组转联合类型可以这样写:
tstype union = ['aaa', 'bbb'][number]\n// type union = 'aaa' | 'bbb'
那么 BEM 就可以这样实现:
tstype 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 通过索引访问来变为联合类型。
字符串类型中遇到联合类型的时候,会每个元素单独传入计算,也就是这样的效果:
tstype 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
tstype Combination<A extends string, B extends string> =\n | A\n | B\n | `${A}${B}`\n | `${B}${A}`
然后构造出来的字符串再和其他字符串组合。
所以全组合的高级类型就是这样:
tstype 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 特点
- 基于
Web Components
开发,跨框架复用,统一所有情况。 TypeScript
开发,有声明和类型文件。- 纯原生手写,基础组件无依赖。
- 文档基于
vitepress
,所有组件实例可交互。 MIT
协议。
Situation 项目情况
git
:https://github.com/chaxus/ran/tree/main/packages/ranui
npm
:https://www.npmjs.com/package/ranui
Usage 使用
大多数情况都可以像原生的 div
标签一样使用。
接下来是一些使用例子
html
js
jsx
vue
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 特点
- 基于
Web Components
开发,跨框架复用,统一所有情况。 TypeScript
开发,有声明和类型文件。- 纯原生手写,基础组件无依赖。
- 文档基于
vitepress
,所有组件实例可交互。 MIT
协议。
Situation 项目情况
git
:https://github.com/chaxus/ran/tree/main/packages/ranui
npm
:https://www.npmjs.com/package/ranui
Usage 使用
大多数情况都可以像原生的 div
标签一样使用。
接下来是一些使用例子
html
js
jsx
vue
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
目前支持password
、number
这几种类型,设置后会出现额外的ui
控件。
密码输入框
支持密码明文和密文切换。
html<r-input icon="lock" type="password"></r-input>
图标icon
可以设置一个icon
来表示标签标识。
html<r-input icon="user"></r-input>
数字输入框
数字输入框,类似于原生input[type=number]
,支持min
、max
、step
属性,支持键盘上下键切换数字。
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>
jsconst 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
目前支持password
、number
这几种类型,设置后会出现额外的ui
控件。
密码输入框
支持密码明文和密文切换。
html<r-input icon="lock" type="password"></r-input>
图标icon
可以设置一个icon
来表示标签标识。
html<r-input icon="user"></r-input>
数字输入框
数字输入框,类似于原生input[type=number]
,支持min
、max
、step
属性,支持键盘上下键切换数字。
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>
jsconst 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 文件预览
支持docx
,pptx
,pdf
,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 文件预览
支持docx
,pptx
,pdf
,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
搭配使用
代码演示
11111 22222 33333 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
搭配使用
代码演示
11111 22222 33333 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
代码展示
tab1 tab2 tab3 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
代码展示
tab1 tab2 tab3 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:约定空树不是任意一个树的子结构)
参考文档
',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:约定空树不是任意一个树的子结构)
参考文档
',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 文件是否被改变 boolean
true 文件改变 false 文件没变
Options
参数 说明 类型 默认值 path 文件路径,需要监听的文件 string
undefined interval 监听文件改变的时间,单位毫秒。 number
20
',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 文件是否被改变 boolean
true 文件改变 false 文件没变
Options
参数 说明 类型 默认值 path 文件路径,需要监听的文件 string
undefined interval 监听文件改变的时间,单位毫秒。 number
20
',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,直到排序完成。
动图演示
代码演示
jsfunction 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,直到排序完成。
动图演示
代码演示
jsfunction 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)
高效与否的关键在于这个分桶函数。将数据分到有限数量的桶里,每个桶再分别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排)。
算法描述
- 设置一个定量的数组当作空桶;
- 遍历输入数据,并且把数据一个一个放到对应的桶里去;
- 对每个不是空的桶进行排序;
- 从不是空的桶里把排好序的数据拼接起来。
代码演示
tsconst 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)
高效与否的关键在于这个分桶函数。将数据分到有限数量的桶里,每个桶再分别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排)。
算法描述
- 设置一个定量的数组当作空桶;
- 遍历输入数据,并且把数据一个一个放到对应的桶里去;
- 对每个不是空的桶进行排序;
- 从不是空的桶里把排好序的数据拼接起来。
代码演示
tsconst 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),即基于比较的时间复杂度存在下界 Ω(nlogn),而不基于比较的排序算法可以突破这一下界。
算法描述
- 找出待排序的数组中最大和最小的元素;
- 统计数组中每个值为 i 的元素出现的次数,存入数组 C 的第 i 项;
- 对所有的计数累加(从 C 中的第一个元素开始,每一项和前一项相加);
- 反向填充目标数组:将每个元素 i 放在新数组的第 C(i)项,每放一个元素就将 C(i)减去 1。
动图演示
代码演示
tsconst 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),即基于比较的时间复杂度存在下界 Ω(nlogn),而不基于比较的排序算法可以突破这一下界。
算法描述
- 找出待排序的数组中最大和最小的元素;
- 统计数组中每个值为 i 的元素出现的次数,存入数组 C 的第 i 项;
- 对所有的计数累加(从 C 中的第一个元素开始,每一项和前一项相加);
- 反向填充目标数组:将每个元素 i 放在新数组的第 C(i)项,每放一个元素就将 C(i)减去 1。
动图演示
代码演示
tsconst 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,则整个排序过程完成。
- 升序用大根堆,降序用小根堆
动图演示
代码演示
tsclass 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,则整个排序过程完成。
- 升序用大根堆,降序用小根堆
动图演示
代码演示
tsclass 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。
动图演示
代码演示
tsconst 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。
动图演示
代码演示
tsconst 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 的子序列;
- 对这两个子序列分别采用归并排序;
- 将两个排序好的子序列合并成一个最终的排序序列。
动图演示
代码演示
tsconst 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 的子序列;
- 对这两个子序列分别采用归并排序;
- 将两个排序好的子序列合并成一个最终的排序序列。
动图演示
代码演示
tsconst 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 进行计数排序(利用计数排序适用于小范围数的特点);
动图演示
代码演示
tsconst 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 进行计数排序(利用计数排序适用于小范围数的特点);
动图演示
代码演示
tsconst 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 趟结束,数组有序化了。
动图演示
代码实现
jsfunction 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 趟结束,数组有序化了。
动图演示
代码实现
jsfunction 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,则启动轮询模式。 number
0
pollingWhenHidden 在页面隐藏时,是否继续轮询。如果设置为 false,在页面隐藏时会暂时停止轮询,页面重新显示时继续上次轮询。 boolean
true
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,则启动轮询模式。 number
0
pollingWhenHidden 在页面隐藏时,是否继续轮询。如果设置为 false,在页面隐藏时会暂时停止轮询,页面重新显示时继续上次轮询。 boolean
true
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('统计执行时间
有的时候,我们需要统计一个函数的执行时间,用于分析性能。因此封装了startTask
和taskEnd
函数。同时介绍其他三种统计方法
new Date().getTime()
,console.time()
和 console.timeEnd()
,performance.now()
一.startTask
,taskEnd
1.startTask
任务开始之前执行
Return
参数 说明 类型 taskId 任务标识 unique symbol
2.taskEnd
任务结束的时候执行,需要传入startTask
返回的任务标识
Options
参数 说明 类型 默认值 taskId 任务标识 unique symbol
无默认值,参数必传,否则无法识别是哪个任务
Return
参数 说明 类型 time
task
执行的时间number
3.使用例子
jsconst 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 执行时间会有两个问题:
- 某些情况下,毫秒级精度可能不够。
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('统计执行时间
有的时候,我们需要统计一个函数的执行时间,用于分析性能。因此封装了startTask
和taskEnd
函数。同时介绍其他三种统计方法
new Date().getTime()
,console.time()
和 console.timeEnd()
,performance.now()
一.startTask
,taskEnd
1.startTask
任务开始之前执行
Return
参数 说明 类型 taskId 任务标识 unique symbol
2.taskEnd
任务结束的时候执行,需要传入startTask
返回的任务标识
Options
参数 说明 类型 默认值 taskId 任务标识 unique symbol
无默认值,参数必传,否则无法识别是哪个任务
Return
参数 说明 类型 time
task
执行的时间number
3.使用例子
jsconst 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 执行时间会有两个问题:
- 某些情况下,毫秒级精度可能不够。
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 类型系统中的类型
- 基本类型: number、boolean、string、object、bigint、symbol、undefined、null
- 复合类型: class、Array、元组(Tuple)、接口(Interface)、枚举(Enum)
- 特殊的类型:void、never、any、unknown
Tuple
元组(Tuple)就是元素个数和类型固定的数组类型:
tstype Tuple = [number, string]
Interface
接口(Interface)可以描述函数、对象、构造器的结构:
- 对象
tsinterface 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 类型系统中的类型
- 基本类型: number、boolean、string、object、bigint、symbol、undefined、null
- 复合类型: class、Array、元组(Tuple)、接口(Interface)、枚举(Enum)
- 特殊的类型:void、never、any、unknown
Tuple
元组(Tuple)就是元素个数和类型固定的数组类型:
tstype Tuple = [number, string]
Interface
接口(Interface)可以描述函数、对象、构造器的结构:
- 对象
tsinterface 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 类型:
tstype p = Promise<'value'>
我们想提取 value 的类型,可以这样做:
tstype GetPromiseValue<T> = T extends Promise<infer value> ? value : never
通过 extends 对传入的类型参数 P 做模式匹配,其中值的类型是需要提取的,通过 infer 声明一个局部变量 Value 来保存,如果匹配,就返回匹配到的 Value,否则就返回 never 代表没匹配到。
tstype result = GetPromiseValue<Promise<'name'>> // name
数组类型
数组类型想提取第一个元素的类型怎么做呢?
tstype arr = [1, 2, 3]
用它来匹配一个模式类型,提取第一个元素的类型到通过 infer 声明的局部变量里返回。
tstype 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 类型:
tstype p = Promise<'value'>
我们想提取 value 的类型,可以这样做:
tstype GetPromiseValue<T> = T extends Promise<infer value> ? value : never
通过 extends 对传入的类型参数 P 做模式匹配,其中值的类型是需要提取的,通过 infer 声明一个局部变量 Value 来保存,如果匹配,就返回匹配到的 Value,否则就返回 never 代表没匹配到。
tstype result = GetPromiseValue<Promise<'name'>> // name
数组类型
数组类型想提取第一个元素的类型怎么做呢?
tstype arr = [1, 2, 3]
用它来匹配一个模式类型,提取第一个元素的类型到通过 infer 声明的局部变量里返回。
tstype 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)比如:
tstype 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)比如:
tstype isTwo<T> = T extends 2 ? true : false
type res = isTwo<1> // true
type res2 = isTwo<2> // false
这种类型也叫做高级类型。
高级类型的特点是传入类型参数,经过一系列类型运算逻辑后,返回新的类型。
推导:infer
如何提取类型的一部分呢?答案是 infer。
比如提取元组类型的第一个元素:
tstype 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 ⚡️记录
每当遇到问题或者可以改进的事情的时候,会把想法先写下来,如果这是你遇到的问题,那么很可能有一群人也遇到同样的困难。
🖖解决
过一段时间后,再回来看看这些问题是否还有意义。然后再研究这个问题,确定是否存在解决方案。
🛠️改进
通常会发现一个工具是为了解决一个问题,我会去考虑如何改进或者简化这个工具。创造一个更好的轮子。
-
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
数据。
开发步骤主要分为:
- 初始化类型声明
-
Parameters 用于提取函数类型的参数类型。
ReturnType 用于提取函数类型的返回值类型。
构造器类型和函数类型的区别就是可以被 new。
Parameters 用于提取函数参数的类型,而 ConstructorParameters 用于提取构造器参数的类型。
提取了构造器参数的类型,自然也可以提取构造器返回值的类型,就是 InstanceType。
比如用模式匹配可以实现:Parameters、ReturnType、ConstructorParameters、InstanceType、ThisParameterType。
用模式匹配 + 重新构造可以实现:OmitThisParameter
用重新构造可以实现:Partial、Required、Readonly、Pick、Record
用模式匹配 + 递归可以实现: Awaited
用联合类型在分布式条件类型的特性可以实现: Exclude
此外还有 NonNullable 和四个编译器内部实现的类型:Uppercase、Lowercase、Capitalize、Uncapitalize。