반응형 기초
API 기본설정
이 페이지와 이후 다른 가이드의 많은 챕터에는 옵션과 컴포지션 API에 대한 다양한 컨텐츠가 포함되어 있습니다. 현재 기본 설정은 컴포지션 API입니다. 좌측 사이드바 상단에 있는 "API 스타일 설정" 스위치를 사용하여 API 스타일을 전환할 수 있습니다.
반응형 상태 설정
reactive()
함수를 사용하여 객체 또는 배열을 반응형으로 만들 수 있습니다:
js
import { reactive } from 'vue'
const state = reactive({ count: 0 })
반응형 객체는 JavaScript Proxy이며 일반 객체처럼 작동합니다. 일반 객체와 차이점은 Vue가 속성에 접근 및 반응형 객체의 변경사항을 감지할 수 있다는 것입니다. 자세한 내용이 궁금하시다면 반응형 심화에서 Vue의 반응형 시스템이 어떻게 작동하는지 설명하지만, 메인 가이드를 마친 후 읽는 것을 권장합니다.
참고: 반응형에 타입 지정하기
컴포넌트의 템플릿에서 반응형 상태를 사용하려면, 컴포넌트의 setup()'
함수에서 반응형 상태를 선언하고 반환해야 합니다:
js
import { reactive } from 'vue'
export default {
// `setup`은 컴포지션 API에서만 사용되는 특별한 훅입니다.
setup() {
const state = reactive({ count: 0 })
// 상태를 템플릿에 노출
return {
state
}
}
}
template
<div>{{ state.count }}</div>
마찬가지로 반응형 상태를 변경하는 함수를 같은 범위에서 선언하고 상태와 함께 메서드로 노출할 수 있습니다:
js
import { reactive } from 'vue'
export default {
setup() {
const state = reactive({ count: 0 })
function increment() {
state.count++
}
// 함수를 반환하는 것을 잊지 마세요.
return {
state,
increment
}
}
}
노출된 메서드는 일반적으로 이벤트 리스너로 사용됩니다:
template
<button @click="increment">
{{ state.count }}
</button>
<script setup>
setup()
훅을 통해 상태와 메서드를 수동으로 노출하는 것은 장황할 수 있습니다. 다행히 빌드 방식을 사용하지 않을 때만 이러한 방법이 필요합니다. 싱글 파일 컴포넌트(*.vue
) 사용 시, <script setup>
과 같이 표기만 하면 되므로 복잡성을 크게 단순화할 수 있습니다:
vue
<script setup>
import { reactive } from 'vue'
const state = reactive({ count: 0 })
function increment() {
state.count++
}
</script>
<template>
<button @click="increment">
{{ state.count }}
</button>
</template>
컴포넌트의 <script setup>
에서 import 또는 최상위 레벨로 선언된 변수나 함수는 해당 템플릿에서 바로 사용할 수 있습니다.
이후 가이드 문서의 컴포지션 API 스타일의 예제는 개발자가 가장 많이 사용하는 SFC +
<script setup>
문법을 사용할 것입니다.
DOM 업데이트 타이밍
반응 상태를 변경하면 DOM이 자동으로 업데이트됩니다. 하지만 DOM 업데이트는 동기적으로 적용되지 않는다는 점에 유의해야 합니다. 대신 Vue는 업데이트 주기의 "다음 틱"까지 버퍼링하여 얼마나 많은 상태 변경을 수행하든 각 컴포넌트가 한 번만 업데이트되도록 합니다.
상태 변경 후, DOM 업데이트가 완료될 때까지 기다리려면 nextTick() 전역 API를 사용할 수 있습니다:
js
import { nextTick } from 'vue'
function increment() {
state.count++
nextTick(() => {
// 업데이트된 DOM에 접근 가능
})
}
깊은 반응형
Vue는 기본적으로 반응형 상태를 내부 깊숙이 추적하므로, 중첩된 객체나 배열을 변경할 때에도 변경 사항이 감지됩니다:
js
import { reactive } from 'vue'
const obj = reactive({
nested: { count: 0 },
arr: ['foo', 'bar']
})
function mutateDeeply() {
// 변경 사항이
obj.nested.count++
obj.arr.push('baz')
}
루트 수준에서만 반응성을 추적하는 얕은 반응형 객체를 명시적으로 생성할 수도 있지만, 이는 일반적으로 고급 사용 사례에서만 필요한 경우입니다.
반응형 재정의 vs. 원본
reactive()
의 반환 값은 원본 객체와 같지 않고 원본 객체를 재정의한 프락시(Proxy)라는 점을 유의하는 것이 중요합니다.
js
const raw = {}
const proxy = reactive(raw)
// 반응형으로 재정의 된 것은 원본과 같지 않습니다.
console.log(proxy === raw) // false
프락시만 반응형입니다. 원본 객체를 변경해도 업데이트가 트리거되지 않습니다. 따라서 객체를 Vue의 반응형 시스템으로 작업할 때 가장 좋은 방법은 상태를 재정의한 프락시만 사용하는 것입니다.
프락시에 대한 일관된 접근을 보장하기 위해, 원본 객체를 reactive()
한 프락시와 프락시를 reactive()
한 프락시는 동일한 프락시를 반환하도록 동작합니다.
js
// 객체를 reactive() 한 반환 값과 프락시는 동일합니다.
console.log(reactive(raw) === proxy) // true
// 프락시를 reactive()한 반환 값과 프락시는 동일합니다.
console.log(reactive(proxy) === proxy) // true
이 규칙은 중첩된 객체에도 적용됩니다. 내부 깊숙이까지 반응형이므로 반응형 객체 내부의 중첩된 객체도 프락시입니다:
js
const proxy = reactive({})
const raw = {}
proxy.nested = raw
console.log(proxy.nested === raw) // false
reactive()
의 제한 사항
reactive()
API는 두 개의 제한 사항이 있습니다:
객체, 배열 그리고
Map
이나Set
과 같은 컬렉션 유형에만 작동합니다.string
,number
또는boolean
과 같은 기본 유형에 사용할 수 없습니다.Vue의 반응형 변경 감지는 속성에 접근함으로써 작동하므로, 항상 반응형 객체에 대한 동일한 참조를 유지해야 합니다. 즉, 첫 번째 참조에 대한 반응형 연결이 손실되기 때문에 반응형 객체를 쉽게 "교체"할 수 없음을 의미합니다.
jslet state = reactive({ count: 0 }) // 위에서 참조한 ({ count: 0 })는 더 이상 추적되지 않습니다. (반응형 연결이 끊어졌습니다.) state = reactive({ count: 1 })
또한 반응형 객체의 속성을 로컬 변수에 할당하거나 분해 할당 또는 함수에 전달할 때 반응형 연결이 끊어짐을 의미합니다:
jsconst state = reactive({ count: 0 }) // n은 state.count에서 연결이 끊긴 로컬 변수입니다. let n = state.count // 원본의 상태(state.count)에 영향을 미치지 않습니다. n++ // 로컬 변수 count는 state.count로부터 연결이 끊깁니다. let { count } = state // 원본의 상태(state.count)에 영향을 미치지 않습니다. count++ // 함수는 일반적인 숫자를 수신하며, // state.count의 변경 사항을 감지할 수 없습니다. callSomeFunction(state.count)
ref()
를 사용한 반응형 변수
Vue는 reactive()
의 제한 사항을 해결하기 위해, 어떠한 유형의 데이터라도 반응형으로 재정의할 수 있는 ref()
함수를 제공합니다:
js
import { ref } from 'vue'
const count = ref(0)
ref()
는 받은 인자를 .value
속성을 포함하는 ref 객체에 래핑 후 반환합니다:
js
const count = ref(0)
console.log(count) // { value: 0 }
console.log(count.value) // 0
count.value++
console.log(count.value) // 1
참고: Refs에 타입 지정하기
반응형 객체의 속성과 유사하게 ref의 .value
속성은 반응형입니다. 또한 객체 유형을 가지고 있는 경우, ref는 자동으로 .value
를 reactive()
로 변환합니다.
ref가 값으로 객체를 가지는 경우, 객체 전체를 반응형으로 대체할 수 있습니다:
js
const objectRef = ref({ count: 0 })
// 이것은 반응형으로 작동합니다
objectRef.value = { count: 1 }
또한 반응형 상태로 함수에 전달되거나 분해 할당될 수 있습니다:
js
const obj = {
foo: ref(1),
bar: ref(2)
}
// 함수가 ref를 전달받습니다.
// .value를 통해 값에 접근해야 하지만
// 반응형 연결 상태가 유지됩니다.
callSomeFunction(obj.foo)
// 분해 할당했지만, 반응형 상태가 유지됩니다.
const { foo, bar } = obj
즉, ref()
를 사용하면 모든 값에 대한 "참조"를 만들어 반응성을 잃지 않고 전달할 수 있습니다. 이 기능은 컴포저블 함수로 로직을 추출할 때 자주 사용되기 때문에 상당히 중요합니다.
템플릿에서 ref 언래핑
최상위 속성의 ref를 템플릿에서 접근하면 자동으로 "언래핑"되므로 .value
를 사용할 필요가 없습니다. 아래는 ref()
를 사용한 카운터 예제입니다:
vue
<script setup>
import { ref } from 'vue'
const count = ref(0)
function increment() {
count.value++
}
</script>
<template>
<button @click="increment">
{{ count }} <!-- .value가 필요하지 않습니다. -->
</button>
</template>
언래핑은 참조가 템플릿 렌더링 컨텍스트에서 최상위 프로퍼티인 경우에만 적용됩니다. 예를 들어 foo
는 최상위 프로퍼티이지만 object.foo
는 최상위 프로퍼티가 아닙니다.
따라서 아래와 같은 객체가 주어졌을 때:
js
const object = { foo: ref(1) }
아래 표현식은 예상대로 작동하지 않습니다:
template
{{ object.foo + 1 }}
object.foo
는 ref 객체이기 때문에 렌더링된 결과는 [object Object]1
가 됩니다. foo
를 최상위 속성으로 만들어 해결할 수 있습니다:
js
const { foo } = object
template
{{ foo + 1 }}
이제 렌더링 결과는 2
가 됩니다.
한 가지 주목해야 할 점은 ref가 {{ }}
또는 v-text=" "
와 같은 텍스트 보간의 최종 평가 값인 경우에도 언래핑되므로 다음은 1
이 렌더링 됩니다:
template
{{ object.foo }}
이것은 텍스트 보간의 편의 기능일 뿐이며 {{ object.foo.value }}
와 동일합니다.
반응형 객체에서 ref 언래핑
ref
가 반응형 객체의 속성으로 접근하거나 변경되면 자동으로 언래핑되어 일반 속성처럼 작동합니다:
js
const count = ref(0)
const state = reactive({
count
})
console.log(state.count) // 0
state.count = 1
console.log(count.value) // 1
ref가 할당된 기존 속성에 새 ref를 할당하면 이전 ref는 대체됩니다:
js
const otherCount = ref(2)
// 기존 ref는 이제 state.count에서 참조가 끊어집니다.
state.count = otherCount
console.log(state.count) // 2
console.log(count.value) // 1
ref의 언래핑은 깊은 반응형 객체 내부에 중첩된 경우에만 발생합니다. 얕은 반응형 객체의 속성으로 접근하는 경우에는 적용되지 않습니다.
배열 및 컬렉션에서 ref 언래핑
반응형 객체와 달리 ref를 반응형 배열의 요소로서 접근하거나 Map
과 같은 기본 컬렉션 유형에서 접근할 때 언래핑이 실행되지 않습니다:
js
const books = reactive([ref('Vue 3 Guide')])
// .value가 필요합니다
console.log(books[0].value)
const map = reactive(new Map([['count', ref(0)]]))
// .value가 필요합니다
console.log(map.get('count').value)
반응형 변환
JavaScript의 언어적 제약으로 인해 ref를 .value
와 같이 사용해야 하는 단점이 있습니다. 그러나 컴파일 시 변환을 사용해 적절한 위치에 .value
를 자동으로 추가하여 개발간 편의성을 개선할 수 있습니다. Vue는 다음과 같이 "카운터" 예제를 작성할 수 있도록 컴파일 시 변환을 제공합니다:
vue
<script setup>
let count = $ref(0)
function increment() {
// .value가 필요하지 않습니다.
count++
}
</script>
<template>
<button @click="increment">{{ count }}</button>
</template>
반응형 변환에 대한 자세한 내용은 해당 섹션에서 확인할 수 있습니다. 현재 아직 실험 단계이며 최종적으로 확정되기 전에 변경될 수 있습니다.