Heim > Schlagzeilen > 9 vue3-Entwicklungsfähigkeiten, um die Effizienz zu verbessern und Ihnen zu helfen, früher von der Arbeit zu gehen!

9 vue3-Entwicklungsfähigkeiten, um die Effizienz zu verbessern und Ihnen zu helfen, früher von der Arbeit zu gehen!

青灯夜游
Freigeben: 2022-09-27 16:05:25
nach vorne
3459 Leute haben es durchsucht

vue3 ist seit langem veröffentlicht. Der Beamte hat auch die Standardversion auf vue3 umgestellt, und es ist auch ein vollständiges Chinesisches Dokument erschienen. Ich frage mich, ob Genossen es bereits verwendet haben. Ich benutze es schon seit einiger Zeit und es läuft immer noch recht reibungslos. Ich hoffe, dass jeder früh Feierabend machen kann. Nutzen Sie h (createVNode) und die Renderfunktionen

Wir wissen, dass eine in vue3 exportierte Funktion „createVNode“ ein vdom erstellen kann. Wenn wir es gut nutzen, können wir zum Beispiel unerwartete Effekte erzielen Implementieren Sie eine Popup-Komponente

unsere übliche Idee besteht darin, eine Komponente zu schreiben und sie im Projekt zu referenzieren und ihre Anzeige und Ausblendung über das V-Modell zu steuern. Es gibt jedoch ein Problem Die Wiederverwendung muss kopiert und eingefügt werden. Wir haben keine Möglichkeit, die Effizienz zu verbessern, indem wir sie beispielsweise in npm kapseln und durch Aufrufen von js verwenden. [Verwandte Empfehlungen: vuejs Video-Tutorial]

Aber da Nach dem Hinzufügen von createVNode und render sind alle Probleme gelöst

// 我们先写一个弹窗组件
        const message = {
            setup() {
                const num = ref(1)
                return {
                    num
                }
            },
            template: `<div>
                        <div>{{num}}</div>
                        <div>这是一个弹窗</div>
                      </div>`
        }
Nach dem Login kopieren
  // 初始化组件生成vdom
  const vm = createVNode(message)
  // 创建容器,也可以用已经存在的
  const container = document.createElement('div')
  //render通过patch 变成dom
  render(vm, container)
// 弹窗挂到任何你想去的地方  
document.body.appendChild(container.firstElementChild)
Nach dem Login kopieren
比如我们要实现一个弹窗组件

我们通常的思路是写一个组件在项目中引用进来,通过v-model来控制他的显示隐藏,但是这样有个问题,我们复用的时候的成本需要复制粘贴。我们没有办法来提高效率,比如封装成npm 通过调用js来使用。【相关推荐:vuejs视频教程

然而,有了 createVNode 和render 之后所有问题就迎刃而解了

1、灵活、灵活、灵活(重要的事情说三遍)
Nach dem Login kopieren
2、一个文件能写好多个组件
Nach dem Login kopieren

经过上面这一通骚操作,我们发现我们可以将他封装为一个方法,放到任何想放的地方。

善用JSX/TSX

文档上说了,在绝大多数情况下,Vue 推荐使用模板语法来搭建 HTML。然而在某些使用场景下,我们真的需要用到 JavaScript 完全的编程能力。这时渲染函数就派上用场了。

jsx和模板语法的优势对比

jsx和模板语法都是vue 支持的的书写范畴,然后他们确有不同的使用场景,和方式,需要我们根据当前组件的实际情况,来酌情使用

什么是JSX

JSX 是一种 Javascript 的语法扩展,JSX = Javascript + XML,即在 Javascript 里面写 XML,因为 JSX 的这个特性,所以他即具备了 Javascript 的灵活性,同时又兼具 html 的语义化和直观性

模板语法的优势

  • 1、模板语法书写起来不怎么违和,我们就像在写html一样
  • 2、在vue3中由于模板的可遍历性,它能在编译阶段做更多优化,比如静态标记、块block、缓存事件处理程序等
  • 3、模板代码逻辑代码严格分开,可读性高
  • 4、对JS功底不那么好的人,记几个命令就能快速开发,上手简单
  • 5、vue官方插件的完美支持,代码格式化,语法高亮等

JSX的优势

  • 3、只要JS功底好,就不用记忆那么多命令,上来就是一通输出
    Nach dem Login kopieren
  • 4、JS和JSX混用,方法即声明即用,对于懂行的人来说逻辑清晰
    Nach dem Login kopieren
  • //btn1.vue
    <template>
      <div>
          这是btn1{{ num }}
          <slot></slot>
      </div>
    </template>
    <script>
    import { ref, defineComponent } from &#39;vue&#39;
    export default defineComponent({
      name: &#39;btn1&#39;,
      setup() {
          const num = ref(1)
          return { num }
      }
    })
    </script>
    //btn2.vue
    <template>
      <div>
          这是btn2{{ num }}
          <slot></slot>
      </div>
    </template>
    <script>
    import { ref, defineComponent } from &#39;vue&#39;
    export default defineComponent({
      name: &#39;btn2&#39;,
      setup() {
          const num = ref(2)
          return { num }
      }
    })
    </script>
    Nach dem Login kopieren
  • // 容器组件
    import btn1 from './btn1.vue'
    import btn2 from './btn2.vue'
    export const renderFn = function (props, context) {
      return props.type == 1 ? <btn1>{context.slots.default()}</btn1> : <btn2>{context.slots.default()}</btn2>
    }
    Nach dem Login kopieren

对比

由于vue对于JSX的支持,社区里,也是争论来争论去,到底要分个高低,然后本渣认为,他俩本来没有高低,您觉得哪个适合,就用哪个即可,缺点放在对的地方他就是优势 要发扬咱们老前辈们传下来的中庸之道,做集大成者,将两者结合使用,就能发挥无敌功效,乱军之中博老板青睐。

接下来说一下本人的一点粗浅理解,我们知道组件类型,分为容器型组件和展示展示型组件 在一般情况下,容器型组件,他由于可能要对于当前展示型组件做一个标准化或者宰包装,那么此时容器型组件中用JSX就再好不过

举个例子:现在有个需求,我们有两个按钮,现在要做一个通过后台数据来选择展示哪一个按钮,我们通常的做法,是通过在一个模板中通过v-if去控制不同的组件Nach dem obigen Vorgang haben wir festgestellt, dass wir ihn als Methode kapseln und an einer beliebigen Stelle platzieren können.

Nutzen Sie JSX/TSX sinnvoll

In der Dokumentation heißt es, dass Vue in den meisten Fällen die Verwendung von Vorlagensyntax zum Erstellen von HTML empfiehlt. In einigen Anwendungsfällen müssen wir jedoch unbedingt die vollständigen Programmierfunktionen von JavaScript nutzen. Hier kommt die rendering-Funktion zum Einsatz.

Vergleich der Vorteile von jsx und der Vorlagensyntax

jsx und Vorlagensyntax sind beide von Vue unterstützte Schreibkategorien, und sie haben unterschiedliche Verwendungsszenarien und -methoden. Wir müssen sie entsprechend der tatsächlichen Situation verwenden Die aktuelle Komponente. Je nach Bedarf verwenden

Was ist JSX?JSX ist eine Syntaxerweiterung von Javascript, d JSX hat es. Es kombiniert die Flexibilität von Javascript mit der semantischen und intuitiven Natur von HTML.

Vorteile der Vorlagensyntax

  • 1. Die Vorlagensyntax ist nicht sehr inkonsistent, wir schreiben einfach HTML
  • 2, aufgrund der Durchgängigkeit von Vorlagen Während der Kompilierungsphase können weitere Optimierungen durchgeführt werden, z. B. statische Tags, Blöcke, zwischengespeicherte Ereignishandler usw.
  • 3 Der Logikcode des Vorlagencodes ist streng getrennt und weist eine hohe Lesbarkeit auf
  • 4. Für diejenigen, die nicht sehr gut in JS sind, können sie sich schnell entwickeln, indem sie sich ein paar Befehle merken, und der Einstieg ist einfach
  • 5 Perfekte Unterstützung für das offizielle Vue-Plug-in und die Codeformatierung , Syntaxhervorhebung usw.
🎜Vorteile von JSX🎜🎜
  • //业务组件
    <template>
     <renderfn>1111111</renderfn>
    </template>
    <script>
    import { renderFn } from &#39;./components&#39;
    console.log(renderFn)
    export default {
     components: {
       renderFn
     },
     setup() {
     },
    };
    </script>
    Nach dem Login kopieren
  • //parent.vue
    <template>
        <child></child>
        <button>添加</button>
    </template>
    
    <script>
    import { defineComponent, provide, ref } from "vue";
    import Child from "./child.vue";
    export default defineComponent({
        components: {
            Child
        },
        setup() {
            const count = ref(0);
            const color = ref(&#39;#000&#39;)
            provide(&#39;count&#39;, count)
            provide(&#39;color&#39;, color)
            function setColor(val) {
                color.value = val
            }
            return {
                count,
                setColor
            }
        }
    })
    </script>
    Nach dem Login kopieren
    Nach dem Login kopieren
  • //child.vue
    //使用inject 注入
    <template>
        <div>这是注入的内容{{ count }}</div>
        <child1></child1>
    </template>
    
    <script>
    import { defineComponent, inject } from "vue";
    import child1 from &#39;./child1.vue&#39;
    export default defineComponent({
        components: {
            child1
        },
        setup(props, { attrs }) {
            const count = inject(&#39;count&#39;);
            console.log(count)
            console.log(attrs)
            return {
                count
            }
        }
    })
    </script>
    Nach dem Login kopieren
    Nach dem Login kopieren
  • //子孙组件child1.vue
    <template>
        <div>这是注入的内容的颜色</div>
    </template>
    
    <script>
    import { defineComponent, inject } from "vue";
    
    export default defineComponent({
        setup(props, { emit }) {
            const color = inject(&#39;color&#39;);
            function setColor() {
                console.log(0)
                emit(&#39;setColor&#39;, &#39;red&#39;)
            }
            return {
                color,
                setColor
            }
        }
    })
    </script>
    Nach dem Login kopieren
    Nach dem Login kopieren
🎜🎜Vergleich🎜🎜🎜Aufgrund der Unterstützung von vue für JSX gibt es in der Community Debatten darüber, ob sie höher oder niedriger eingestuft werden sollten. Dann gibt es meiner Meinung nach keinen Unterschied zwischen ihnen. Was auch immer Sie für geeignet halten, verwenden Sie es einfach. Schwächen können Vorteile haben, wenn sie an der richtigen Stelle platziert werden Wir müssen die von unseren Vorgängern weitergegebene goldene Mitte weiterführen und ein Meister sein, indem wir beides kombinieren , können Sie eine unbesiegbare Wirkung ausüben und im Chaos Boss-Gefallen gewinnen. 🎜🎜 Lassen Sie mich als Nächstes über mein grobes Verständnis sprechen. Wir wissen, dass Komponententypen in Containerkomponenten und Anzeigekomponenten unterteilt sind. Im Allgemeinen können Containerkomponenten für die aktuelle Anzeige verwendet werden Typkomponente, zu diesem Zeitpunkt wäre es am besten, JSX in der Containertypkomponente zu verwenden. Zum Beispiel: Jetzt müssen wir zwei Schaltflächen erstellen, um basierend auf dem Hintergrund auszuwählen, welche angezeigt werden sollen Für eine Schaltfläche besteht unser üblicher Ansatz darin, verschiedene Komponenten über v-if in einer Vorlage zu steuern🎜🎜Bei JSX und Funktionskomponenten haben wir jedoch festgestellt, dass die Logik klarer und der Code prägnanter und höher ist Qualität und noch mehr /Inject)🎜🎜🎜Bevor wir die Abhängigkeitsinjektion sinnvoll nutzen, wollen wir zunächst einige Konzepte verstehen, die uns helfen, die Vergangenheit und Gegenwart der Abhängigkeitsinjektion umfassender zu verstehen.🎜🎜🎜🎜Was sind IOC und DI?🎜🎜🎜

控制反转(Inversion of Control,缩写为IoC),是面向对象编程中的一种设计原则,可以用来减低计算机代码之间的耦合度。其中最常见的方式叫做依赖注入(Dependency Injection,简称DI),还有一种方式叫“依赖查找”(Dependency Lookup)。通过控制反转,对象在被创建的时候,由一个调控系统内所有对象的外界实体,将其所依赖的对象的引用传递(注入)给它。

什么是依赖注入

依赖注入 用大白话来说:就是将实例变量传入到一个对象中去

vue中的依赖注入

在vue中,我们套用依赖注入的概念,其实就是在父组件中声明依赖,将他们注入到子孙组件实例中去,可以说是能够很大程度上代替全局状态管理的存在

9 vue3-Entwicklungsfähigkeiten, um die Effizienz zu verbessern und Ihnen zu helfen, früher von der Arbeit zu gehen!

我们先来看看他的基本用法

父组件中声明provide

//parent.vue
<template>
    <child></child>
    <button>添加</button>
</template>

<script>
import { defineComponent, provide, ref } from "vue";
import Child from "./child.vue";
export default defineComponent({
    components: {
        Child
    },
    setup() {
        const count = ref(0);
        const color = ref(&#39;#000&#39;)
        provide(&#39;count&#39;, count)
        provide(&#39;color&#39;, color)
        function setColor(val) {
            color.value = val
        }
        return {
            count,
            setColor
        }
    }
})
</script>
Nach dem Login kopieren
Nach dem Login kopieren

子组件中注入进来

//child.vue
//使用inject 注入
<template>
    <div>这是注入的内容{{ count }}</div>
    <child1></child1>
</template>

<script>
import { defineComponent, inject } from "vue";
import child1 from &#39;./child1.vue&#39;
export default defineComponent({
    components: {
        child1
    },
    setup(props, { attrs }) {
        const count = inject(&#39;count&#39;);
        console.log(count)
        console.log(attrs)
        return {
            count
        }
    }
})
</script>
Nach dem Login kopieren
Nach dem Login kopieren

正因为依赖注入的特性,我们很大程度上代替了全局状态管理,相信谁都不想动不动就引入那繁琐的vuex

接下来我们来举个例子,现在我么有个页面主题色,他贯穿所有组件,并且可以在某一些组件内更改主题色,那我们常规的解决方案中,就是装个vuex然后通过他的api下发颜色值,这时候如果想改,首先要发起dispatch到Action ,然后在Action中触发Mutation接着在Mutation中再去改state,如此一来,你是不是发现有点杀鸡用牛刀了,我就改个颜色而已!

我们来看有了依赖注入 应该怎么处理

首先我们知道vue是单项数据流,也就是子组件不能修改父组件的内容,于是我们就应该想到使用$attrs使用它将方法透传给祖先组件,在组件组件中修改即可。

我们来看代码

//子孙组件child1.vue
<template>
    <div>这是注入的内容的颜色</div>
</template>

<script>
import { defineComponent, inject } from "vue";

export default defineComponent({
    setup(props, { emit }) {
        const color = inject(&#39;color&#39;);
        function setColor() {
            console.log(0)
            emit(&#39;setColor&#39;, &#39;red&#39;)
        }
        return {
            color,
            setColor
        }
    }
})
</script>
Nach dem Login kopieren
Nach dem Login kopieren

将当前子孙组件嵌入到child.vue中去,就能利用简洁的方式来修改颜色了

善用Composition API抽离通用逻辑

众所周知,vue3最大的新特性,当属Composition API 也叫组合api ,用好了他,就是你在行业的竞争力,你也有了不世出的技能

我们一步步来分析

什么是Composition API

使用 (datacomputedmethodswatch) 组件选项来组织逻辑通常都很有效。然而,当我们的组件开始变得更大时,逻辑关注点的列表也会增长。尤其对于那些一开始没有编写这些组件的人来说,这会导致组件难以阅读和理解。

于是在vue3中为了解决当前痛点,避免在大型项目中出现代码逻辑分散,散落在当前组件的各个角落,从而变得难以维护,Composition API横空出世

所谓Composition API 就是在组件配置对象中声明 setup函数,我们可以将所有的逻辑封装在setup函数中,然后在配合vue3中提供的响应式API 钩子函数、计算属性API等,我们就能达到和常规的选项式同样的效果,但是却拥有更清晰的代码以及逻辑层面的复用

基础使用

<template>
    <div>测试compositionApi</div>
</template>
<script>
import { inject, ref, onMounted, computed, watch } from "vue";
export default {
    // setup起手
    setup(props, { attrs, emit, slots, expose }) {

        // 获取页面元素
        const composition = ref(null)
        // 依赖注入
        const count = inject(&#39;foo&#39;, &#39;1&#39;)
        // 响应式结合
        const num = ref(0)
        //钩子函数
        onMounted(() => {
            console.log(&#39;这是个钩子&#39;)
        })
        // 计算属性
        computed(() => num.value + 1)
        // 监听值的变化
        watch(count, (count, prevCount) => {
            console.log(&#39;这个值变了&#39;)
        })
        return {
            num,
            count
        }

    }
}
</script>
Nach dem Login kopieren

通过以上代码我们可以看出,一个setup函数我们干出了在传统选项式中的所有事情,然而这还不是最绝的,通过这些api的组合可以实现逻辑复用,这样我们就能封装很多通用逻辑,实现复用,早点下班

举个例子:大家都用过复制剪贴板的功能,在通常情况下,利用navigator.clipboard.writeText 方法就能将复制内容写入剪切板。然而,细心的你会发现,其实赋值剪切板他是一个通用功能,比如:你做b端业务的,管理系统中到处充满了复制id、复制文案等功能。

于是Composition API的逻辑复用能力就派上了用场

import { watch, getCurrentScope, onScopeDispose, unref, ref } from "vue"
export const isString = (val) => typeof val === 'string'
export const noop = () => { }
export function unrefElement(elRef) {
    const plain = unref(elRef)// 拿到本来的值
    return (plain).$el ?? plain //前面的值为null、undefined,则取后面的值,否则都取前面的值
}
export function tryOnScopeDispose(fn) {
    // 如果有活跃的effect
    if (getCurrentScope()) {
        //在当前活跃的 effect 作用域上注册一个处理回调。该回调会在相关的 effect 作用域结束之后被调用
        //能代替onUmounted
        onScopeDispose(fn)
        return true
    }
    return false
}
//带有控件的setTimeout包装器。
export function useTimeoutFn(
    cb,// 回调
    interval,// 时间
    options = {},
) {
    const {
        immediate = true,
    } = options

    const isPending = ref(false)

    let timer

    function clear() {
        if (timer) {
            clearTimeout(timer)
            timer = null
        }
    }

    function stop() {
        isPending.value = false
        clear()
    }

    function start(...args) {
        // 清除上一次定时器
        clear()
        // 是否在pending 状态
        isPending.value = true
        // 重新启动定时器
        timer = setTimeout(() => {
            // 当定时器执行的时候结束pending状态
            isPending.value = false
            // 初始化定时器的id
            timer = null
            // 执行回调
            cb(...args)
        }, unref(interval))
    }
    if (immediate) {
        isPending.value = true

        start()
    }

    tryOnScopeDispose(stop)

    return {
        isPending,
        start,
        stop,
    }
}
//轻松使用EventListener。安装时使用addEventListener注册,卸载时自动移除EventListener。
export function useEventListener(...args) {
    let target
    let event
    let listener
    let options
    // 如果第一个参数是否是字符串
    if (isString(args[0])) {
        //结构内容
        [event, listener, options] = args
        target = window
    }
    else {
        [target, event, listener, options] = args
    }
    let cleanup = noop
    const stopWatch = watch(
        () => unrefElement(target),// 监听dom
        (el) => {
            cleanup() // 执行默认函数
            if (!el)
                return
            // 绑定事件el如果没有传入就绑定为window
            el.addEventListener(event, listener, options)
            // 重写函数方便改变的时候卸载
            cleanup = () => {
                el.removeEventListener(event, listener, options)
                cleanup = noop
            }
        },
        //flush: 'post' 模板引用侦听
        { immediate: true, flush: 'post' },
    )
    // 卸载
    const stop = () => {
        stopWatch()
        cleanup()
    }

    tryOnScopeDispose(stop)

    return stop
}

export function useClipboard(options = {}) {
    //获取配置
    const {
        navigator = window.navigator,
        read = false,
        source,
        copiedDuring = 1500,
    } = options
    //事件类型
    const events = ['copy', 'cut']
    // 判断当前浏览器知否支持clipboard
    const isSupported = Boolean(navigator && 'clipboard' in navigator)
    // 导出的text
    const text = ref('')
    //导出的copied
    const copied = ref(false)
    // 使用的的定时器钩子
    const timeout = useTimeoutFn(() => copied.value = false, copiedDuring)

    function updateText() {
        //解析系统剪贴板的文本内容返回一个Promise
        navigator.clipboard.readText().then((value) => {
            text.value = value
        })
    }

    if (isSupported && read) {
        // 绑定事件
        for (const event of events)
            useEventListener(event, updateText)
    }
    // 复制剪切板方法
    //navigator.clipboard.writeText 方法是异步的返回一个promise
    async function copy(value = unref(source)) {
        if (isSupported && value != null) {
            await navigator.clipboard.writeText(value)
            // 响应式的值,方便外部能动态获取
            text.value = value
            copied.value = true
            timeout.start()// copied.value = false 
        }
    }

    return {
        isSupported,
        text,
        copied,
        copy,
    }
}
Nach dem Login kopieren

这时我们就复用了复制的逻辑,如下代码中直接引入在模板中使用即可

<template>
    <div>
        <p>
            <code>{{ text || '空' }}</code>
        </p>
        <input>
        <button>
            <span>复制</span>
            <span>复制中!</span>
        </button>
    </div>
    <p>您的浏览器不支持剪贴板API</p>
</template>
<script>
import { ref, getCurrentScope } from &#39;vue&#39;
import { useClipboard } from &#39;./copy.js&#39;
const input = ref(&#39;&#39;)
const { text, isSupported, copied, copy } = useClipboard()
console.log(text)// 复制内容
console.log(isSupported)// 是否支持复制剪切板api 
console.log(copied)//是否复制完成延迟
console.log(copy) // 复制方法
</script>
Nach dem Login kopieren

以上代码参考vue版本的Composition API库所有完整版请参考

善于使用getCurrentInstance 获取组件实例

getCurrentInstance 支持访问内部组件实例, 通常情况下他被放在 setup中获取组件实例,但是getCurrentInstance 只暴露给高阶使用场景,典型的比如在库中。

强烈反对在应用的代码中使用 getCurrentInstance。请不要把它当作在组合式 API 中获取 this 的替代方案来使用。

那他的作用是什么呢?

还是逻辑提取,用来代替Mixin,这是在复杂组件中,为了整个代码的可维护性,抽取通用逻辑这是必须要去做的事情,我们可以看element-plus 中table的复用逻辑,在逻辑提取中由于涉及获取props、proxy、emit 以及能通过当前组件获取父子组件的关系等,此时getCurrentInstance 的作用无可代替

如下element-plus代码中利用getCurrentInstance 获取父组件parent中的数据,分别保存到不同的变量中,我们只需要调用当前useMapState即可拿到数据

// 保存数据的逻辑封装
function useMapState<t>() {
  const instance = getCurrentInstance()
  const table = instance.parent as Table<t>
  const store = table.store
  const leftFixedLeafCount = computed(() => {
    return store.states.fixedLeafColumnsLength.value
  })
  const rightFixedLeafCount = computed(() => {
    return store.states.rightFixedColumns.value.length
  })
  const columnsCount = computed(() => {
    return store.states.columns.value.length
  })
  const leftFixedCount = computed(() => {
    return store.states.fixedColumns.value.length
  })
  const rightFixedCount = computed(() => {
    return store.states.rightFixedColumns.value.length
  })

  return {
    leftFixedLeafCount,
    rightFixedLeafCount,
    columnsCount,
    leftFixedCount,
    rightFixedCount,
    columns: store.states.columns,
  }
}</t></t>
Nach dem Login kopieren

善用$attrs

$attrs 现在包含了所有传递给组件的 attribute,包括 class 和 style

$attrs在我们开发中到底有什么用呢?

通过他,我们可以做组件的事件以及props透传

首先有一个标准化的组件,一般是组件库的组件等等

//child.vue
<template>
    <div>这是一个标准化组件</div>
    <input>
</template>

<script>
import { defineComponent } from "vue";

export default defineComponent({
    props: [&#39;num&#39;],
    emits: [&#39;edit&#39;],
    setup(props, { emit }) {
        function setInput(val) {
            emit(&#39;edit&#39;, val.target.value)
        }
        return {
            setInput
        }
    }
})
</script>
Nach dem Login kopieren

接下来有一个包装组件,他对当前的标准化组件做修饰,从而使结果变成我们符合我们的预期的组件

//parent.vue
 <template>
    <div>这一层要做一个单独的包装</div>
    <child></child>
</template>

<script>
import { defineComponent } from "vue";
import child from &#39;./child.vue&#39;
export default defineComponent({
    components: {
        child
    },
    setup(props, { emit }) {
        function edit(val) {
            // 对返回的值做一个包装
            emit(&#39;edit&#39;, `${val}time`)
        }
        return {
            edit
        }
    }
})
</script>
Nach dem Login kopieren

我们发现当前包装组件中使用了$attrs,通过他透传给标准化组件,这样一来,我们就能对比如element UI中的组件做增强以及包装处理,并且不用改动原组件的逻辑。

优雅注册全局组件技巧

vue3的组件通常情况下使用vue提供的component 方法来完成全局组件的注册

代码如下:

const app = Vue.createApp({})

app.component('component-a', {
  /* ... */
})
app.component('component-b', {
  /* ... */
})
app.component('component-c', {
  /* ... */
})

app.mount('#app')
Nach dem Login kopieren

使用时

<div>
  <component-a></component-a>
  <component-b></component-b>
  <component-c></component-c>
</div>
Nach dem Login kopieren

然而经过大佬的奇技淫巧的开发,我们发现可能使用注册vue插件的方式,也能完成组件注册,并且是优雅的!

vue插件注册

插件的格式

//plugins/index.js
export default {
  install: (app, options) => {
      // 这是插件的内容
  }
}
Nach dem Login kopieren

插件的使用

import { createApp } from 'vue'
import Plugin from './plugins/index.js'
const app = createApp(Root)
app.use(Plugin)
app.mount('#app')
Nach dem Login kopieren

其实插件的本质,就是在use的方法中调用插件中的install方法,那么这样一来,我们就能在install方法中注册组件。

index.js中抛出一个组件插件

// index.js
import component from './Cmponent.vue'
const component = {
    install:function(Vue){
        Vue.component('component-name',component)
    }  //'component-name'这就是后面可以使用的组件的名字,install是默认的一个方法 component-name 是自定义的,我们可以按照具体的需求自己定义名字
}
// 导出该组件
export default component
Nach dem Login kopieren

组件注册

// 引入组件
import install from './index.js'; 
// 全局挂载utils
Vue.use(install);
Nach dem Login kopieren

上述案例中,就是一个简单的优雅的组件注册方式,大家可以发现包括element-plus、vant 等组件都是用如此方式注册组件。

善用setup>

<script setup></script> 是在单文件组件 (SFC) 中使 的编译时语法糖。相比于普通的 <script></script> 语法,它具有更多优势:

  • 更少的样板内容,更简洁的代码。
  • 能够使用纯 Typescript 声明 props 和抛出事件。
  • 更好的运行时性能 (其模板会被编译成与其同一作用域的渲染函数,没有任何的中间代理)。
  • 更好的 IDE 类型推断性能 (减少语言服务器从代码中抽离类型的工作)。

它能代替大多数的setup函数所表达的内容,具体使用方法,大家请看请移步文档

但是由于setup函数它能返回渲染函数的特性,在当前语法糖中却无法展示,于是遍寻资料,找到了一个折中的办法

<script>
import { ref,h } from &#39;vue&#39;

const msg = ref(&#39;Hello World!&#39;)
const dynode = () => h(&#39;div&#39;,msg.value);

</script>

<template>
    <dynode></dynode>
  <input>
</template>
Nach dem Login kopieren

如此一来,我们就能在语法糖中返回渲染函数了

v-model的最新用法

我们知道在vue2中想要模拟v-model,必须要子组件要接受一个value props 吐出来一个 叫input的emit

然而在vue3中他升级了

父组件中使用v-model

 <template>
    <child></child>
</template>

<script>
import { defineComponent, ref } from "vue";
import child from &#39;./child.vue&#39;
export default defineComponent({
    components: {
        child
    },
    setup(props, { emit }) {
        const pageTitle = ref(&#39;这是v-model&#39;)
        return {
            pageTitle
        }
    }
})
</script>
Nach dem Login kopieren

子组件中使用 title的props 以及规定吐出update:title的emit

<template>
    <div>{{ title }}</div>
    <input>
</template>

<script>
import { defineComponent } from "vue";

export default defineComponent({
    props: [&#39;title&#39;],
    emits: [&#39;update:title&#39;],
    setup(props, { emit }) {
        function setInput(val) {
            emit(&#39;update:title&#39;, val.target.value)
        }
        return {
            setInput
        }
    }
})
</script>
Nach dem Login kopieren

有了以上语法糖,我们在封装组件的时候,就可以随心所欲了,比如我自己封装可以控制显示隐藏的组件我们就能使用v-model:visible单独控制组件的显示隐藏。使用正常的v-model 控制组件内部的其他逻辑,从而拥有使用更简洁的逻辑,表达相同的功能

最后

目前开发中总结的经验就分享到这里了,错误之处,请大佬指出!

然后对vue源码有兴趣的大佬,可以看下这个文章 写给小白(自己)的vue3源码导读

也可以直接看本渣的源码解析github   vue-next-analysis

其中包含了vue源码执行思维导图,源码中的代码注释,整个源码的结构,各个功能的单独拆解等。错误之处请大佬指出!

(学习视频分享:web前端开发编程基础视频

Verwandte Etiketten:
Quelle:juejin.cn
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage