Computed
<template>
<section v-if="hasFruits">
<h1>Fruits</h1>
<ul>
<li
v-for="fruit in fruits"
:key="fruit">
</li>
</ul>
</section>
</template>
<script>
export default {
data() {
return {
fruits: [
'Apple', 'Lemon', 'Grape'
]
}
},
computed: {
hasFruit() {
return this.fruits.length > 0
}
}
}
</script>
computed는 한마디로 말해서 계산된 데이터이다.
계산된 데이터(computed)는 데이터 옵션에 정의해둔 특정한 데이터(fruits)를 추가적으로 어떤 연산(this.fruits.length > 0)을 통해서 정의를 한다. 이 정의된 값을 반환해서 사용할 수 있는 새로운 데이터(hasFruit)이다.
hasFruit은 fruits 배열 데이터의 길이가 0보다 커야만 반복문을 사용하여 데이터 값을 출력하도록 한다.
여기서, fruits 배열 데이터에 아무 값도 존재하지 않다면 h1 태그(제목) 또한 출력되지 않는다.
computed 사용 예시
<template>
<section>
<h1>Reverse Fruits</h1>
<ul>
<li
v-for="fruit in reverseFruits"
:key="fruit">
</li>
</ul>
</section>
</template>
<script>
export default {
data() {
return {
fruits: [
'Apple', 'Lemon', 'Grape'
]
}
},
computed: {
reverseFruits() {
return this.fruits.map(fruit => {
return fruit.split('').reverse().join('')
})
}
}
}
</script>
reverseFruits의 계산은 다음과 같다.
‘Apple’ ⇒ [‘A’,’p’,’p’,’l’,’e’] ⇒ [‘e’,’l’,’p’,’p’,’A’] ⇒ ‘elppA’
배열 데이터 fruit를 각 알파벳으로 쪼개고 해당 배열을 뒤짚은 다음에 다시 이어준다.
Computed 캐싱
-
캐싱하지 않은 코드
메소드를 불러올 때 마다 계산을 한다.
즉, 같은 값을 계속 불러와도 같은 연산을 계속 반복하여 값을 출력하는 것이다.<template> <h1></h1> <h1></h1> <h1></h1> </template> <script> export default { data() { return { msg: 'Hello Computed!' } }, methods: { reverseMessage() { return this.msg.split('').reverse().join('') } } } </script>
-
계산된 데이터 (캐싱)
마치 데이터처럼 적용을 하면 된다.
computed 옵션 부분에 만들어 둔 계산된 데이터는 캐싱이라는 기능이 있기 때문에 한번 연산을 해 놓은 값이 있으면 반복적으로 화면에 데이터처럼 출력할 때, 다시 한번 연산하지 않는다.
캐싱이 되어져 있기 때문에 저장된(캐싱된) 값으로 해당하는 내용을 출력해준다.
첫번째만 연산을 실행하고 그 이후의 두번째, 세번째, … 100번째는 연산을 진행하지 않고 첫번째 연산에서 찾은 값을 저장해 두었다가 출력한다.<template> <h1></h1> <h1></h1> <h1></h1> </template> <script> export default { data() { return { msg: 'Hello Computed!' } }, computed: { reversedMessage() { return this.msg.split('').reverse().join('') } } } </script>
Getter, Setter
-
Getter
값을 얻어 내는 것이다.
computed 옵션에 작성해둔 reversedMessage는 계산된 데이터로, 읽기 전용(Readonly)이다.
method의 add의+= ‘?!’
는 할당연산자로, 어떤 값을 할당하더라도 반응적으로 동작할 수 있는 구조가 아니다.
즉 reversedMessage는 내부의 기본 로직을 통해서 값을 얻어내는 용도로만 사용한다. 이것이 Getter!computed: { // Getter reversedMessage() { return this.msg.split('').reverse().join('') } }
-
Setter
값을 지정하는 것이다.
computed는 기본적으로 얻어내는 것만 가능하다.
단, vue.js 문법을 통해서 계산된 데이터 부분의 Setter를 지정해 줄 수 있다.computed: { reversedMessage: { // Getter get() { return this.msg.split('').reverse().join('') }, // Setter set(value) { this.msg = value } } }, methods: { add() { this.reversedMessage += '!?' } }
계산된 데이터 부분에 getter와 setter로 구분되는 방식이 일반적으로 잘 활용되지는 않는다. 단, Vuex(Store, 중앙 집중식 저장소)에 간간히 유용하게 사용할 수 있다.
Watch
특정한 데이터들의 변경사항을 감시하는 용도의 옵션이다.
감시하고 싶은 데이터가 있다면, 해당 데이터를 watch에 하나의 메소드처럼 만들어서 메소드의 로직으로 데이터가 변경되었을 때, 어떤 내용을 실행할지 명시하면 된다.
<script>
export default {
data() {
return {
msg: 'Hello?'
}
},
computed: {
reversedMessage() {
return this.msg.split('').reverse().join('')
}
},
watch: {
msg(newValue) {
console.log('msg:', newValue)
},
reversedMessage() {
console.log('reversedMessage:', this.reversedMessage)
}
},
methods: {
changeMessage() {
this.msg = 'Good!'
}
}
}
</script>
watch를 통해서 msg 데이터는 감시되고 있는 상황이다.
감시가 되어있다가 변경사항이 있으니까, 감시되고 있는 해당 내용(콘솔에 msg 출력)의 로직을 실행한다.
watch는 기존 데이터 뿐만이 아니라 계산된 데이터도 감시가 가능하다.
changeMessage를 통해서 msg의 원본 데이터만 수정하더라도 계산된 데이터 부분의 내용도 변경된다.
따라서 watch의 reversedMessage의 변경이 감시되어 해당 내용의 로직이 실행된다.
Comments