아디봉의.net

[펌] 02 Custums Attributes 정의 본문

C#

[펌] 02 Custums Attributes 정의

아디봉 2012. 7. 9. 10:05
02 | 사용자 정의형(Custom Attributes)
지난 작성된 글로부터 어트리뷰트란 무엇인지에 대해 알아보고 닷넷 프레임워크에서 미리 정의되어 있는 어트리뷰트에 대해서 살펴 보았습니다. 사용자 정의형 어트리뷰트에 대해서 정의와 사용법에 대해서 알아보도록 하겠습니다.

* Attributes Scope 정의
지난 아티클로 통해 Conditional Attribute에 대한 적용되는 요소에 대해서 설명을 하였습니다. 그렇다면 사용자가 직접 정의한 어트리뷰트 또한 적용 되지 않을까요? AttributeUsage를 이용한다면 작성한 어트리뷰트가 어떤 데이터형, 어떤 곳에 적용 시킬수 있을지 구체적으로 명시를 할 수 있습니다.

[System.AttributeUsage(AttributeTargets.Class)]
public class MyAttribute : Attribute
{
    //TODO
}

위 코드를 보시면 어트리뷰트를 클래스에 적용 하듯이 똑 같은 형태입니다.

이제부터 작성한 어트리뷰트는 적용시 AttributeUsage 어트리뷰트를 접미어를 선언하여 이용하시면 됩니다. 여기서 AttributeTargets.Class는 위치지정 파라미터가 됩니다. 만약 클래스가 아닌 다른 타입에 적용시 컴파일시 에러가 발생되니 유념하시길 바랍니다.
기본적으로 어트리뷰트는 System.Attribute클래스로부터 상속을 받아 정의합니다. 어트리뷰트도 하나의 클래스이므로 생성자나 멤버변수(필드), 메소드등을 가질수 있으며 상속도 가능합니다. 어트리뷰트 또한 일반적인 클래스처럼 생성자가 명시적으로 써주지 않아도 컴파일러에 의해 묵시적으로 디폴트 생성자를 가질 수 있습니다.

이 어트리뷰트의 용도는 해당 어트리뷰트를 사용할 수 있는 타겟(아래 표 참조)을 지정하고 어트리뷰트의 중복을 허용하거나 어트리뷰트에 대한 상속 허용 등을 지정할 때 사용합니다.

다음 System.AttributeTargets에 정의되어 있는 AttributeUsage안에 정의되어있는 파라미터들은 열거형 변수로 다음과 같습니다.

Member 어트리뷰트가 타겟 되어지는 곳
Class Class
Construct constructor
Delegate delegate
Enum enum
Event event
Field field
Sturct return value
Assembly struct
Property property
Parameter parameter
Method method
Interface interface
Module module
ClassMembers class, struct, enum, method, field, delegate, interface, constructor, property, event
All 모든 곳

* Attributes Class 정의(선언)
예)
[System.AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, AllowMultiple=true)]
class MyAttribute : Attribute {
    private String firstName;
    private String lastName;

    public MyAttribute(String firstName, String lastName) {
        this.firstName = firstName;
        this.lastName = lastName;
    }
}

class My2Attribute : MyAttribute {
   private String middleName;
   public String MiddleName {
       get { return middleName; }
       set { middleName = value; }
   }
public My2Attribute() : base("defaultFirstName","defaultLastName") {}
}

위 예제에서는 어트리뷰트를 클래스와 메소드 레벨에서 사용할 수 있도록 지정하였으며 중복을 허용합니다. 그리고 상속을 지정하지 않는 경우 true 값이 설정됩니다.
위 예제처럼 여러 개의 데이터형에 라벨을 지정 하려면 ‘ | ‘ 파이프 연산자로 구분해주면 됩니다.
[AttributeUsage(AttributeTargets.Method | AttributeTargets.Constructor)]

모든 데이터형에 라벨을 붙이고 싶다면 AttributeTarget.All을 사용해주시면 됩니다.
[AttributeUsage(AttributeTargets.All)]

어트리뷰트 클래스 선언 방법은 보통 클래스 선언하는 것과 동일한 방법으로 선언하고 다만 System.Attribute로 부터 상속을 받으면 그 클래스가 어트리뷰트가 됩니다.
일반 클래스오 구별하귀 위해서 클래스 이름 접미사로 ‘Attribute’를 붙이는 것을 권장합니다. 부가적인 정보를 기록한다는 의미로 다음 예제와 같이 클래스 이름을 ‘MyInfoAttribute’ 어트리뷰트를 선언하여 사용 하겠습니다.

* 어트리뷰트 클래스 선언

[AttributeUsage(AttributeTargets.Class)]
public class MyInfoAttribute : Attribute
{
       private string _korName;
       private string _engName;
       private string _birthday;

       public MyInfoAttribute(string korName, string engName, string birthday)
       {
             this._korName = korName;
             this._engName = engName;
             this._birthday = birthday;
       }
       public string KorName 
       {
             get { return this._korName; }
       }
       public string EngName
       {
             get { return this._engName; }
             set { this._engName = value; }
       }
       public string Birthday 
       {
             get { return this._birthday; }
       } 


* 어트리뷰트 사용
[MyInfo("정은성", EngName = "Jeong Eun Seong", Birthday = "1984-12-02")]    class People { }

* 어트리뷰트 정보 얻기
class Test
{
       static void Main(string[] args) 
       {
               Type type = typeof(People);
                foreach (Attribute attri in type.GetCustomAttributes(true))
                {
                         MyInfoAttribute myInfo = attri as MyInfoAttribute;
                         if (null != myInfo)
                         {
                                 Console.WriteLine("한국 이름은 {0}, \n" +
                                                            "영문 이름은 {1}, \n" +
                                                            "생년 월일은 {2}",
                                  myInfo.KorName, myInfo.EngName, myInfo.Birthday);
                         }
                }
        }
}
* 결과
한국 이름은 정은성,
영문 이름은 Jeong Eun Seong,
생년 월일은 1984-12-02


위 예제 소스를 통해 어트리뷰트 클래스 선언과 사용에 대해 보셨습니다

코드를 통해 어트리뷰트 전체 이름(MyInfoAttribute)을 사용하셔도 되며 어트리뷰트 명에서 Attribute 접미사를 제외한 이름(MyInfo)만 사용해도 됩니다.여기서
주의사항은 어트리뷰트는 클래스 이므로 생성자를 가질수 있지만 반드시 생성자는 하나이어야만 합니다. 즉, 생성자를 오버로딩을 할 수 없습니다.

만약 추가적인 데이터를 집어 넣고 싶거나 변경하고 싶다면 일반 클래스에서는 생성자를 오버로딩해서 생성자를 하나 만들어 사용할 수 있었지만 어트리뷰트 클래스에서는 생성자를 꼭 하나의 가져야 하기 때문에 동일한 방법으로 할 수가 없습니다. 하지만 꼭 넣어야 할 데이터가 있다면 위치 지정 파리미터를 이용하여 추가적 데이터의 성격을 갖는 것을 명명 파리미터로 받아 이런 문제점을 해결합니다.

위 코드 중 class People { } 위에 다음과 같은 어트리뷰트를 보시면


([MyInfo("정은성", EngName = "Jeong Eun Seong", Birthday = "1984-12-02")])


클래스에 어트리뷰트를 타겟을 한 모습을 볼 수 있습니다.

근데 무언가 다르게 보이지 않나요?어떤 것은 인수만, 어떤것은 LValue에 RValue를 할당의 형태를 볼수 있습니다. 이들의 차이점를 분류 한다면 위치지정 파라미터와 명명 파리미터로 분류 할수 있습니다.

위치지정 파리미터는 생성자에 값을 그대로 전달하면 되지만 명명 파라미터는 파라미터 이름에 값을 대입해서 생성자에 넘겨야 합니다 
Tip>사용자가 정보(ex>영어표기법이 변동이 생길 경우)를 변경/추가이 있을경우 명명 파라미터를 이용

이렇게 선언 위치는 일반적으로 대부분의 어트리뷰트는 타깃 타입의 바로 위에 선언하며 어트리뷰트 식별 (Attribute Identifier)를 이용하여 어트리뷰트의 적용 대상을 명확히 할 수 있습니다.그 외 러턴값에 어트리뷰트를 적용하고 싶다면 메소드 위에 선언할 때에 [retrun:어트리뷰트]와 같이 선언하면 되며 어셈블리에 대한 어트리뷰트의 경우 AssemblyInfo파일에 [assembly:MyAttribute(“정은성”,”JeongEunSeong”)]와같이 선언 하면 됩니다.

People 클래스에 라벨을 붙여 타겟(사용)을 하였다면 People클래스로부터 어트리뷰트 정보를 가져오는 예제에 대해서 앞서 살펴 보셨습니다. 그렇다면 어떤 원리에 의해서 가져왔을까요? 살펴보도록 하겠습니다.


Type type = typeof(People);

어트리뷰트 정보를 얻기위해서 우선 People 클래스의 특성을 알아내기 위해서 typeof 연산자를 이용하여 Type 객체를 생성합니다.

type.GetCustomAttributes(true);

Tip> Type 클래스(System.Type)에 보시면 GetCustomAttributes(bool inherit) 메소드가 존재합니다. 이 메소드는 파생된 클래스로부터 멤버의 상속 체인 여부나 재정의되는 경우 특성을 포함하는 배열을 반환합니다. 즉, 클래스에 부착된 모든 어트리뷰트를 배열로 반환 합니다.

Attributes 컬렉션들은 배열로 반환하므로 반복 제어문에 의해서  배열의 개수 만큼 벨리데이션 체크를 하여 타겟 어트리뷰트가 존재하는지 비교검사를 해야합니다.

foreach (Attribute attri in type.GetCustomAttributes(true)) {
   //TODO :
}


Attributes 컬렉션을 Attribute 타입으로 담아왔지만 어떤 어트리뷰트들이 넘어올지 모릅니다.위  예제에서는 단 하나의 어트리뷰트에 대한 정보를 기술 했지만 2개 이상의 어트리뷰트를 정의 했더라면 해당 어트리뷰트에 대한 다운 캐스팅을 하여 처리해야 합니다.
Tip> 각각에 대한 캐스팅을 올바르지 하지 않는 다면 예외처리가 발생할수 있으므로 주의 하세요 ^^;

MyInfoAttribute myInfo = attri as MyInfoAttribute

위 코드와 as 연산자를 사용하여 해당 어트리뷰트가 아닐경우 null 처리하고 null 이 아니라면 원하는 필드의 특성에 대한 정보를 조회 하실수 있습니다.^^

* 사용자 정의 어트리뷰트의 처리과정
컴파일러가 구문 실행중 어트리뷰트를 파싱하기 까지 이것을 어떻게 처리 해야할지? 궁금하지 않으신가요?
사용자 정의 어트리뷰트의 처리 과정은 다음과 같은 과정을 거쳐서 결정하게 됩니다.


사용자 정의 어트리뷰트의 처리과정
  1. 어트리뷰트 클래스 찾는다.
  2. 어트리뷰트의 범위를 체크한다.
  3. 어트리뷰트의 생성자를 체크 한다.
  4. 객체의 인스턴스를 생성한다.
  5. 명명 파라미터들을 체크한다.
  6. 명명 파리미터 값으로 필드나 프로퍼티 값을 설정한다.
  7. 어트리뷰트 클래스의 현재 상태를 저장한다.

다음 코드를보며 처리과정에 숨겨진 사실?! 알려드리겠습니다.
[AttributeUsage(AttributeTargets.Class)]
public class MyInfoAttribute : Attribute
{
    //TODO :
}
[MyInfo("정은성", EngName = "Jeong Eun Seong", Birthday = "1984-12-02")]
class People { }

코드를 보시면 Pepele 클래스위에 존재하는 MyInfo 과 어트리뷰트 정의한 이름인 MyInfoAttribute이 있습니다. 2개의 이름이 다르며 왜 달라?! 라고 의심스러울수도 있습니다. 이는 컴파일러하고 실행 하여도 아무런 문제가 없습니다. 왜 그럴까요?

OOP기법으로 설계를 하면서 Attribute를 생략하여도 아무런 문제 없도록 고안하였습니다

구체적으로 살펴본다면 컴파일시 해당 어트리뷰트를 찾게 되고 존재하지 않는다면 Attribute접미사를 붙여서 해당 어트리뷰트를 찾게됩니다. 마치 생성자를 생성할 때 묵시적으로 생성하는 개념하고 비슷하지 않나요 ^^? 이해가 되셨으면 합니다.

즉, People 클래스가 컴파일 될 때 컴파일러는 MyInfo 클래스를 찾게 되고 만약 존재하지 않는다면 컴파일러는 MyInfoAttribute 클래스를 찾게 됩니다. 그 이후 어트리뷰트가 클래스에 타겟되는 것을 허용하는지 체크를 합니다.

체크가 끝나면 생성자에게 넘겨진 파리미터와 어트리뷰트에 정의한 파라미터들이 일치하는지 검사하고 생성자에 넘겨진 값들을 가지고 객체의 인스턴스를 생성합니다. 만일 파라미터중에 명명 파라미터가 있다면 컴파일러는 어트리뷰트에서 그 파라미터와 매치되는 필드나 프로퍼티를 설정해줍니다. 그후 어트리뷰트 클래스의 현재 상태는 프로그램 요소들에 적용되는 메타 데이터로 저장이 됩니다.


* Mutilple Attribute의 사용
타겟이 되는 요소에 한개이상의 어트리뷰트 라벨을 부여 할수 있습니다.
public class CS {  
      ...(생략)   
      [conditional(“CLR”)]   
      [LOVE(“I Love C sharp”)]   
      public void method() {  }
}

하지만 제가 예제로 보여드렸던 코드에서 다음과 같이 기술한다면 에러가 발생합니다.
[MyInfo("정은성", "Jeong Eun Seong","")]
[MyInfo("데브피아", "Devpia","")]
[MyInfo("마이크로소프트", "Microsoft","")]
class People { }

왜냐하면 AllowMultiple = true로 설정을 해두지 못하여 발생되었습니다

[AttributeUsage(AttributeTargets.Class, AllowMultiple = true)]
public class MyInfoAttribute : Attribute
{
     //TODO :
}

포스팅을 마치며...

이번 포스팅은 너무나 늦게 올려드린것 같아요. 새로운 블로그로 이사 하는 과정에서 문제가 발생하여 아직 공개를 할수가 없었습니다. ^^ 용서 해주실꺼죠 ^^?
항상 좋은 하루 되시길 기원합니다.
정은성 드림
저작자 표시 비영리 변경 금지
크리에이티브 커먼즈 라이선스
Creative Commons License
Posted by LenAWeb
.NET Framework/C# Common2010/05/29 03:56

 회차
 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

이러한 기법은 COM+ 기반을 발전하면서 애플리케이션의 트랙젹선 설정과 풀리, 보안 설정 등의 메타데이터 정보를 구성요소 서비스와 같은 별도의 저장소를 이용해 등록한 후 이정보를 런타임시 참조를 해야 동작이 가능합니다.

하지만 이러한 기능을
닷넷 에서는 이보다 한 단계 더 발전된 형태로 어트리뷰트를 별도의 저장소가 아닌 프로그램 내에 직접 저장할 수 있고 동작이 가능 합니다.

어트리뷰트는 클래스 뿐만 아니라 데이터 구조, 열거자 그리고 어셈블리와 같은 프로그래밍적 요소들의 실행시 행동에 대한 정보를 기술 할수 있습니다. 어트리뷰트를 아티클을 보시는 여러분들은 코드에 대해서 어떻게 지정되고 사용되는지에 대한 주석으로 생각 할수도 있습니다. 하지만 일반 주석의 의해 전달되는 정보들은 일정한 틀이 없고 그것을 정형화하기 힘듭니다. 또한 그 정보들에 대해 런타임상의 접근하기가 어렵습니다.


어트리뷰트는 메타데이터지만 일종의 클래스입니다.
어트리뷰트를 정의해서 사용하게 되면 어떠한 클래스에 대해 설명할 때 이 어트리뷰트를 사용해서 설명할수 있고 주석처럼 직접 소스를 보아야 알수 있는 정보들을 어셈블리 안에 직접 넣거나 그 정보을 보는 방법도 코드레벨에서 가능합니다. 즉, 어트리뷰트는 작성한 클래스와 어떤 정보들(ex>이 클래스는 누가 디자인했고 라이센스는 누구한테 있으며 버전은 몇버전이다) 을 연결시켜주는 기술을 지원합니다.

* 어트리뷰트 사용
어트리뷰트를 사용할 수 있는 요소는 다음과 같습니다.
어트리뷰트를 사용할 수 있는 요소들
 어셈블리, 모듈, 클래스, 구조체, 열거형, 변수, 생성자, 메소드, 프로퍼티, 필드, 이벤트, 인터페이스, 파라미터,
 반환값, 델리게이트


* 어트리뷰트 문장 형식
attribute(positional_parameters, name_parameter = value, ...)]

'[ ]' 안에 무슨 메소드와 같은 것이 쓰여져 있습니다. 그리고 그 메소드 안에 파라미터들이 있습니다.

여기서 파리미터는 두가지를 의미합니다. 하나는 위치지정 파라미터(positional_parameter)로 꼭 필요한 정보들을 기술할때 쓰며 위치 지정 파라미터는 반드시 넣어줘야하는 정보들입니다. 두번째로는 명명 파라미터(name_parameter)는 꼭 필요하지 않지만 사용자에 따라서 넣을 수도 있고 그렇지 않을수도 있습니다. 어트리뷰트 안에 추가적인 정보를 집어 넣을때 사용됩니다.


* 어트리뷰트 사용 예
[AdditionalInfo("2009-05-20")]
public class Test
{
   //TODO :  내용
}
여기서 AdditionalInfo는 어트리뷰트 이며 "2009-05-20"는 위치지정 파리미터가 됩니다. 개발하고자하는 코드바로위에 어트리뷰트를 써주면 됩니다.

* 예약되어 있는 어트리뷰트
일반적인 어트리뷰트
어트리뷰트 적용 요소 설명
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();  
           }
       }
}


C#코드는 디버깅을 지원하기 위해서 Conditonal 어트리뷰트를 사용 하며 지원을 합니다. Conditional 어트리뷰트는 사용자가 정의한 값에 의존해서 해당 메소드를 실행 시키도록 합니다. Conditional 어트리뷰트는 System.Diagnositcs 네임스페이스 안에 정의 되어 있습니다.

즉, #define을 설정해주지 않으면 실행 되지 않으며 설정시 실행이 됩니다.
Conditional 어트리뷰트는 클래스나 구조체 안에 있는 메소드에만 접착시킬 수 있다. 또한 그 메소드는 반드시 void형이어야 하며, 여러 Conditional어트리뷰트를 접착시키면 그중 하나만 위치지정 파라미터가 정의 되어 있어도 해당 메소드는 실행됩니다.

[conditional("MODEM"), Conditional("ADSL")]  
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);  
        }  
}

다음은 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);  
     }

위코드는 외부의 비관리(Unmanaged)코드에 있는 DLL을 특정 프로세스 메모리에 로딩하여 해당 메소드를 호출하여 사용한 예입니다.

※ extern 키워드 : 어떤 메서드가 현재 프로그램 외부에 있음을 나타내는 키워드

 

출처 : http://blog.tobegin.net/11 write by 정은성

'C#' 카테고리의 다른 글

[C#] getter, setter 의 사용  (0) 2012.07.25
[펌] visual 2010 도움되는기능  (0) 2012.07.12
c# 제네릭(Generic) 완벽정리해보쟈!  (0) 2012.07.11
[펌] 01.overview of Attributes  (0) 2012.07.09
[C#] .NET remoting 서비스 정리  (0) 2012.05.30