[ES6] Class의 상속, super, this
Class
class
의 기본적인 형태를 알아봅시다.
class
의 개념적인 부분은 생략하고 javacript 에서 어떻게 사용되는지 위주로 다루도록 하겠습니다.
생성자는 constructor
를 통해 구현가능하고 instance
를 생성할 때는 new
를 사용하여 정의합니다.
class User {
constructor(username, password){
this.username = username;
this.password = password;
}
introduce(){
console.log(`My name is ${this.username}`);
}
}
user = new User("minsung","1234");
user.introduce();
// My name is minsung
상속
기존에 존재하던 class
를 상속받기 위해서는 extends
를 사용합니다.
class User {
constructor(username, password){
this.username = username;
this.password = password;
}
introduce(){
console.log(`My name is ${this.username}`);
}
}
class Admin extends User {
adminIntro(){
console.log(`I am ${this.username}(admin)`);
}
}
admin = new Admin("root", "root");
admin.adminIntro();
// I am root(admin)
super
prameter 를 object 로 받기 위해서
destructure
를 사용해 코드를 약간 수정하였습니다.
super
는 상속받은, 원시 클래스를 호출해줍니다.
Admin
에서 constructor
를 추가하기 위해서는 원시 클래스 User
의 contructor
를 먼저 불러와야 합니다.
class Admin extends User {
constructor({username, password, email}){
super({username, password});
this.email = email;
}
adminIntro(){
console.log(`I am ${this.username}(admin)`);
}
}
admin = new Admin({
username: "root",
password: "root",
email: "root@gmail.com"
});
admin.adminIntro();
// I am root(admin)
this
this
는 상당히 까다롭습니다.
상황에 따라 호출하는 것도 다르지만 이번 포스팅에서는 class
내에서의 this
에 대해서 알아보도록 하겠습니다.
class
안에서, 정확히는 생성자 안에서 사용되는this
는 새로 생긴 객체 묶입니다.
그렇기 때문에 user
를 만들었을 경우 생성자안에 this
는 아래와 같은 의미를 같습니다.
constructor(username, password){
this.username = username;
this.password = password;
}
constructor(username, password){
user.username = username;
user.password = password;
}
하지만 다음의 예제를 보면 this
가 다른 객체를 가르킵니다.
11, 18, 22 에서 출력하는 this
가 전부 다른 객체를 가르키게 되죠.
특정 객체에 EventListener
를 만들면 그 이벤트 핸들러 안에서의 this
는 해당 객체, 이벤트 타켓을 가르키게 됩니다. 따라서 다른 결과가 나오게 되는거죠.
class Counter {
constructor({initialNumber = 0, counterId, plusId, minusId}){
this.count = initialNumber;
this.counter = document.getElementById(counterId);
this.plusBtn = document.getElementById(plusId);
this.minusBtn = document.getElementById(minusId);
this.addEventListeners();
}
addEventListeners(){
console.log(this);
// Counter {count: 0, counter: span#count, plusBtn: button#add, minusBtn: button#minus}
this.plusBtn.addEventListener("click", this.increase);
this.minusBtn.addEventListener("click", this.decrease);
}
increase(){
console.log(this);
// <button id="add">+</button>
}
decrease(){
console.log(this);
// <button id="minus">-</button>
}
}
new Counter({counterId: "count", plusId: "add", minusId: "minus"});
이러한 현상을 방지하기 위해서, 즉 this
가 class
내부에서는 class
를 가르키게 할 수 있습니다.
바로 arrow function
을 사용하는 것입니다.
이 경우 모든 this
가 class
인 Counter
를 가르키게 됩니다.
addEventListeners(){
console.log(this);
// Counter {count: 0, counter: span#count, plusBtn: button#add, minusBtn: button#minus}
this.plusBtn.addEventListener("click", this.increase);
this.minusBtn.addEventListener("click", this.decrease);
}
increase = () => {
console.log(this);
// Counter {count: 0, counter: span#count, plusBtn: button#add, minusBtn: button#minus}
}
decrease = () => {
console.log(this);
// Counter {count: 0, counter: span#count, plusBtn: button#add, minusBtn: button#minus}
}
최종적인 코드는 다음과 같습니다.
class Counter {
constructor({initialNumber = 0, counterId, plusId, minusId}){
this.count = initialNumber;
this.counter = document.getElementById(counterId);
this.counter.innerText = initialNumber;
this.plusBtn = document.getElementById(plusId);
this.minusBtn = document.getElementById(minusId);
this.addEventListeners();
}
addEventListeners = () => {
this.plusBtn.addEventListener("click", this.increase);
this.minusBtn.addEventListener("click", this.decrease);
}
increase = () => {
this.count = this.count + 1;
this.repaintCount();
}
decrease = () => {
this.count = this.count - 1;
this.repaintCount();
}
repaintCount = () => {
this.counter.innerText = this.count;
}
}
new Counter({counterId: "count", plusId: "add", minusId: "minus"});
html 파일입니다.
<body>
<span id="count">0</span>
<button id="add">+</button>
<button id="minus">-</button>
</body>