클래스의 모든 멤버 변수와 멤버 함수는 정적(static) 멤버로 지정될 수 있다. 정적 멤버 변수는 인스턴스가 생성될 때마다 독립적으로 생기는 멤버 변수와 달리 해당 클래스에 하나만 생성되고 모든 인스턴스에서 공동으로 접근할 수 있는 변수이다. 마찬가지로 정적 멤버 함수도 클래스당 하나만 생기며 모든 인스턴스에서 공동으로 호출할 수 있다.
멤버를 정적 멤버로 선언하려면 선언문 앞에 static 이라는 지정자를 붙이면 된다. 모든 멤버들이 static으로 선언될 수 있으며 정적 멤버들도 접근 지정자 (public, protected, private)을 붙일 수 있다.
1 정적 멤버 변수
class Led {public:Led();void turnOn();void turnOff();static int iCount; // 정적 멤버 변수private:byte _pin;}; |
이 예제에서 iCount는 정적 멤버 변수로 선언되었다. 정적 멤버 변수를 초기화하기 위해서는 클래스 내부에서는 불가능하다. 다음과 같이 초기화 하려고 하면 에러를 발생한다.
class Led {public:Led();void turnOn();void turnOff();static int iCount = 0; // 오류 발생private:byte _pin;}; |
따라서 클래스 외부에서 전역 변수처럼 초기화시켜야 한다.
int Led::iCount = 0; |
정적 변수를 접근하는 방법은 두 가지가 있는데 먼저 클래스 이름으로 접근하는 것이다. 이때 범위 지정 연산자인 ::을 이용한다.
Led::iCount ++; // iCount값을 하나 증가시킨다. |
또는 인스턴스를 이용해서도 접근할 수 있다.
Led led1(12), led2(11);led1.iCount = 10;led2.iCount = 30; |
비록 인스턴스 멤버 변수를 접근하는 문법과 똑같지만 서로 다른 인스턴스를 통해서 정적 변수를 참조했다고 할지라도 결국 같은 하나의 정적 변수 iCount를 접근하는 것이다.
정적 변수의 첫 번째 활용 목적은 인스턴스들 사이에서 공유할 변수를 만들고자 하는 것이다. 예를 들어서 인스턴스가 몇 개나 생성되었는지를 기록하고자 할 때 앞의 예에서 iCount를 사용하면 될 것이다. 이 경우 생성자 내에서 iCount 값을 증가시키면 새로운 인스턴스가 생성될 때마다 이 변수값이 증가되므로 이것으로 개수를 알 수 있다.
int Led::iCount = 0; // 전역에서 정적 멤버 변수 초기화Led::Led(int pin) { //생성자_pin = pin;pinMode(_pin, OUTPUT);iCount++; // 인스턴스 생성 시 하나 증가}Led::~Led(int pin) { //소멸자iCount--; // 인스턴스 소멸 시 하나 감소} |
또한 전역 변수를 쓰는 대신 관련된 클래스 내부에 정적 멤버 변수를 집어넣어서 C언어의 전역 변수처럼 사용할 수도 있다.
2 정적 멤버 함수
멤버 함수 앞에 static 이라는 키워드를 붙이면 정적 멤버 함수가 된다. 정적 변수와 마찬가지로 정적 멤버 함수는 인스턴스에 속하는 멤버 함수가 아니라 클래스에 속하는 하나의 함수이다. 정적 함수 내부에서는 오직 정적 변수만을 사용할 수 있으며 함수도 정적 함수만을 호출할 수 있다. 반대로 일반 멤버 함수에서는 정적 멤버를 접근하는데 전혀 제약이 없다. 다음 예를 보자.
class Util {public:int iA;static double dA;void getA() {iA += (int)dA; // 정적과 비정적 변수를 모두 사용가능return iA;}static void getB() { // 정적 함수return dA; //정적 변수만 사용 가능}}; |
비정적 함수 getA() 내부에서는 변수 iA와 정적 변수 dA를 사용하는데 아무런 제약이 없으나 정적함수 getB() 에서는 정적 변수 dA만 사용할 수 있다. 마찬가지로 정적 함수 내에서는 정적인 함수만 호출할 수 있으며 일반적인 멤버 함수는 호출할 수 없다.
댓글 없음:
댓글 쓰기