이번에는 Vue.js의 Watcher, 컴포넌트 및 라우팅 제어에 대해 소개하겠습니다. Vue.js의 Watchers, 컴포넌트 및 라우팅 제어에 대한 주의사항은 무엇인가요?
대부분의 사람들은 Vue.js의 몇 가지 기본 API를 익히고 나면 이미 정상적으로 프런트엔드 웹사이트를 개발할 수 있습니다. 하지만 Vue를 사용하여 보다 효율적으로 개발하고 Vue.js 마스터가 되고 싶다면 아래에서 가르쳐드릴 5가지 요령을 진지하게 공부해야 합니다.
첫 번째 단계: 단순화된 감시자
장면 복원:
created(){ this.fetchPostList() }, watch: { searchInputValue(){ this.fetchPostList() } }
컴포넌트가 생성되면 목록을 한 번 가져오고 동시에 입력 상자를 모니터링하며 필터링된 목록이 있을 때마다 다시 가져옵니다. 시나리오가 매우 일반적입니다. 이를 최적화할 수 있는 방법이 있습니까?
이동 분석:
우선, watcher에서는 함수의 리터럴 이름을 직접 사용할 수 있습니다. 둘째, Immediate: true를 선언하면 구성 요소가 생성되는 즉시 실행된다는 의미입니다.
watch: { searchInputValue:{ handler: 'fetchPostList', immediate: true } }
두 번째 이동: 모든 구성 요소 등록을 위해 한 번만
장면 복원:
import BaseButton from './baseButton' import BaseIcon from './baseIcon' import BaseInput from './baseInput' export default { components: { BaseButton, BaseIcon, BaseInput } }
<BaseInput v-model="searchText" @keydown.enter="search" /> <BaseButton @click="search"> <BaseIcon name="search"/> </BaseButton>
기본 UI 구성 요소를 여러 개 작성하고 이러한 구성 요소를 사용해야 할 때마다 먼저 가져오고 그런 다음 구성 요소를 선언하는 것은 매우 지루합니다! 가능한 한 게으르다는 원칙을 고수하면서 최적화 방법을 찾아야 합니다!
이동 분석:
자동 동적 요구 구성 요소를 달성하려면 아티팩트 웹팩을 사용하고 require.context() 메서드를 사용하여 자체 (모듈) 컨텍스트를 생성해야 합니다. 이 방법은 검색할 폴더 디렉터리, 해당 하위 디렉터리도 검색해야 하는지 여부, 파일을 일치시킬 정규식 등 3가지 매개 변수를 사용합니다.
Components 폴더에 global.js라는 파일을 추가하고 webpack을 사용하여 이 파일에 필요한 모든 기본 구성 요소를 동적으로 패키징합니다.
import Vue from 'vue' function capitalizeFirstLetter(string) { return string.charAt(0).toUpperCase() + string.slice(1) } const requireComponent = require.context( '.', false, /\.vue$/ //找到components文件夹下以.vue命名的文件 ) requireComponent.keys().forEach(fileName => { const componentConfig = requireComponent(fileName) const componentName = capitalizeFirstLetter( fileName.replace(/^\.\//, '').replace(/\.\w+$/, '') //因为得到的filename格式是: './baseButton.vue', 所以这里我们去掉头和尾,只保留真正的文件名 ) Vue.component(componentName, componentConfig.default || componentConfig) })
마지막으로 main.js에서 'comComponents/global.js'를 가져오면 이러한 기본 구성 요소를 수동으로 소개하지 않고도 언제 어디서나 사용할 수 있습니다.
세 번째 비결: 돈을 빼앗는 라우터 키
장면 복원:
다음 장면은 정말 많은 프로그래머들의 마음을 아프게 합니다... 우선 기본적으로 모든 사람들은 Vue-router를 사용하여 제어합니다. 라우팅.
블로그 웹사이트를 작성 중이고 /post-page/a에서 /post-page/b로 점프해야 한다는 요구사항이 있다고 가정해 보겠습니다. 그런데 놀랍게도 페이지가 이동한 후 데이터가 업데이트되지 않았다는 사실을 발견했습니다. ! 그 이유는 vue-router가 이것이 동일한 컴포넌트임을 "지능적으로" 발견한 후 이 컴포넌트를 재사용하기로 결정했기 때문에 생성된 함수에 작성한 메소드가 전혀 실행되지 않았기 때문입니다. 일반적인 해결책은 다음과 같이 $route의 변경 사항을 모니터링하여 데이터를 초기화하는 것입니다.
data() { return { loading: false, error: null, post: null } }, watch: { '$route': { handler: 'resetData', immediate: true } }, methods: { resetData() { this.loading = false this.error = null this.post = null this.getPost(this.$route.params.id) }, getPost(id){ } }
버그는 해결되었지만 매번 이렇게 작성하기에는 너무 촌스러운가요? 가능하다면 게으르다는 원칙을 고수하면서 코드가 다음과 같이 작성되기를 바랍니다:
data() { return { loading: false, error: null, post: null } }, created () { this.getPost(this.$route.params.id) }, methods () { getPost(postId) { // ... } }
이동 분석:
어떻게 그러한 효과를 얻을 수 있습니까? 답은 라우터 뷰에 고유 키를 추가하는 것입니다. , 공개 구성 요소라도 URL이 변경되면 이 구성 요소가 다시 생성됩니다. (약간 성능이 떨어지긴 하지만 무한 버그를 방지합니다.) 동시에 경로의 전체 경로에 직접 키를 설정하여 일석이조라는 점에 유의하세요.
<router-view :key="$route.fullpath"></router-view>
네 번째 트릭: 전능한 렌더링 기능
장면 복원:
vue에서는 각 구성 요소에 루트 요소가 하나만 있어야 합니다. vue는 오류를 보고합니다.
<template> <li v-for="route in routes" :key="route.name" > <router-link :to="route"> {{ route.title }} </router-link> </li> </template> ERROR - Component template should contain exactly one root element. If you are using v-if on multiple elements, use v-else-if to chain them instead.
이동 분석:
해결 방법이 있을까요? 대답은 '예'입니다. 하지만 이때 템플릿 대신 HTML을 생성하려면 render() 함수를 사용해야 합니다. 실제로 js를 사용하여 html을 생성하는 이점은 매우 유연하고 강력하며 v-for 및 v-if와 같은 Vue의 제한된 기능 명령어 API를 사용하는 방법을 배울 필요가 없다는 것입니다. (reactjs는 템플릿을 완전히 삭제합니다.)
functional: true, render(h, { props }) { return props.routes.map(route => <li key={route.name}> <router-link to={route}> {route.title} </router-link> </li> ) }
다섯 번째 트릭: 높은 수준의 컴포넌트를 얻기 위한 트릭은 없습니다
핵심 포인트: 이 트릭은 무한히 강력합니다. 꼭 숙지하시기 바랍니다
우리가 컴포넌트를 작성할 때 일반적으로 우리는 모두 상위 구성 요소에서 하위 구성 요소로 일련의 소품을 전달해야 하며 동시에 상위 구성 요소는 하위 구성 요소에서 발생하는 일련의 이벤트를 수신합니다. 예:
//父组件 <BaseInput :value="value" label="密码" placeholder="请填写密码" @input="handleInput" @focus="handleFocus> </BaseInput> //子组件 <template> <label> {{ label }} <input :value="value" :placeholder="placeholder" @focus=$emit('focus', $event)" @input="$emit('input', $event.target.value)" > </label> </template>
에는 다음과 같은 최적화 지점이 있습니다.
1.每一个从父组件传到子组件的props,我们都得在子组件的Props中显式的声明才能使用。这样一来,我们的子组件每次都需要申明一大堆props, 而类似placeholer这种dom原生的property我们其实完全可以直接从父传到子,无需声明。方法如下:
<input :value="value" v-bind="$attrs" @input="$emit('input', $event.target.value)" >
$attrs包含了父作用域中不作为 prop 被识别 (且获取) 的特性绑定 (class 和 style 除外)。当一个组件没有声明任何 prop 时,这里会包含所有父作用域的绑定,并且可以通过 v-bind="$attrs" 传入内部组件——在创建更高层次的组件时非常有用。
2.注意到子组件的@focus=$emit('focus', $event)"其实什么都没做,只是把event传回给父组件而已,那其实和上面类似,我完全没必要显式地申明:
<input :value="value" v-bind="$attrs" v-on="listeners" > computed: { listeners() { return { ...this.$listeners, input: event => this.$emit('input', event.target.value) } } }
$listeners包含了父作用域中的 (不含 .native 修饰器的) v-on 事件监听器。它可以通过 v-on="$listeners" 传入内部组件——在创建更高层次的组件时非常有用。
3.需要注意的是,由于我们input并不是BaseInput这个组件的根节点,而默认情况下父作用域的不被认作 props 的特性绑定将会“回退”且作为普通的 HTML 特性应用在子组件的根元素上。所以我们需要设置inheritAttrs:false,这些默认行为将会被去掉, 以上两点的优化才能成功。
相信看了本文案例你已经掌握了方法,更多精彩请关注php中文网其它相关文章!
推荐阅读:
위 내용은 Vue.js의 감시자, 구성 요소 및 라우팅 제어의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!