회차 01. Attributes - Overview of Attributes 02. Attributes - Custom Attributes |
01 | 개요(Overview of Attributes)
어트리뷰트는 클래스 안에 메타정보를 포함 시킬 수 있는 새로운 기술입니다. 어트리뷰트는 선언적 컴파일을 지원하기 때문에 코딩에 많은 이점을 주고 있으며 특히 컴포넌트를 만들 때 유용하게 쓸 수 있습니다.
실제, 어트리뷰트는 클래스 멤버변수로 관리하기에는 적절하지 않고 주로 코드 외부에서 어떤 자료형에 대한 서술이 필요할 때 많이 사용하는 편입니다.
Attributes 소개
어트리뷰트는 최근 자바 이노테이션(Annotation)의 등장으로 어트리뷰트 오리엔티드 프로그래밍(attribute oriented programming)에 대한 관심이 많아 지고 있습니다. 하지만 어트리뷰트 오리엔티드 프로그래밍은 마이크로소프트(이하 MS)진영에서 COM, COM+개발에서 부분적으로 이미 사용하고 있고 닷넷 프레임워크 경우 어트리뷰티드 프로그래밍이라는 이름으로 지원하고 있습니다.
어트리뷰트 오렌티드 프로그래밍이란 프로그램 레벨에서의 마킹 기법입니다.
이를 사용하면 프로그램을 개발할 때 프로그램과 관련된 메타데이터를 정보를 저장하고 이 정보를 이용하여 디자인&컴파일 타임 로딩 또는 런타임 시에 원하는 동작을 수행 하도록 할 수 있습니다.
예를 들면 트랜잭션(Transaction)이라는 어트리뷰트를 선언한 클래스는 클래스의 메소드를 실행할 때 트랜잭션 컨텍스트 내에서 수행 되도록 한다. 이러한 기법은 코어 비즈니스 로직과 트랜잭션과 로깅과 같은 횡단 관심사(Cross-cutting concerns)를 분리한다는 점에서 AOP(Aspect Oriented Programming) 기법과 유사합니다.
* 어트리뷰티드 프로그래밍 패러다임의 시작
어트리뷰티드 프로그래밍 기법은 COM 기반 프로그래밍에서의 인터페이스 정의 언어(Interface Definition Language)에서 사용되기 시작했다. 라이브러리와 클래스의 uuid나 helpstring같은 프로그램에 대한 메타데이터 값들을 라이브러리나 클래스의 정의에 다음과 같이 선언해 컴파일 시에 참조할 수 있도록 지원 합니다.
[uuid(74731E6F-A0E8-47FB-833C-8D6C75E85981),
helpstring("MessengerAx Control"), control]
coclass MessengerAx
helpstring("MessengerAx Control"), control]
coclass MessengerAx
이러한 기법은 COM+ 기반을 발전하면서 애플리케이션의 트랙젹선 설정과 풀리, 보안 설정 등의 메타데이터 정보를 구성요소 서비스와 같은 별도의 저장소를 이용해 등록한 후 이정보를 런타임시 참조를 해야 동작이 가능합니다.
하지만 이러한 기능을 닷넷 에서는 이보다 한 단계 더 발전된 형태로 어트리뷰트를 별도의 저장소가 아닌 프로그램 내에 직접 저장할 수 있고 동작이 가능 합니다.
어트리뷰트는 클래스 뿐만 아니라 데이터 구조, 열거자 그리고 어셈블리와 같은 프로그래밍적 요소들의 실행시 행동에 대한 정보를 기술 할수 있습니다. 어트리뷰트를 아티클을 보시는 여러분들은 코드에 대해서 어떻게 지정되고 사용되는지에 대한 주석으로 생각 할수도 있습니다. 하지만 일반 주석의 의해 전달되는 정보들은 일정한 틀이 없고 그것을 정형화하기 힘듭니다. 또한 그 정보들에 대해 런타임상의 접근하기가 어렵습니다.
어트리뷰트는 메타데이터지만 일종의 클래스입니다.
어트리뷰트를 정의해서 사용하게 되면 어떠한 클래스에 대해 설명할 때 이 어트리뷰트를 사용해서 설명할수 있고 주석처럼 직접 소스를 보아야 알수 있는 정보들을 어셈블리 안에 직접 넣거나 그 정보을 보는 방법도 코드레벨에서 가능합니다. 즉, 어트리뷰트는 작성한 클래스와 어떤 정보들(ex>이 클래스는 누가 디자인했고 라이센스는 누구한테 있으며 버전은 몇버전이다) 을 연결시켜주는 기술을 지원합니다.
* 어트리뷰트 사용
어트리뷰트를 사용할 수 있는 요소는 다음과 같습니다.
어트리뷰트를 사용할 수 있는 요소들 |
어셈블리, 모듈, 클래스, 구조체, 열거형, 변수, 생성자, 메소드, 프로퍼티, 필드, 이벤트, 인터페이스, 파라미터, 반환값, 델리게이트 |
* 어트리뷰트 문장 형식
attribute(positional_parameters, name_parameter = value, ...)]
'[ ]' 안에 무슨 메소드와 같은 것이 쓰여져 있습니다. 그리고 그 메소드 안에 파라미터들이 있습니다.
여기서 파리미터는 두가지를 의미합니다. 하나는 위치지정 파라미터(positional_parameter)로 꼭 필요한 정보들을 기술할때 쓰며 위치 지정 파라미터는 반드시 넣어줘야하는 정보들입니다. 두번째로는 명명 파라미터(name_parameter)는 꼭 필요하지 않지만 사용자에 따라서 넣을 수도 있고 그렇지 않을수도 있습니다. 어트리뷰트 안에 추가적인 정보를 집어 넣을때 사용됩니다.
* 어트리뷰트 사용 예
[AdditionalInfo("2009-05-20")]
public class Test
{
//TODO : 내용
}
여기서 AdditionalInfo는 어트리뷰트 이며 "2009-05-20"는 위치지정 파리미터가 됩니다. 개발하고자하는 코드바로위에 어트리뷰트를 써주면 됩니다.public class Test
{
//TODO : 내용
}
* 예약되어 있는 어트리뷰트
일반적인 어트리뷰트 | ||
어트리뷰트 | 적용 요소 | 설명 |
Conditional | Method | 지정된 심볼이 정의되어 있는지에 따라 함수를 실행할 것인지 아닌지를 결정한다. |
Dlllmport | Method | 명시된 Dll 파일 안의 Unmanaged 코드를 사용 할 수 있다 |
COM Interoperability 어트리뷰트 | ||
어트리뷰트 | 적용 요소 | 설명 |
ComRegisterFunction | Assembly | 닷넷 어셈블리를 COM이 사용할 수 있도록 레지스트에 등록할 때 호출되는 함수임을 나타낸다. |
ComImport | Class | 클래스나 인터페이스의 정의가 COM 타입 라이브러리로부터 가져온 것을 나타낸다. |
ComUnregisterFunction | Assembly | 닷넷 어셈블리를 레지스트리로부터 삭제할 때 호출 되는 함수임을 나타낸다. |
InterfaceType | Interface | 관리된 인터페이스가 COM에 노출 되었을 때 그 인터페이스가 IDispatch인지 IUnknown인지 또는 dual인터페이스인지 나타낸다. |
Displd | Method, Property |
함수(메소드) 나 속성에 사용될 Dispatch ID를 나타낸다. |
In | Field, Parameter |
필드나 파라미터가 input값으로 사용됨을 나타냄 |
Out | Field, Parameter |
해당 인자 값을 통해서 불려진 함수가 결과 값을 리턴할 때 마샬링을 해줘야 함을 나타낸다. |
Progld | Class | 클래스를 사용하기 위한 Prog ID를 나타낸다. |
MarshalAs | Field, Parameter |
데이터가 COM과 관리(managed)되는 사이에서 어떻게 마샬링되는지 정보를 나타낸다. |
HasDefaultInterface | Class | 클래스가 디폴트 COM 인터페이스를 가지고 있음을 명시적으로 나타낸다. |
Visual Designer Component-Building 어트리뷰트 | ||
어트리뷰트 | 적용 요소 | 설명 |
Browseable | Property, event |
프로퍼티나 이벤트가 비주얼 디자이너의 프로퍼티 창에 나타나야하는지 명시 |
DefaultEvent | Class | 컴포넌트를 위한 디폴트 이벤트를 명시한다. |
Persistable | Property | 프로퍼티 저장되어야 하는지, 저장되어야 한다면 어떻게 저장 되어야하는지 명시 |
Category | Property, Event |
프로퍼티 창에서 프로퍼티나 이벤트가 어느 카테고리에 들어가야 하는지 명시 |
Description | Property, Event |
프로퍼티나 이벤트가 선택 됐을 때 프로퍼티 창 밑에 나오는지 설명을 정의 |
Localizable | Property | 프로퍼티가 폼이 지역화되었을 때 리소스 파일에 저장되어야함을 명시 |
Persistable | Property | 프로퍼티가 저장되어야 하는지, 저장되어야 한다면 어떻게 저장되어야 하는지 명시 |
DefaultProperty | Class | 컴포넌트를 위한 디폴트 프로퍼티를 명시 |
DefaultValue | Property | 컴포넌트를 위한 기본 값이 프로퍼티임 명시 |
Bindable | Property | 컴포넌트를 쓰고 있는 클라이언트들에게 프로퍼티가 변경되었음을 알려주어야 함을 명시 |
Transaction Handling 어트리뷰트 | ||
어트리뷰트 | 적용 요소 | 설명 |
Transaction | Class | 컴포넌트가 트랜잭션을 지원하거나 트랜잭션이 필요한지 새로운 트랜잭션의 환경에서 불려져야 하는지 혹은 무시 되거나 지원이 안되는 것은 아닌지 명시한다. |
* Conditional 어트리뷰트 사용
#define DEBUG //DEBUG
//어트리뷰트 사용
// #undef DEBUG //어튜리뷰트 사용 해제
using System;
using System.Diagnostics;
namespace CSharpStudy
{
class MainClass
{
[Conditional("DEBUG")]
public static void DebugPrint()
{
Console.WriteLine("Debug");
}
[STAThread]
static void Main(string[] args)
{
DebugPrint();
}
}
}
//어트리뷰트 사용
// #undef DEBUG //어튜리뷰트 사용 해제
using System;
using System.Diagnostics;
namespace CSharpStudy
{
class MainClass
{
[Conditional("DEBUG")]
public static void DebugPrint()
{
Console.WriteLine("Debug");
}
[STAThread]
static void Main(string[] args)
{
DebugPrint();
}
}
}
C#코드는 디버깅을 지원하기 위해서 Conditonal 어트리뷰트를 사용 하며 지원을 합니다. Conditional 어트리뷰트는 사용자가 정의한 값에 의존해서 해당 메소드를 실행 시키도록 합니다. Conditional 어트리뷰트는 System.Diagnositcs 네임스페이스 안에 정의 되어 있습니다.
즉, #define을 설정해주지 않으면 실행 되지 않으며 설정시 실행이 됩니다.
Conditional 어트리뷰트는 클래스나 구조체 안에 있는 메소드에만 접착시킬 수 있다. 또한 그 메소드는 반드시 void형이어야 하며, 여러 Conditional어트리뷰트를 접착시키면 그중 하나만 위치지정 파라미터가 정의 되어 있어도 해당 메소드는 실행됩니다.
[conditional("MODEM"), Conditional("ADSL")]
static void print()
{
Console.WritleLine("인터넷을 사용할 수 있습니다");
}
static void print()
{
Console.WritleLine("인터넷을 사용할 수 있습니다");
}
* DllImport 어트리뷰트
DllImport 어트리뷰트는 C#안에서 Unmanaged 코드를 사용 할 수 있게 합니다. Unmanaged 코드란 닷넷 환경밖에서 개발된 코드를 말하며 예를들면 DLL 파일들안에 컴파일된 표준 , DllImport 어트리뷰트는 System.Runtime.InteropServices 네임스페이스안에서 정의 되어있습니다.
using System
using System.Runtime.InteopServices;
class testClass
{
[DllImport("User32.dll"]
public static extern int MessageBox(int h, string m, string c, int type);
static void Main()
{
MessageBox(0, "Hello world!", "C#", 0);
}
}
using System.Runtime.InteopServices;
class testClass
{
[DllImport("User32.dll"]
public static extern int MessageBox(int h, string m, string c, int type);
static void Main()
{
MessageBox(0, "Hello world!", "C#", 0);
}
}
다음은 Win32 API중 User32.dll안에 있는 MessageBox함수를 호출형태 입니다.
int MessageBox(HWND hWnd, LPCTSTR lpText, LPCTSTR lpCaption, UINT uType);
[DllImport("kernel32.dll")]
private static extern bool Beep(int freq, int dur);
[STAThread]
static void Main(string[] args)
{
MessageBox(0, "MessageBox Text", "DllImport Test", 2);
Beep(2600,1000);
}
private static extern bool Beep(int freq, int dur);
[STAThread]
static void Main(string[] args)
{
MessageBox(0, "MessageBox Text", "DllImport Test", 2);
Beep(2600,1000);
}
위코드는 외부의 비관리(Unmanaged)코드에 있는 DLL을 특정 프로세스 메모리에 로딩하여 해당 메소드를 호출하여 사용한 예입니다.
※ extern 키워드 : 어떤 메서드가 현재 프로그램 외부에 있음을 나타내는 키워드