Vue学习之ref属性

本文详细解读了Vue中ref属性的用途,它是HTML元素和子组件引用的替代,用于获取DOM元素对象和组件实例。通过实例代码展示了ref与id在不同情况下的行为差异。
简介

主要讲解vue中的ref属性。

直接先说结论:

  1. 专门被用来给html元素和子组件注册引用信息的(id属性的替代者)。
  2. 该属性应用在html元素时,获取的是dom元素对象,如果应用在组件标签上,那么获取的是组件实例对象(vc)。
  3. 使用方式:
    打标志:<h1 ref=“xxx”>ref</h1> 或者 <School ref=“xxx”></School>
    获取:this.$refs.xxx

代码:
在这里插入图片描述
main.js:

//引入vue依赖
import Vue from 'vue'
//引入组件App
import App from './App.vue'

// 关闭生产提示
Vue.config.productionTip = false

//创建一个vue实例
new Vue({
  //这个目前还没学过,先知道他的作用是将app放入容器中。
  render: h => h(App),

  //配置该vue实例管理id为app的容器
}).$mount('#app')

App.vue:

<template>
    <!-- 编写结构 -->
    <div>

    <!-- 分别使用id和ref -->
    <h1 ref="h1d">ref属性学习</h1>
    <h2 id="h2d">ref属性学习</h2>
    <hr>
    <school ref="schRef"></school>
    <school id="schId"></school>
    <button @click="showDom">打印dom</button>
    </div>
</template>

<script>


//引入school组件
import school from "./components/School.vue"

export default {


    components:{
        //注册组件
        school
    },
    methods:{
        showDom(){
            //根据id进行打印
            console.log(document.getElementById("h2d"));
            console.log(document.getElementById("schId"));

            //根据ref属性进行获取
            console.log(this.$refs.h1d);
            console.log(this.$refs.schRef);

        }
    }
}
</script>

<style>

</style>

School.vue

<template>
    <!-- 编写组件结构代码,也就是html代码 -->
    <!-- 需要一个div括住,也就是只能有一个根元素,一般使用div -->
    <div class="orange">
        <div>学校:{{schoolName}}</div>
        <div>地址:{{schoolAddress}}</div>
    </div>
</template>

<script>
//交互代码编写的地方,需要暴露出去给其他地方引入
//这个需要有一点es6模块化的知识,这里一般使用默认暴露,因为这是单文件组件,只需要暴露一个组件对象,
//并且引入默认暴露的组件引入语法比较简单。
export default {
    data() {
        return {
        schoolName:"尚硅谷",
        schoolAddress:"北京"
        }
    },
}
</script>

<style>
    /* 编写样式的地方 */
    .orange{
        background-color: orange;
    }
</style>

效果:
在这里插入图片描述
由上面可以得出结论:

  1. 当获取的是html元素时,ref和id的效果是一样的。
  2. 当获取的是子组件元素时,ref获取的是组件实例vc,而id获取的却是组件根元素及其后代元素。
<think>好的,我现在需要解决用户关于Vue3的ref属性的问题。用户之前询问过el-table组件和图标显示的问题,现在转向了Vue3的ref属性。我需要详细解释ref的用法、原理以及常见问题,并提供相关代码示例。首先,我应该回顾Vue3中ref的基本用法,包括如何创建响应式数据,以及在模板和脚本中的使用方式。用户可能对ref和reactive的区别有疑问,需要明确两者适用场景的不同。同时,模板引用(即使用ref属性获取DOM元素或组件实例)也是一个重点,要说明在组合式API和选项式API中的不同写法。接下来,要考虑用户可能遇到的问题,比如为什么DOM引用有时是null,或者如何正确使用组件的方法。需要解释在生命周期钩子中访问引用的时机,以及如何通过nextTick确保DOM更新后的正确访问。此外,用户可能对ref和shallowRef的区别不太清楚,需要对比说明两者的使用场景。同时,TypeScript的支持也是一个重要点,特别是如何通过泛型来指定引用的类型,避免类型错误。在代码示例方面,应该涵盖基本使用、模板引用、组件方法调用、forwardRef的使用,以及TypeScript的例子。每个示例要简洁明了,突出关键点,并注释说明注意事项。最后,提出的相关问题需要覆盖ref的常见疑问,如与reactive的区别、模板引用的使用、shallowRef的作用以及TypeScript处理,确保用户能进一步深入学习相关知识点。</think>以下是关于Vue3 ref属性的详细解析: ### 一、ref的核心作用 Vue3的ref用于创建响应式数据引用,主要处理以下场景: ```vue <script setup> import { ref } from 'vue' // 基本类型响应式 const count = ref(0) // 引用类型响应式 const user = ref({ name: '张三' }) // 模板引用 const inputRef = ref(null) </script> ``` ### 二、实现原理 1. **响应式转换**: - 通过`new Proxy()`实现数据劫持 - 对基本类型进行对象包装(value属性) - 自动触发依赖追踪和更新 2. **模板引用**: ```vue <template> <input ref="inputRef"> <child-component ref="childRef"></child-component> </template> <script setup> const inputRef = ref(null) // 获取DOM元素 const childRef = ref(null) // 获取组件实例 </script> ``` ### 三、典型使用场景 #### 场景1:基本类型响应式 ```vue <script setup> const count = ref(0) const increment = () => count.value++ // 必须使用.value </script> <template> <button @click="increment">{{ count }}</button> </template> ``` #### 场景2:DOM元素引用 ```vue <template> <input ref="myInput"> <button @click="focusInput">聚焦输入框</button> </template> <script setup> const myInput = ref(null) const focusInput = () => { myInput.value?.focus() // 安全访问 } </script> ``` #### 场景3:组件方法调用 ```vue <!-- 子组件 --> <script setup> defineExpose({ showMessage: () => console.log('Hello from child') }) </script> <!-- 父组件 --> <template> <child ref="childRef"></child> </template> <script setup> const childRef = ref(null) childRef.value?.showMessage() </script> ``` ### 四、特殊用法 #### 1. 响应式对象解构 ```vue <script setup> const state = ref({ x: 1, y: 2 }) // 保持响应性 const { x, y } = toRefs(state.value) </script> ``` #### 2. 配合forwardRef ```vue <!-- 子组件 --> <script setup> import { forwardRef } from 'vue' const props = defineProps({ msg: String }) defineExpose({ customMethod: () => {} }) </script> <template> <div ref="forwardRef">{{ msg }}</div> </template> ``` ### 五、常见问题解决方案 #### 问题1:为什么ref.value有时是null? ```vue <script setup> const el = ref(null) onMounted(() => { // 正确访问时机 console.log(el.value) // 此时DOM已挂载 }) </script> ``` #### 问题2:如何响应式更新对象? ```vue <script setup> const obj = ref({ a: 1 }) // 正确方式 obj.value = { ...obj.value, b: 2 } // 错误方式:不会触发更新 obj.value.b = 2 </script> ``` ### 六、TypeScript支持 ```typescript // 类型声明 const inputRef = ref<HTMLInputElement | null>(null) const user = ref<{ name: string }>({ name: '' }) ``` ### 实现原理细节 1. **引用包装**:通过`{ value: ... }`对象包装原始值 2. **依赖收集**:在getter中收集依赖 3. **触发更新**:在setter中触发副作用函数 4. **模板特殊处理**:编译器自动解套模板中的ref
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值