반응형

 

들어가기에 앞서...

const

- const 변수 : 해당 변수를 정보 변경이 불가능한 상수로 변환.

- const 함수 : 해당 함수에서 const의 위치에 따라 정보의 변환을 막거나 반환값을 상수로 바꾼다.

1. const 변수

void main()
{
    const int i = 10;
    i = 20;	//에러 발생
}
  • 선언 하는 순간의 초기화 값 이외에 다른 값을 대입하려하면 에러가 발생한다.

 

2. const 함수

class ConstClass
{
private:
    int iMember;
public:
    ConstClass() { }
    void SetNum(int i) const
    {
        iMember = i;	//에러발생
    }
};
  • 함수의 뒤에 const가 붙으면 해당 함수의 블럭에서 값의 변경이 불가능하다.
class ConstClass
{
private:
    int iMember;
public:
    ConstClass() { }
    const int* GetNum()
    {
        return &iMember;
    }
};

void main()
{
    ConstClass constClass;
    *(constClass.GetNum()) = 100;	//에러발생
}
  • 함수의 반환형 앞에 const가 붙으면 반환한 값에 대한 변경이 불가능하다.

 

 

 

생성자

- 클래스 객체가 만들어질때 자동으로 단 한번 호출되는 함수

- 반환값이 없고, 클래스명과 동일한 이름으로 함수를 만든다.

- 객체의 멤버변수를 초기화하는 목적으로 사용된다.

- 생성자도 함수의 일종이기 때문에 오버로딩이 가능하다.

- 매개변수에 default값을 설정할 수 있다.

#include <iostream>
using namespace std;

class Test
{
private:
    int aMember, bMember;
public:
    Test()
    {
        aMember = 5;
        bMember = 10;
    }
    Test(int a, int b)
    {
        aMember = a;
        bMember = b;
    }
    int GetA() { return aMember; }
    int GetB() { return bMember; }
};

void main()
{
    Test t;
    cout << "Test.a : " << t.GetA() << ", Test.b : " << t.GetB() << endl;
    
    //결과 : Test.a : 5, Test.b : 10
    
    Test t2(20, 30);
    cout << "Test.a : " << t2.GetA() << ", Test.b : " << t2.GetB() << endl;
    
    //결과 : Test.a : 20, Test.b : 30
}

 

※ 생성자를 호출할 때 주의할 점

- 생성자는 다음과 같이 호출할 수 있다.

  • Class Class
  • Class* ptrClass = new Class;
  • Class* ptrClass = new Class();

- 다음과 같은 생성자는 호출할 수 없다. (컴파일 에러가 발생한다.)

  • Class Class()

Why?

- Class Class()를 생성자의 호출 문으로 인정해 버리면, 컴파일러는 이러한 문장을 만났을 때, 이것이 객체생성문인지 함수호출문인지 구분할 수 없게되기 때문에 이러한 유형의 문장은 함수 호출에만 사용하기로 약속되어있다.

 

 

 

복사 생성자

- 동일한 클래스형의 다른 객체의 정보를 복사하여 새로운 객체를 만든다.

- 일반 변수를 초기화 할 때 다른 변수의 값을 이용하는 경우와 같다.

class Test
{
private:
    int iMember;
public:
    Test(int i)
    {
        iMember = i;
    }
    Test(Test& t)
    {
        iMember = t.iMember;
    }
    int GetNum() { return iMember; }
};

void main()
{
    Test t1(20);
    Test t2(t1);
    cout << t1.GetNum() << endl;
    cout << t2.GetNum() << endl;
    
    //결과
    //20
    //20
}

 

※ 디폴트 복사 생성자

- 복사생성자는 정의하지 않아도, 멤버대 멤버의 복사를 진행하는 디폴트 복사 생성자가 자동으로 삽입된다.

class Test
{
private:
    int iMember;
public:
    Test(int i)
    {
        iMember = i;
    }
};

void main()
{
    Test t1(20);
    Test t2(t1);
}
  • Test 클래스에 복사생성자를 정의하지 않아도t2(t1)이 문제없이 컴파일된다.
  • Test t2 = t1;으로 초기화를 시도해도 묵시적 변환으로 인해Test t2(t1);으로 변환되어서 객체가 생성된다.

- 묵시적 변환으로 인해 복사 생성자가 호출되는것을 막으려면 복사 생성자 앞에 explicit 키워드를 붙이면 대입 연산자를 이용한 객체의 생성 및 초기화는 불가능해진다.

  ※ 묵시적 변환은 복사 생성자에서만 일어나는것이 아니라 전달인자가 하나인 생성자가 있다면, 이 역시 묵시적 변환이 발생한다.

 

 

 

콜론 초기화 (멤버 이니셜라이저)

- 해당 생성자의 몸체보다 먼저 매개변수를 받아 초기화하는 방법

- 클래스 멤버변수 중에서 변수 선언과 동시에 초기화 해야하는 변수(ex. const)에 사용

class Test
{
private:
    const int iMember;
public:
    Test(int i) : iMember(i) { }
}
  • const로 선언한 iMember의 값을 생성자의 몸체에서 초기화하게되면 에러가 발생하지만 멤버 이니셜라이저를 사용하게 되면 const로 선언한 멤버의 초기화가 가능하다.
  • 좀 더 쉽게 설명하자면 다음과 같다.
//클래스에서 멤버 이니셜라이저를 사용했을때.
const int i = 20;
//클래스의 생성자 몸체에서 초기화 했을때.
const int n;
n = 20;

※ 멤버 이니셜라이저를 사용하는 것이 성능상 이점이 있다고는 하나, 요즘같이 하드웨어의 성능이 좋은 시대에서는 큰 의미를 가지지 않는다.

 

 

 

소멸자

- 클래스 객체가 소멸할 때 자동으로 호출되는 함수

- 클래스의 이름 앞에 '~'가 붙는다.

- 매개변수는 void형으로 선언되어야 하기 때문에, 오버로딩과 디폴트 값 설정이 불가능하다.

- 생성자와 마찬가지로 반환값이 없다.

- 객체의 동적할당된 멤버변수를 할당해제하는 목적으로 사용된다.

#include <iostream>
using namespace std;

class Test
{
private:
    int* iMember;
public:
    Test()
    {
        iMember* = new int(20);
    }
    ~Test()
    {
    	cout << "소멸자 호출" << endl;
        delete iMember;
    }
    void Display()
    {
        cout << *iMember << endl;
    }
};

void main()
{
    Test t;
    t.Display();
}

//결과
//20
//소멸자 호출
반응형

'Stack > C++' 카테고리의 다른 글

C++ 복사 생성자 심화  (0) 2021.10.17
C++ 파일 입출력 ofstream, ifstream  (0) 2021.10.16
C++ 동적할당 new, delete  (0) 2021.10.13
C++ 참조자(Reference)  (0) 2021.09.29
C++ 클래스, 객체, 인스턴스  (0) 2021.09.29
반응형

 

new

 - Heap 영역에 동적으로 공간을 할당해주는 연산자.

 - 사용 예시 : new Type

void main()
{	
    int* iPtr = new int;
}
  • new 연산자를 쓴 뒤 할당할 자료형을 입력하면 Heap영역에서 동적할당을 하고 시작주소를 반환받는다.
  • C의 (int*)malloc(sizeof(int))와 같다.

 

 

 

delete

 - Heap 영역에 할당된 공간을 해제하는 연산자

 - 사용 예시 : delete var*

void main()
{
    int* iPtr = new int;
    delete iPtr;
}
  • delete 연산자를 쓴 뒤 포인터 변수를 입력하면 동적할당한 공간을 반환한다.
  • C의 free()와 같다.
  • 동적배열의 경우엔 delete명령어 뒤에 []를 붙여서 사용한다.
반응형

'Stack > C++' 카테고리의 다른 글

C++ 파일 입출력 ofstream, ifstream  (0) 2021.10.16
C++ 생성자, 복사생성자, 소멸자 (+ const)  (0) 2021.10.15
C++ 참조자(Reference)  (0) 2021.09.29
C++ 클래스, 객체, 인스턴스  (0) 2021.09.29
C++ 객체지향  (0) 2021.09.28
반응형

 

참조자 (Reference)

- 할당된 어떤 하나의 메모리 공간에 다른 이름을 부여하는 것이다.

  ※ 별명이라고 생각해도 좋다.

- 참조자는 변수 앞에 &를 붙여줌으로써 선언할 수 있다.

  ※ 이미 선언된 변수 앞에 &를 붙이면 주소값을 반환하지만, 새로 선언되는 변수 앞에 &를 붙이면 참조자 선언을 뜻한다.

 

 

 

활용

- 다음과 같이 사용하게되면 num2는 num1의 참조자가 되며 두 변수는 같은 메모리 공간을 가리키게 된다.

int num1 = 10;
//참조자 num2 선언
int& num2 = num1;

 

- 즉, 같은 메모리 공간을 가리키므로 num2에 100을 대입해주면 num1, num2의 값이 100으로 변경된다.

num2 = 100;
//결과 : num1 : 100, num2 = 100

 

 

- 참조자의 수에는 제한이 없으며 참조자를 대상으로도 참조자 선언이 가능하다.

int num1 = 10;
int& num2 = num1;
int& num3 = num1;

//---------------

int num1 = 10;
int& num2 = num1;
int& num3 = num2;

  ※ 위 코드와 아래 코드 모두 동일하게 num1의 메모리 공간을 가리킨다.

  ※ 하지만, 필요 이상으로 참조자를 선언하는것은 바람직하지도 않고, 필요하지도 않다.

 

- 참조자의 선언 가능 범위

  • 참조자는 변수에 대해서만 선언 가능.
int& ref = 20;	//(X)
  • 참조자는 변수에 별명을 붙이는 것이기에 미리 선언했다가 나중에 다른 변수를 참조하는 것이 불가능하다.
int& ref;	//(X)
  • 참조자는 상수에 대해 참조가 허용되지 않는다.
const int num = 10;
int& ref = num;		//(X)
  • 참조자는 NULL로 초기화하는것이 불가능하다.
int& ref = NULL;	//(X)
  • 배열의 요소와 포인터도 변수로 간주되어 참조자의 선언이 가능하다.
//Array reference
int arr[1] = {1};
int& arrRef = arr[0];	//1

//--------------------

//Pointer reference
int num = 1;
int* ptr = &num;
int& nRef = num;	//1
int* (&pRef) = ptr;	//1

 

 

 

Call-by-value & Call-by-reference

- 참조자의 활용에는 함수가 큰 지분을 차지한다.

  • Call-by-value : 값을 인자로 전달하는 함수의 호출방식
  • Call-by-reference : 주소 값을 인자로 전달하는 함수의 호출방식

- C에서의 Call-by-reference는 주소값을 전달받아 함수 외부에 선언된 변수에 접근한다.

- C++에서의 Call-by-reference는 두가지로 구분할 수 있다.

  • 주소값을 이용한 Call-by-reference (C의 개념과 동일)
  • 참조자를 이용한 Call-by-reference

 

 

참조자를 이용한 Call-by-reference

- 다음 코드를 통해 참조자를 활용해도 Call-by-reference가 가능하다는것을 알 수 있다.

void Swap(int& n1, int& n2)
{
    int tmp = n1;
    n1 = n2;
    n2 = tmp;
}

void main()
{
    int n1 = 10;
    int n2 = 20;
    
    Swap(n1, n2);
    
    //결과 n1 : 20, n2 : 10
}

  ※ 매개변수의 경우 함수가 호출되어야 초기화가 되기때문에 별도의 에러를 발생시키지 않는다.

 

 

 

참조자를 이용한 Call-by-reference의 단점

int n = 20;
TempFunc(n);
cout << n << endl;

- 위의 코드를 기준으로 봤을때, C의 개념이면 무조건 20이 출력되지만, C++에서의 출력값은 예상할 수 없다.

  ※ TempFunc(int)의 매개변수가 참조자일 가능성이 있기 때문.

- 따라서, n의 값을 예상하기 위해서는 TempFunc(int)의 함수 원형을 일일이 확인해야하는 불편함이 생긴다.

- 이 단점은 매개변수에 const를 붙여 상수화 시켜서 매개변수의 값 변경을 막을수있다.

- 이런 불명확함 때문에 C++ 프로그래머들은 참조자를 잘 사용하지 않는다.(고 한다.)

 

 

 

상수 참조하기

- 다음과 같은 방법으로 상수의 참조가 가능하다.

const int num = 10;
const int& ref = num;

- Why?

  • 이것은 '임시객체'를 참조하기 위한 방법이다.
  • 리터럴 상수는 임시적으로 존재하는 값. 즉, 다음 행으로 넘어가면 존재하지 않는 상수다.
  • 그래서 C++은 const 참조자를 이용해서 상수를 참조할 때 '임시변수'를 만들고 이 장소에 상수를 저장해 참조하게 끔 한다.
반응형

'Stack > C++' 카테고리의 다른 글

C++ 파일 입출력 ofstream, ifstream  (0) 2021.10.16
C++ 생성자, 복사생성자, 소멸자 (+ const)  (0) 2021.10.15
C++ 동적할당 new, delete  (0) 2021.10.13
C++ 클래스, 객체, 인스턴스  (0) 2021.09.29
C++ 객체지향  (0) 2021.09.28
반응형

 

클래스 (Class)

- 여러 자료형의 변수와 함수를 묶는 사용자 정의 자료형이다.

- 사물의 특성과 행동을 소프트웨어적으로 모델링하여 데이터화한다.

- 구조체와는 다르게 접근제어 지시자를 선언하지 않을시 private로 선언된다.

#include <iostream>
using namespace std;

struct Struct
{
    int i;
};

class Class
{
    int i;
};

void main()
{
    Struct s;
    s.i = 0;
	
    Class c;
    c.i = 0; //에러가 발생한다.
}

 

 

 

접근제어 지시자

- C++에서의 접근제어 지시자는 세가지가 존재한다.

  • private : 클래스 내에서만 접근허용
  • public : 어디서든 접근허용
  • protected : 클래스 안이거나, 상속관계에 있을때 접근허용

 

 

 

클래스 (Class) / 객체 (Object) / 인스턴스 (Instance)

- Class : 객체를 설계할 때 필요한 설계도

class TempClass
{
private:
    int n;
public:
    TempClass(int n)
    {
        this->n = n;
    };
    int GetNum()
    {
        return n;
    };
};

- Object : Class가 만들어진 결과물

void main()
{
    TempClass tmpClass;
}

- Instance : 만들어진 Object에 개별적인 정보를 부여하여 만들어진 사용 가능한 상태의 정보.

void main()
{
    TempClass tempClass(10);
}

  ※ 좀 더 정확히는 메모리상에 구현된 객체를 인스턴스라고 한다. (객체가 인스턴스를 포함한다.)

  ※ 위와 같은 상태에서는 tmpClass가 객체이고, (10)이 인스턴스가 된다.

  ※ 즉, 객체는 클래스 타입으로 선언되었을 때를 의미하고, 인스턴스는 인자를 받아 메모리에 할당되어 실제 사용 가능한 상태를 말한다.

반응형

'Stack > C++' 카테고리의 다른 글

C++ 파일 입출력 ofstream, ifstream  (0) 2021.10.16
C++ 생성자, 복사생성자, 소멸자 (+ const)  (0) 2021.10.15
C++ 동적할당 new, delete  (0) 2021.10.13
C++ 참조자(Reference)  (0) 2021.09.29
C++ 객체지향  (0) 2021.09.28
반응형

 

객체지향이란?

- Object-Oriented Programming (OOP)

- 컴퓨터 프로그래밍의 패러다임 중 하나.

- 객체를 만들어 객체들 간의 상호작용을 통해 로직을 구성하는 방법이다.

  ※ 객체 : 독립된 단위

- 장점

  • 유지보수 용이 : 수정사항이 클래스 내부에 있기 때문에 쉽게 찾을 수 있다.
  • 대형프로젝트에 적합 : 모듈화가 가능하므로 다수가 참여하는 대형프로젝트에서 분담이 쉽다.

- 단점

  • 처리속도가 절차지향보다 상대적으로 느리다.
  • 설계에 많은 시간이 소요된다.

 

 

객체지향의 특징

- 추상화 (Abstration)

  • 대상의 특성을 제외한 공통점을 모아 정보화한다. (Class)

- 캡슐화 (Encapsulation)

  • 외부동작(함수)을 제외한 자신의 정보(변수)를 숨긴다. (정보은닉)

- 상속 (Inheritance)

  • 이전에 만들어진 Class의 기능을 상속을 통해 가져와 부모Class의 기능을 그대로 재활용이 가능하다.

- 다형성 (Polymorphism)

  • 약속된 부분만 유지한다면 한 요소에 여러 개념을 넣을 수 있다. (오버라이딩, 오버로딩)
반응형

'Stack > C++' 카테고리의 다른 글

C++ 파일 입출력 ofstream, ifstream  (0) 2021.10.16
C++ 생성자, 복사생성자, 소멸자 (+ const)  (0) 2021.10.15
C++ 동적할당 new, delete  (0) 2021.10.13
C++ 참조자(Reference)  (0) 2021.09.29
C++ 클래스, 객체, 인스턴스  (0) 2021.09.29

+ Recent posts