哈喽,大家好 我是xy 👨🏻?💻。 从我最初接触vue3 版本到现在已经有一年的时间。由于 vue3.2 版本的发布,<script setup> 的实验性标志已经去掉,已经陆陆续续有不少公司开始使用 vue3.2 开发项目了。这篇文章就来帮助大家如何快速使用 vue3.x ,typeScript , vite 搭建一套企业级的开发脚手架 🤖。废话不多说,直接上手开搞 💪
搭建前准备
Vscode : 前端人必备写码神器Chrome :对开发者非常友好的浏览器(反正我是很依赖它的)Nodejs &npm :配置本地开发环境,安装 Node 后你会发现 npm 也会一起安装下来Vue.js devtools :浏览器调试插件Vue Language Features (Volar) :Vscode 开发 vue3 必备插件,提供语法高亮提示,非常好用Vue 3 Snippets :vue3 快捷输入
由于Vue.js devtools 需要到谷歌扩展商店才能下载,贴心 ?? 的xy 已经为大家准备好了crx 文件了,公众号回复:【VueDevTools 】可自动获取哦 💪
Vue2 与 Vue3 的区别
Vue3 由于完全由TS 进行重写,在应用中对类型判断的定义和使用有很强的表现。同一对象的多个键返回值必须通过定义对应的接口(interface )来进行类型定义。要不然在 ESLint 时都会报错。
vue2 的双向数据绑定是利用 ES5 的一个 API Object.definePropert() 对数据进行劫持 结合 发布订阅 模式的方式来实现的。Vue3 中使用了 es6 的 ProxyAPI 对数据代理。
Vue3 支持碎片(Fragments )
Vue2 与 Vue3 最大的区别: Vue2 使用Options API 而 Vue3 使用的Composition API
生命周期钩子变化:
Vue2 ~~~~~~~~~~~ vue3
beforeCreate -> setup()
created -> setup()
beforeMount -> onBeforeMount
mounted -> onMounted
beforeUpdate -> onBeforeUpdate
updated -> onUpdated
beforeDestroy -> onBeforeUnmount
destroyed -> onUnmounted
activated -> onActivated
deactivated -> onDeactivated
介绍 vite
Vite:下一代前端开发与构建工具
- 💡 极速的开发服务器启动
- ?? 轻量快速的热模块重载(HMR)
- 🛠? 丰富的功能
- 📦 自带优化的构建
- 🔩 通用的插件接口
- 🔑 完全类型化的 API
Vite (法语意为 “迅速”,发音 /vit/)是一种全新的前端构建工具,它极大地改善了前端开发体验。
它主要由两部分组成:
-
一个开发服务器,它基于 原生 ES 模块 提供了 丰富的内建功能,如速度快到惊人的 模块热更新(HMR)。 -
一套构建指令,它使用 Rollup 打包你的代码,并且它是预配置的,可以输出用于生产环境的优化过的静态资源。 -
Vite 意在提供开箱即用的配置,同时它的 插件 API 和 JavaScript API 带来了高度的可扩展性 ,并有完整的类型支持。
使用 vite 快速创建脚手架
兼容性注意:Vite 需要 Node.js 版本 >= 12.0.0 。
- 第一步: 在需要创建项目文件目录下打开
cmd 运行以下命令
npm init @vitejs/app vite_vue3_ts --template
npm init @vitejs/app vite_vue3_ts -- --template
yarn create @vitejs/app vite_vue3_ts --template
这里我采用 yarn 来安装
- 第二步: 选择
vue 回车 => vue-ts 回车
- 第三步:
cd 到项目文件夹,安装依赖,启动项目
cd vite_vue3_ts
yarn
yarn dev
约束代码风格
Eslint 支持
yarn add eslint --dev
yarn add eslint-plugin-vue --dev
yarn add @typescript-eslint/eslint-plugin --dev
yarn add eslint-plugin-prettier --dev
yarn add @typescript-eslint/parser --dev
注意: 如果 eslint 安装报错:
可以尝试运行以下命令:
yarn config set ignore-engines true
运行成功后再次执行 eslint 安装命令
项目下新建 .eslintrc.js
配置 eslint 校验规则:
module.exports = {
root: true,
env: {
browser: true,
node: true,
es2021: true,
},
parser: 'vue-eslint-parser',
extends: [
'eslint:recommended',
'plugin:vue/vue3-recommended',
'plugin:@typescript-eslint/recommended',
'plugin:prettier/recommended',
'prettier',
],
parserOptions: {
ecmaVersion: 12,
parser: '@typescript-eslint/parser',
sourceType: 'module',
ecmaFeatures: {
jsx: true,
},
},
plugins: ['vue', '@typescript-eslint', 'prettier'],
rules: {
'@typescript-eslint/ban-ts-ignore': 'off',
'@typescript-eslint/no-unused-vars': 'off',
'@typescript-eslint/explicit-function-return-type': 'off',
'@typescript-eslint/no-explicit-any': 'off',
'@typescript-eslint/no-var-requires': 'off',
'@typescript-eslint/no-empty-function': 'off',
'@typescript-eslint/no-use-before-define': 'off',
'@typescript-eslint/ban-ts-comment': 'off',
'@typescript-eslint/ban-types': 'off',
'@typescript-eslint/no-non-null-assertion': 'off',
'@typescript-eslint/explicit-module-boundary-types': 'off',
'no-var': 'error',
'prettier/prettier': 'error',
'no-console': 'warn',
'no-debugger': 'warn',
'no-duplicate-case': 'warn',
'no-empty': 'warn',
'no-extra-parens': 'off',
'no-func-assign': 'warn',
'no-unreachable': 'warn',
curly: 'warn',
'default-case': 'warn',
'dot-notation': 'warn',
eqeqeq: 'warn',
'no-else-return': 'warn',
'no-empty-function': 'warn',
'no-lone-blocks': 'warn',
'no-multi-spaces': 'warn',
'no-redeclare': 'warn',
'no-return-assign': 'warn',
'no-return-await': 'warn',
'no-self-assign': 'warn',
'no-self-compare': 'warn',
'no-useless-catch': 'warn',
'no-useless-return': 'warn',
'no-shadow': 'off',
'no-delete-var': 'off',
'array-bracket-spacing': 'warn',
'brace-style': 'warn',
camelcase: 'warn',
indent: 'off',
'max-depth': 'warn',
'max-statements': ['warn', 100],
'max-nested-callbacks': ['warn', 3],
'max-params': ['warn', 3],
'max-statements-per-line': ['warn', { max: 1 }],
'newline-per-chained-call': ['warn', { ignoreChainWithDepth: 3 }],
'no-lonely-if': 'warn',
'no-mixed-spaces-and-tabs': 'warn',
'no-multiple-empty-lines': 'warn',
semi: ['warn', 'never'],
'space-before-blocks': 'warn',
'space-in-parens': 'warn',
'space-infix-ops': 'warn',
'space-unary-ops': 'warn',
'switch-colon-spacing': 'warn',
'arrow-spacing': 'warn',
'no-var': 'warn',
'prefer-const': 'warn',
'prefer-rest-params': 'warn',
'no-useless-escape': 'warn',
'no-irregular-whitespace': 'warn',
'no-prototype-builtins': 'warn',
'no-fallthrough': 'warn',
'no-extra-boolean-cast': 'warn',
'no-case-declarations': 'warn',
'no-async-promise-executor': 'warn',
},
globals: {
defineProps: 'readonly',
defineEmits: 'readonly',
defineExpose: 'readonly',
withDefaults: 'readonly',
},
}
项目下新建 .eslintignore
node_modules
dist
prettier 支持
yarn add prettier --dev
解决 eslint 和 prettier 冲突
解决 ESLint 中的样式规范和 prettier 中样式规范的冲突 ,以 prettier 的样式规范为准 ,使 ESLint 中的样式规范自动失效
yarn add eslint-config-prettier --dev
项目下新建 .prettier.js
配置 prettier 格式化规则:
module.exports = {
tabWidth: 2,
jsxSingleQuote: true,
jsxBracketSameLine: true,
printWidth: 100,
singleQuote: true,
semi: false,
overrides: [
{
files: '*.json',
options: {
printWidth: 200,
},
},
],
arrowParens: 'always',
}
项目下新建 .prettierignore
node_modules
dist
package.json 配置:
{
"script": {
"lint": "eslint src --fix --ext .ts,.tsx,.vue,.js,.jsx",
"prettier": "prettier --write ."
}
}
上面配置完成后,可以运行以下命令 测试下代码检查个格式化 效果:
yarn lint
yarn prettier
配置 husky + lint-staged
使用husky + lint-staged 助力团队编码规范, husky&lint-staged 安装推荐使用 mrm , 它将根据 package.json 依赖项中的代码质量工具来安装和配置 husky 和 lint-staged,因此请确保在此之前安装并配置所有代码质量工具,如 Prettier 和 ESlint
首先安装 mrm
npm i mrm -D --registry=https://registry.npm.taobao.org
husky 是一个为 git 客户端增加 hook 的工具。安装后,它会自动在仓库中的 .git/ 目录下增加相应的钩子;比如 pre-commit 钩子就会在你执行 git commit 的触发。
那么我们可以在 pre-commit 中实现一些比如 lint 检查 、单元测试 、代码美化 等操作。当然,pre-commit 阶段执行的命令当然要保证其速度不要太慢,每次 commit 都等很久也不是什么好的体验。
lint-staged ,一个仅仅过滤出 Git 代码暂存区文件(被 git add 的文件)的工具;这个很实用,因为我们如果对整个项目的代码做一个检查,可能耗时很长,如果是老项目,要对之前的代码做一个代码规范检查并修改的话,这可能就麻烦了呀,可能导致项目改动很大。
所以这个 lint-staged ,对团队项目和开源项目来说,是一个很好的工具,它是对个人要提交的代码的一个规范和约束
安装 lint-staged
mrm 安装 lint-staged 会自动 把 husky 一起安装下来
npx mrm lint-staged
安装成功后会发现 package.json 中多了一下几个配置:
因为我们要结合 prettier 代码格式化,所有修改一下配置:
"husky": {
"hooks": {
"pre-commit": "lint-staged"
}
},
"lint-staged": {
"*.{js,jsx,vue,ts,tsx}": [
"yarn lint",
"prettier --write",
"git add"
]
}
好了,到这里代码格式化配置基本大功告成了!!!
可以修改部分代码尝试 git commit ,你会发现代码将自动格式化:
提交前的代码(发现编辑器爆红 了):
执行 commit 操作,控制台可以看到走了哪些流程:
commit 后的代码,是不是已经被格式化了
配置文件引用别名 alias
直接修改 vite.config.ts 文件配置:
import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'
import path from 'path'
export default defineConfig({
plugins: [vue()],
resolve: {
alias: {
'@': path.resolve(__dirname, 'src'),
},
},
})
修改 tsconfig.json
{
"compilerOptions": {
"target": "esnext",
"module": "esnext",
"moduleResolution": "node",
"strict": true,
"jsx": "preserve",
"sourceMap": true,
"resolveJsonModule": true,
"esModuleInterop": true,
"lib": ["esnext", "dom"],
"baseUrl": ".",
"paths": {
"@/*":["src/*"]
}
},
"include": ["src/**/*.ts", "src/**/*.d.ts", "src/**/*.tsx", "src/**/*.vue"]
}
配置 css 预处理器 scss
虽然 vite 原生支持 less/sass/scss/stylus ,但是你必须手动安装他们的预处理器依赖
安装
yarn add dart-sass --dev
yarn add sass --dev
配置全局 scss 样式文件
在 src/assets 下新增 style 文件夹,用于存放全局样式文件
新建 main.scss , 设置一个用于测试的颜色变量 :
$test-color: red;
如何将这个全局样式文件全局注入 到项目中呢?配置 Vite 即可:
css:{
preprocessorOptions:{
scss:{
additionalData:'@import "@/assets/style/mian.scss";'
}
}
},
组件中使用
不需要任何引入可以直接使用全局scss 定义的变量
.test{
color: $test-color;
}
路由
yarn add vue-router@4
在 src 文件下新增 router 文件夹 => router.ts 文件,内容如下:
import { createRouter, createWebHistory, RouteRecordRaw } from 'vue-router'
const routes: RouteRecordRaw[] = [
{
path: '/',
name: 'Login',
component: () => import('@/pages/login/Login.vue'),
},
]
const router = createRouter({
history: createWebHistory(),
routes,
})
export default router
修改入口文件 mian.ts :
import { createApp } from 'vue'
import App from './App.vue'
import router from './router/index'
const app = createApp(App)
app.use(router)
app.mount('#app')
到这里路由的基础配置已经完成了,更多配置信息可以查看 vue-router 官方文档:
vue-router: https://next.router.vuejs.org/zh/guide/
vue-router4.x 支持 typescript ,配置路由的类型是 RouteRecordRaw ,这里 meta 可以让我们有更多的发挥空间,这里提供一些参考:
title :string ; 页面标题,通常必选。icon? :string ; 图标,一般配合菜单使用。auth? :boolean ; 是否需要登录权限。ignoreAuth? :boolean ; 是否忽略权限。roles? :RoleEnum[] ; 可以访问的角色keepAlive? :boolean ; 是否开启页面缓存hideMenu? :boolean ; 有些路由我们并不想在菜单中显示,比如某些编辑页面。order? :number ; 菜单排序。frameUrl? :string ; 嵌套外链。
这里只提供一些思路,每个项目涉及到的业务都会存在些差异,这里就不作详细讲解了,根据自己业务需求做配置即可。
统一请求封装
使用过 vue2.x 的同学应该对 axios 很熟悉了,这里我们直接使用 axios 做封装:
yarn add axios
yarn add nprogress
yarn add @types/nprogress --dev
实际使用中可以根据项目修改,比如RESTful api 中可以自行添加put 和delete 请求,ResType 也可以根据后端的通用返回值动态的去修改
新增 service 文件夹,service 下新增 http.ts 文件以及 api 文件夹:
http.ts : 用于axios 封装
import axios, { AxiosRequestConfig } from 'axios'
import NProgress from 'nprogress'
axios.defaults.baseURL = '/api'
axios.defaults.timeout = 10000
axios.defaults.headers.post['Content-Type'] = 'application/json;charset=UTF-8'
axios.interceptors.request.use(
(config): AxiosRequestConfig<any> => {
const token = window.sessionStorage.getItem('token')
if (token) {
config.headers.token = token
}
return config
},
(error) => {
return error
}
)
axios.interceptors.response.use((res) => {
if (res.data.code === 111) {
sessionStorage.setItem('token', '')
}
return res
})
interface ResType<T> {
code: number
data?: T
msg: string
err?: string
}
interface Http {
get<T>(url: string, params?: unknown): Promise<ResType<T>>
post<T>(url: string, params?: unknown): Promise<ResType<T>>
upload<T>(url: string, params: unknown): Promise<ResType<T>>
download(url: string): void
}
const http: Http = {
get(url, params) {
return new Promise((resolve, reject) => {
NProgress.start()
axios
.get(url, { params })
.then((res) => {
NProgress.done()
resolve(res.data)
})
.catch((err) => {
NProgress.done()
reject(err.data)
})
})
},
post(url, params) {
return new Promise((resolve, reject) => {
NProgress.start()
axios
.post(url, JSON.stringify(params))
.then((res) => {
NProgress.done()
resolve(res.data)
})
.catch((err) => {
NProgress.done()
reject(err.data)
})
})
},
upload(url, file) {
return new Promise((resolve, reject) => {
NProgress.start()
axios
.post(url, file, {
headers: { 'Content-Type': 'multipart/form-data' },
})
.then((res) => {
NProgress.done()
resolve(res.data)
})
.catch((err) => {
NProgress.done()
reject(err.data)
})
})
},
download(url) {
const iframe = document.createElement('iframe')
iframe.style.display = 'none'
iframe.src = url
iframe.onload = function () {
document.body.removeChild(iframe)
}
document.body.appendChild(iframe)
},
}
export default http
api : 项目中接口做统一管理,按照模块来划分
在 api 文件下新增 login 文件夹,用于存放登录模块的请求接口,login 文件夹下分别新增 login.ts types.ts :
login.ts :
import http from '@/service/http'
import * as T from './types'
const loginApi: T.ILoginApi = {
login(params){
return http.post('/login', params)
}
}
export default loginApi
types.ts :
export interface ILoginParams {
userName: string
passWord: string | number
}
export interface ILoginApi {
login: (params: ILoginParams)=> Promise<any>
}
至此,一个简单地请求封装完成了!!!
除了自己手动封装 axios ,这里还推荐一个 vue3 的请求库: VueRequest ,非常好用,下面来看看 VueRequest 有哪些比较好用的功能吧!!!
- 🚀 所有数据都具有响应式
- 🔄 轮询请求
- 🤖 自动处理错误重试
- 🗄 内置请求缓存
- 💧 节流请求与防抖请求
- 🎯 聚焦页面时自动重新请求
- ?? 强大的分页扩展以及加载更多扩展
- 📠 完全使用 Typescript 编写,具有强大的类型提示
- ?? 兼容 Vite
- 🍃 轻量化
- 📦 开箱即用
是不是很强大 💪
官网链接: https://www.attojs.com/
状态管理 pinia
由于 vuex 4 对 typescript 的支持让人感到难过,所以状态管理弃用了 vuex 而采取了 pinia. pinia 的作者是 Vue 核心团队成员
尤大好像说 pinia 可能会代替 vuex ,所以请放心使用。
安装 pinia
Pinia 与 Vuex 的区别:
id 是必要的,它将所使用 store 连接到 devtools。- 创建方式:
new Vuex.Store(...) (vuex3),createStore(...) (vuex4)。 - 对比于 vuex3 ,state 现在是一个
函数返回对象 。 - 没有
mutations ,不用担心,state 的变化依然记录在 devtools 中。
yarn add pinia@next
main.ts 中增加
# 引入
import { createPinia } from "pinia"
# 创建根存储库并将其传递给应用程序
app.use(createPinia())
在 src 文件夹下新增 store 文件夹,接在在 store 中新增 main.ts
创建 store , mian.ts :
import { defineStore } from 'pinia'
export const useMainStore = defineStore({
id: 'mian',
state: () =>({
name: '超级管理员'
})
})
组建中获取 store :
<template>
<div>{{mainStore.name}}</div>
</template>
<script setup lang="ts">
import { useMainStore } from "@/store/mian"
const mainStore = useMainStore()
</script>
getters 用法介绍
Pinia 中的 getter 与 Vuex 中的 getter 、组件中的计算属性具有相同的功能
store => mian.ts
import { defineStore } from 'pinia'
export const useMainStore = defineStore({
id: 'mian',
state: () => ({
name: '超级管理员',
}),
getters: {
nameLength: (state) => state.name.length,
}
})
组件中使用:
<template>
<div>用户名:{{ mainStore.name }}<br />长度:{{ mainStore.nameLength }}</div>
<hr/>
<button @click="updateName">修改store中的name</button>
</template>
<script setup lang="ts">
import { useMainStore } from '@/store/mian'
const mainStore = useMainStore()
const updateName = ()=>{
mainStore.$patch({
name: '名称被修改了,nameLength也随之改变了'
})
}
</script>
actions
这里与 Vuex 有极大的不同,Pinia 仅提供了一种方法来定义如何更改状态的规则,放弃 mutations 只依靠 Actions ,这是一项重大的改变。
Pinia 让 Actions 更加的灵活:
- 可以通过组件或其他
action 调用 - 可以从其他
store 的 action 中调用 - 直接在
store 实例上调用 - 支持
同步 或异步 - 有任意数量的参数
- 可以包含有关如何更改状态的逻辑(也就是 vuex 的 mutations 的作用)
- 可以
$patch 方法直接更改状态属性
import { defineStore } from 'pinia'
export const useMainStore = defineStore({
id: 'mian',
state: () => ({
name: '超级管理员',
}),
getters: {
nameLength: (state) => state.name.length,
},
actions: {
async insertPost(data:string){
this.name = data;
}
},
})
环境变量配置
vite 提供了两种模式:具有开发服务器的开发模式 (development)和生产模式 (production)
项目根目录新建:.env.development :
NODE_ENV=development
VITE_APP_WEB_URL= 'YOUR WEB URL'
项目根目录新建:.env.production :
NODE_ENV=production
VITE_APP_WEB_URL= 'YOUR WEB URL'
组件中使用:
console.log(import.meta.env.VITE_APP_WEB_URL)
配置 package.json :
打包区分开发环境和生产环境
"build:dev": "vite build --mode development",
"build:pro": "vite build --mode production",
使用组件库 Naive UI
组件库选择,这里我们选择 Naive UI 至于为什么选择它?我可以直接说尤大大 推荐的吗?
- 官方介绍:
- 一个
Vue 3 组件库 - 比较完整,
主题可调 ,使用 TypeScript ,不算太慢 - 有点意思
介绍还是比较谦虚的,既然尤大 推荐,肯定有它的优势了!!!
安装 Naive UI
yarn add naive-ui
yarn add vfonts
如何使用
import { NButton } from "naive-ui"
<n-button>naive-ui</n-button>
全局配置 Config Provider
全局化配置设置内部组件的主题 、语言 和组件卸载于其他位置的 DOM 的类名。
<n-config-provider :locale="zhCN" :theme="theme">
</n-config-provider>
尤其是主题配置这个功能,我真的很喜欢 ??
组件库选择上不做任何强制,根据自己的项目需要选择合适的组件库即可
Vite 常用基础配置
基础配置
运行 代理 和 打包 配置
server: {
host: '0.0.0.0',
port: 3000,
open: true,
https: false,
proxy: {}
},
生产环境去除 console debugger
build:{
...
terserOptions: {
compress: {
drop_console: true,
drop_debugger: true
}
}
}
生产环境生成 .gz 文件
开启 gzip 可以极大的压缩静态资源,对页面加载的速度起到了显著的作用。
使用 vite-plugin-compression 可以 gzip 或 brotli 的方式来压缩资源,这一步需要服务器端的配合,vite 只能帮你打包出 .gz 文件。此插件使用简单,你甚至无需配置参数,引入即可。
yarn add --dev vite-plugin-compression
plugins 中添加:
import viteCompression from 'vite-plugin-compression'
viteCompression({
verbose: true,
disable: false,
threshold: 10240,
algorithm: 'gzip',
ext: '.gz',
}),
最终 vite.config.ts
import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'
import path from 'path'
import viteCompression from 'vite-plugin-compression'
export default defineConfig({
base: './',
plugins: [
vue(),
viteCompression({
verbose: true,
disable: false,
threshold: 10240,
algorithm: 'gzip',
ext: '.gz',
}),
],
resolve: {
alias: {
'@': path.resolve(__dirname, 'src'),
},
},
css:{
preprocessorOptions:{
scss:{
additionalData:'@import "@/assets/style/mian.scss";'
}
}
},
server: {
host: '0.0.0.0',
port: 8000,
open: true,
https: false,
proxy: {}
},
build: {
terserOptions: {
compress: {
drop_console: true,
drop_debugger: true,
},
},
},
})
常用插件
可以查看官方文档:https://vitejs.cn/plugins/
@vitejs/plugin-vue 提供 Vue 3 单文件组件支持@vitejs/plugin-vue-jsx 提供 Vue 3 JSX 支持(通过 专用的 Babel 转换插件)@vitejs/plugin-legacy 为打包后的文件提供传统浏览器兼容性 支持unplugin-vue-components 组件的按需自动导入vite-plugin-compression 使用 gzip 或者 brotli 来压缩资源- …
非常推荐使用的 hooks 库
因为vue3.x 和react hooks 真的很像,所以就称为 hooks
VueUse :https://vueuse.org/
看到这个库的第一眼,让我立马想到了 react 的 ahooks
VueUse 是一个基于 Composition API 的实用函数集合。通俗的来说,这就是一个工具函数 包,它可以帮助你快速实现一些常见的功能,免得你自己去写,解决重复的工作内容。以及进行了基于 Composition API 的封装。让你在 vue3 中更加得心应手。
💡想要入手 vue3 的小伙伴,赶快学习起来吧!!!
💡最后给大家奉上仓库地址吧:https://github.com/xushanpei/vite_vue3_ts
写在最后
公众号 :前端开发爱好者 专注分享 web 前端相关技术文章 、视频教程 资源、热点资讯等,如果喜欢我的分享,给 🐟🐟 点一个赞 👍 或者 ?关注 都是对我最大的支持。
欢迎长按图片加好友 ,我会第一时间和你分享前端行业趋势 ,面试资源 ,学习途径 等等。
关注公众号后,在首页:
- 回复
面试题 ,获取最新大厂面试资料。 - 回复
简历 ,获取 3200 套 简历模板。 - 回复
React实战 ,获取 React 最新实战教程。 - 回复
Vue实战 ,获取 Vue 最新实战教程。 - 回复
ts ,获取 TypeAcript 精讲课程。 - 回复
vite ,获取 精讲课程。 - 回复
uniapp ,获取 uniapp 精讲课程。 - 回复
js书籍 ,获取 js 进阶 必看书籍。 - 回复
Node ,获取 Nodejs+koa2 实战教程。 - 回复
数据结构算法 ,获取 数据结构算法 教程。 - 回复
架构师 ,获取 架构师学习资源教程。 - 更多教程资源应用尽有,欢迎
关注获取
|