[swift] Swift 기초 - 3

Swift 2014. 10. 29. 13:32



Swift 스터디 세 번째 시간입니다. 





이번 시간에는 

구조체와 클래스, 속성과 메서드,

서브스크립트와 상속, 생성자와 소멸자에 대해서 알아보겠습니다. 

(생성자와 소멸자라고 번역하는게 맞는지 모르겠지만, 당분간 이 용어를 사용하겠습니다.)





클래스는 class 키워드를 사용하여 정의합니다.

신기하게도 Swift 에서는 클래스를 정의하면, 디폴트로 어떠한 클래스(NSObject와 같은...) 상속하지 않습니다.





콜론: 을 사용하여 클래스 상속을 표현합니다.

운송수단이라는 클래스를 상속받아 자전거 클래스를 정의했습니다.





 저장속성은 단순히 데이터를 담는 속성입니다.

Vehicle(운송수단) 클래스는 바퀴의 개수 저장속성과 승객수 저장속성을 가지고 있습니다.





계산속성은 저장속성을 바탕으로 계산되는 속성입니다. 

자동차 클래스의 스피드 속성은 최저속도, 가속도 속성을 바탕으로 

매번 계산되는 계산속성의 좋은 예입니다.

계산속성은 set, get 키워드를 사용하여 setter, getter 를 정의합니다.





운송수단 클래스에 바퀴수 저장속도를 사용하여 description 계산속성을 정의했습니다.

계산속성은 접근할 때마다 매번 가공되는(계산되는) 속성이라 할 수 있습니다.

계산속성은 선언시 반드시 타입을 정의해야 합니다.





운송수단 클래스를 사용하는 예제입니다.

멤버연산자(.dot)를 사용하여 저장속성과 계산속성에 접근합니다.





클래스는 디폴트로 init() 생성자 함수를 가지고 있습니다.

서브클래스는 생성자를 재정의(override) 할 수 있습니다.

운송수단 클래스를 상속하는 자전거 클래스는 

init 생성자를 재정의하여 바퀴수를 2로 초기화합니다





계산속성은 메소드 처럼 서브클래스에서 재정의가 가능합니다.

운송수단 클래스에서 정의한 description 계산속성을

자동차 서브클래스에서 재정의하여 사용했습니다.





재정의한 계산속성을 사용하는 예제입니다.





Swift에서는 속성 옵저버를 제공하여 쉽게 옵저버 패턴을 구현할 수 있습니다.

계산속성을 정의할 때, willSet, didSet 블럭을 지정합니다.

속성이 변경되기 전, 변경된 후에 수행할 작업을 명시하면 됩니다.

willSet 블럭에서는 새롭게 변경될 속성 값을 newValue 변수로 접근할 수 있으며,

didSet 블럭에서는 변경되기 전 속성값을 oldValue 변수로 접근할 수 있습니다.





클래스의 메서드는 함수와 동일하게 func 키워드를 사용하여 클래스 내부에 정의합니다.





Swift에서 함수는 독특하게도, 내부 파라미터 이름과 외부 파라미터 이름을 갖습니다.

내부 파라미터 이름은 함수의 몸체 즉 내부에서 사용하는 이름입니다.

외부 파라미터 이름은 함수를 사용(호출)하는 곳에서 사용하는 이름입니다.





파라미터 이름 앞에 #을 사용하면 컴파일러는 

내부 파라미터 이름과 외부 파라미터 이름을 동일하게 다룹니다.

각각 이름을 명시해야하는 번거로움을 피할 수 있습니다.





구조체 또한 클래스와 같이 속성과 메서드를 가질 수 있습니다.





구조체도 클래스처럼 생성자를 가지고 있습니다.  

다만 구조체의 디폴트 생성자는, 파라미터 이름과 포함하고 있는 속성이름이 일치하는데,

member-wise 생성자라고 합니다.





구조체도 저장속성과 메서드를 정의할 수 있습니다.





이 시점에서 생기는 의문점!  

"구조체가 속성과 메서드를 가질수 있다면 클래스와 다른게 무엇인가?" 

핵심은 클래스는 참조로 전달되고, 구조체는 값이 복사 된다는 점입니다. 

Window 클래스의 frame 속성을 할당한 newFrame의 속성을 변경해도, window.frame은 수정되지 않지만,

setup 함수로 전달된 window의 frame은 변경사항이 반영되는 것을 알수 있습니다.





var, let 키워드의 규칙은 기본형 뿐만 아니라 클래스, 구조체에도 일관되게 적용됩니다. 

다만 조금의 차이는 있습니다.

let 으로 선언된 클래스 변수에는 다른 클래스 참조를 할당하면 컴파일 에러가 발생합니다.

let 으로 선언된 클래스 변수 자체가 상수가 됩니다. 





let으로 선언된 구조체는 멤버 속성값을 변경할 수 없습니다.

즉 구조체의 멤버 속성들이 상수가 됩니다.

클래스는 참조, 구조체는 값이 복사된다는 특성을 알면 쉽게 이해할 수 있습니다.





재미있는 특성중에 하나로, 구조체의 메소드 내부에서는 속성을 변경할 수 없습니다. 

속성을 수정하려면 명시적으로 mutating 키워드를 메서드 앞에 지정해야 합니다.





앞서 구조체는 member-wise 생성자를 갖는다고 이야기했습니다.





구조체도 커스텀 생성자를 정의할 수 있습니다. 

다만 모든 속성은 사용하기 전에 초기화 되어야 합니다. 

blue 속성을 초기화 하기 전에  validateColor() 함수를 호출하면

컴파일 에러가 발생합니다.




클래스의 모든 속성도 사용하기 전에 초기화 되어야 하는 것은 동일합니다.

모든 변수는 사용하기 전에 초기화되야 한다는 것은 Swift 언어의 공통된 규칙이지요.





기본 생성자 외에 편의상의 목적으로 조금씩 다른 여러 생성자를 정의할 수 있습니다. 

Swift에서는 이런 목적의 생성자에 친절하게도 convenience 키워드를 제공합니다. 

클래스 설계자의 의도대로 생성되도록 제공하는, 의도된(designated) 생성자 외에 

편의성을 위해 제공하는 생성자라 생각하면 됩니다.

보통 convenience 생성자는 다른 생성자를 다시 호출하는 체이닝(chaining)을 통해 구현됩니다.





소멸자는 클래스의 메모리가 해제되는 시점에 사용한 자원을 정리하는 역할을 수행하는 함수입니다.

파일을 열어 사용했다면, 소멸자에서 파일을 닫도록 명시해 의도치 않은 메모리 누수를 막을 수 있습니다.





필요치 않은 자원을 조기부터 낭비하는 것을 막기위해, 

실제 자원을 사용할때 메모리를 할당하는 레이지 로딩이라는 기법이 있습니다.

Swift에서는 속성에 lazy 키워드를 사용하여 레이지 로딩 기법을 제공합니다.

lazy로 선언된 MultiplayerManager 객체는 

실제로 객체가 최초로 사용 될때(multiplayerManager.addPlayer(player) 호출 시점에)  메모리에 할당됩니다.





서브스크립트는 구조체나 클래스의 속성을  인덱스를 제공하여 접근할 수 있는 인터페이스를 제공합니다.

 subscript 키워드를 사용하여, get, set 블럭을 정의합니다.

인덱스 형태로 접근하면 다루기 쉬운 특정 데이터 구조를 다룰때 편리한 문법입니다.

대표적으로 행렬을 예로 들수 있습니다.





구구단을 출력하기 위해 TimesTable 구조체에 서브스크립트를 정의하여 사용하는 예제입니다.

서브스크립트를 사용했을  때의 2가지 장점을 엿볼 수 있습니다.

첫째는 속성에 접근하는 단순화된 문법과 둘째는, 메모리 절약입니다.

3단을 출력하기 위한 모든 데이터를 저장 할 필요없이 

인덱스와 단수 변수만을 사용하여 메모리를 절약했습니다.





행렬 구조체에 서브스크립트를 정의하고 있습니다.





메소드를 정의하여 똑같이 구현할수 있는 기능이지만 

서브스크립트를 사용하면 쉽고 가독성이 좋습니다.







'Swift' 카테고리의 다른 글

[swift2] New in Swift2.0  (0) 2015.06.19
[swift] Apple Swift Blog  (0) 2015.06.10
[swift] Swift 기초 - 4  (0) 2014.10.29
[swift] Swift 기초 - 3  (1) 2014.10.29
[swift] Swift 기초 - 2  (0) 2014.10.29
[swift] Swift 기초 - 1  (1) 2014.10.29
  • 차바 2016.12.16 11:42 ADDR 수정/삭제 답글

    다른 swift 파일에 있는 클래스 내 메소드는 어떻게 호출해야 하나요?

    클래스이름.메소드() 로 하니 에러가 나네요..ㅠㅜ

(2) 리플렉션(Reflection) 사용하기

언어로그/Java 2011. 4. 7. 19:53



지난 포스팅에 이어 리플렉션을 사용하여 객체를 생성하고, private 필드에 접근하는 방법 그리고  메서드를 호출하는
방법에 대해 알아보자.



1. 리플렉션을 사용하여 객체생성
리플렉션을 사용하여 런타임에 클래스에 생성자들을 검사하고,  생성자 객체를 통해 객체를 생성하는 과정을 알아보자.
아래와 같이 3가지 단계를 거치게 되며 가장 먼저 클래스의 생성자 객체  java.lang.reflect.Consturctor 를 얻어야 한다. 


1.1  Constructor 객체 획득하기
다음과 같이 Class 객체로부터 Constructor 클래스를 얻는다. 
// 생성자 목록 얻기 
Class aClass = ... // 이전에 얻은 클래스 객체
Constructor[] constructors = aClass.getConstructors();


Constructor[] 배열은 클래스에 선언된 모든 public 생성자의 Constructor 인스턴스를 가집니다.  특정한 파라미터를 갖는
특정한 생성자는 다음과 같이 얻을 수 있다. 
// 특정 파라미터를 갖는 생성자 얻기 
Class aClass = ... // 이전에 얻은 클래스 객체
Constructor constructor =
        aClass.getConstructor(new Class[]{String.class});


위 예는 하나의 String 타입 파라미터를 갖는 생성자를 반환하는데, 일치하는 파라미터를 갖는 생성자가 없으면 
NoSuchMethodException 예외가 발생한다.


1.2 Constructor 파라미터 얻기  
 다음과 같이 생성자에 포함된 파라미터 타입 목록을 얻을 수 있다.
// 생성자의 파라미터 타입목록 얻기
Class aClass = ... // 이전에 얻은 클래스 객체
Class[] parameterTypes = constructor.getParameterTypes();


1.3 Constructor 객체를 사용하여 객체 생성하기
다음과 같이 생성자 객체로부터 객체를 생성한다. 
// 하나의 String 파라미터를 갖는 생성자를 얻는다. 
Constructor constructor = MyObject.class.getConstructor(String.class);
// 생성 
MyObject myObject = (MyObject)constructor.newInstance("constructor-arg1");

Constructor.newInstance() 메서드는 선택적인 개수의 파라미터를 취한다.  하지만 반드시 해당 생성자에 맞는 개수와 타입의 
파라미터를 제공해야 한다는 것에 주의하자.
 


 

2. 리플렉션을 사용하여 Field 에 접근하기
리플렉션을 사용하여 클래스의 모든 멤버 변수를 검사할 수 있으며, 런타임에 값을 얻어오거나 설정할 수 있다.
이때 하나의 프로퍼티 당 하나의 java.lang.reflect.Field 클래스 객체를 사용하게 된다. 

2.1 Field  객체 얻기 
다음과 같이 Field 객체를 얻는다. 
// 클래스에 선언된 public 속성의 Field 객체얻기
Class aClass = ... // 이전에 얻은 클래스 객체
Field[] fields = aClass.getFields();


Field[] 배열은  클래스에 선언된 각 public field  당 하나의 Field 객체를 갖는다.(public field 만을 갖는다는 것에 주의)
접근하려는 필드의 이름을 안다면,  다음과 같이 접근할 수  있다. 
// Field에 접근하기 
Class  aClass = MyObject.class
Field field = aClass.getField("someField");


위 예제는 아래 MyObject 에 선언된 someField 에 대응하는 Field 인스턴스를 반환합니다. 
public class MyObject{
  public String someField = null;
}

 getField() 메서드의 파라미터에 해당하는 이름의 필드가 클래스에 존재하지 않으면  NoSuchFieldException 예외가 발생한다.


2.2 Field 이름 얻기 
Field 인스턴스를 획득하면, 다음과 같이 Field.getName()을 사용하여 이름을 얻을 수 있다.
// Field 이름 얻기
Field field = ... // Field 객체를 얻는다. 
String fieldName = field.getName();


2.3 Field  타입 얻기
Field.getType() 메서드를 사용하여 필드의 타입을 얻을 수 있다. 
// Field 타입 얻기 
Field field = aClass.getField("someField");
Object fieldType = field.getType();


2.4 Field 값 조회하고 설정하기 
Field 에 대한 참조를 얻게되면, Field.get(), Field.set() 메소드를 사용하여 값을 얻거나 설정할수 있다.
// Field 값 설정하고 조회하기
Class  aClass = MyObject.class
Field field = aClass.getField("someField");
MyObject objectInstance = new MyObject();
Object value = field.get(objectInstance);
field.set(objetInstance, value);

field.get(), set() 메서드에는 해당 필드를 소유하는 객체가 인자로 전달되야 하며 만일 static 필드라면 null을 전달한다.
 
  
2.5 Private Field 에 접근하기 
클래스의 Private 필드는 외부클래스에서 접근 할 수 없지만,  리플렉션을 사용하면 접근이 가능하다.  캡슐화를 깨는 동작일 
수 있지만, 단위테스트와 하이버네이트와 같은 프레임워크에서 유용하게 사용되기도 한다. 
private 필드에 접근하기 위해서는 Class.getDeclaredField(String name)와 Class.getDeclaredFields() 메서드를 사용한다. (Class.getField(String name) 와 Class.getFields() 메서드는 public 필드만을 반환한다.)

다음은 private field에 접근하는 예이다. 

Board board = new Board();
board.setContents("test contents...");
		
Field field = cls.getDeclaredField("contents");
field.setAccessible(true);
String contents = (String) field.get(board);
		
System.out.println("Private Contents Field: " + contents);

Field 객체의 setAccessible(true)를 호출하지 않고, private  필드값을 조회하려고 하면 IllegalAccessException 예외가 발생한다. 





3. 리플렉션을 사용하여 Method 호출하기
 java.lang.reflect.Method 클래스를 사용하여  메서드를 검사하고 호출 하는 방법을 알아보자.



3.1 Method 객체 얻기  
Method 클래스를 다음과 같이 획득한다.  
// Method 객체목록 얻기 
Class aClass = ... // 이전에 얻은 클래스 객체
Method[] methods = aClass.getMethods();


역시 클래스에 선언된 public 메서드당 하나의 Method 인스턴스를 갖으며, 메서드의 구체적인 파라미터 타입들을 알고 있으면
해당 메서드의 인스턴스를 얻을 수 있다. 다음은 String 파라미터 하나를 갖는 doSomething 메소드의 Method 객체를 얻는다.
// 파라미터를 갖는 메소드의 Method 객체 얻기 
Class aClass = ... // 이전에 얻은 클래스 객체
Method method = aClass.getMethod("doSomething", new Class[]{String.class});


만일 일치하는 메서드가 없으면, NoSuchMethodException이 발생한다. 파라미터가 없는 메서드를 얻기 위해서는 파라미터
배열 대신 null을 전달한다.
// 파라미터가 없는 메서드의 Method 객체 얻기 
Class aClass = ... // 이전에 얻은 클래스 객체
Method method = aClass.getMethod("doSomething", null);



3.2 Method 파라미터와 반환값 얻기
다음과 같이 해당 메서드의 파라미터들을 얻을 수 있다.
// 메서드의 파라미터 타입목록 얻기 
Method method = ... //
Class[] parameterTypes = method.getParameterTypes();

메서드의 반환타입은 다음과 같이 접근한다.
// 메서드의 반환값 타입 얻기 
Method method = ... //
Class returnType = method.getReturnType();



3.3  Method 객체를 사용하여 메서드 호출하기(Invoking)
다음과 같이 메서드를 호출할 수 있다.
// 메서드 호출 
Method method = MyObject.class.getMethod("doSomething", String.class);
Object returnValue = method.invoke(null, "parameter-value1");

invoke() 메소드에는 호출하기를 원하는 객체를 전달하며, static 메서드이면 null을 대신 전달한다.
Method.invoke(Object target, Object...parameters) 메서드는 선택적인 개수의 파라미터를 취하지만,  메서드가 필요로 하는
정확한 개수의 파라미터를 전달해야 한다. 


3.4 Private Method 에 접근하기 
Private Field 에 접근하는 것과 유사하게 Class.getDeclaredField(String name)와 Class.getDeclaredFields() 메서드를 사용한다. 
(Class.getField(String name) 와 Class.getFields() 메서드는 public 필드만을 반환한다.)


'언어로그 > Java' 카테고리의 다른 글

컬렉션(Collection)  (0) 2011.04.11
리플렉션으로 Getter 와 Setter 검사하기  (0) 2011.04.07
(2) 리플렉션(Reflection) 사용하기  (0) 2011.04.07
(1) 리플렉션(Reflection)  (3) 2011.04.07
예외처리 (Exception Handling)  (0) 2011.03.24
[Java] 어노테이션 사용하기  (1) 2011.03.19