Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 11개 있습니다.)

(번역글) .NET Internals Cookbook Part 11 - Various C# riddles

이번에도 .NET Internals Cookbook 시리즈의 11번째 글을 번역한 것입니다.

(번역글) .NET Internals Cookbook Part 1 - Exceptions, filters and corrupted processes
; https://www.sysnet.pe.kr/2/0/11838

(번역글) .NET Internals Cookbook Part 2 - GC-related things
; https://www.sysnet.pe.kr/2/0/11869

(번역글) .NET Internals Cookbook Part 3 - Initialization tricks
; https://www.sysnet.pe.kr/2/0/11871

(번역글) .NET Internals Cookbook Part 4 - Type members
; https://www.sysnet.pe.kr/2/0/11872

(번역글) .NET Internals Cookbook Part 5 - Methods, parameters, modifiers
; https://www.sysnet.pe.kr/2/0/11873

(번역글) .NET Internals Cookbook Part 6 - Object internals
; https://www.sysnet.pe.kr/2/0/11874

(번역글) .NET Internals Cookbook Part 7 - Word tearing, locking and others
; https://www.sysnet.pe.kr/2/0/11876

(번역글) .NET Internals Cookbook Part 8 - C# gotchas
; https://www.sysnet.pe.kr/2/0/11877

(번역글) .NET Internals Cookbook Part 9 - Finalizers, queues, card tables and other GC stuff
; https://www.sysnet.pe.kr/2/0/11878

(번역글) .NET Internals Cookbook Part 10 - Threads, Tasks, asynchronous code and others
; https://www.sysnet.pe.kr/2/0/11879

.NET Internals Cookbook Part 11 - Various C# riddles
; https://blog.adamfurmanek.pl/2019/04/27/net-internals-cookbook-part-11/

(번역글) .NET Internals Cookbook Part 12 - Memory structure, attributes, handles
; https://www.sysnet.pe.kr/2/0/11891





75. TimeSpan의 분해능은?

문서에 따라,

The value of a TimeSpan object is the number of ticks that equal the represented time interval. A tick is equal to 100 nanoseconds, or one ten-millionth of a second.


100 나노초 단위입니다.





76. AppDomain 간에 공유되는 것은?

다음의 정보들이 공유됩니다.

  • 도메인 중립 타입의 객체들
  • PropertyInfo와 같은 리플렉션 타입들
  • 문자열
  • 스레드

다중 도메인 간에 공유되는 객체를 일컬어 "marshal-by-bleed"라고 합니다. 이런 타입들은 특히 동기화를 위한 lock을 걸 때 더 주의를 요합니다. 서로 다른 AppDomain 간의 동기화를 할 수 있으므로 한편으로는 유용하긴 하지만 무심코 lock(typeof(Foo))와 같은 식으로 리플렉션 타입에 lock을 거는 경우 전체 AppDomain 들 간에 동기화가 걸리므로 다중 웹 애플리케이션을 호스팅하는 ASP.NET과 같은 환경에서 성능 저하를 유발할 수 있습니다.

참고로, 다음은 CrossDomainData 타입의 인스턴스를 AppDomain 간에 공유해 값을 조작하는 예제입니다.

using Endjin.Assembly.ChangeDetection.Infrastructure;
using System;
using System.Linq;
using System.Reflection;
using System.Runtime;

class Program
{
    [LoaderOptimization(LoaderOptimization.MultiDomain)]
    static public void Main(string[] args)
    {
        for (int i = 0; i < 10000; i++)
        {
            var other = AppDomain.CreateDomain("Test" + i.ToString(), AppDomain.CurrentDomain.Evidence, new AppDomainSetup
            {
            });

            DomainGate gate = (DomainGate)other.CreateInstanceAndUnwrap(Assembly.GetExecutingAssembly().FullName, typeof(DomainGate).FullName);

            CrossDomainData data = new CrossDomainData();
            data.Input = Enumerable.Range(0, 10).ToList();

            DomainGate.Send(gate, data);

            Console.WriteLine("Calculation in other AppDomain got: {0}", data.Aggregate);
        }
    }
}

class DomainGate : MarshalByRefObject
{
    public void DoSomething(int gcCount, IntPtr objAddress)
    {
        if (gcCount != ObjectAddress.GCCount)
        {
            throw new NotSupportedException("During the call a GC did happen. Please try again.");
        }

        CrossDomainData data = (CrossDomainData)PtrConverter<Object>.ConvertFromIntPtr(objAddress);

        foreach (var x in data.Input)
        {
            Console.WriteLine(x);
        }

        data.Aggregate = data.Input.Aggregate((x, y) => x + y);
    }

    public static void Send(DomainGate gate, object o)
    {
        var old = GCSettings.LatencyMode;
        try
        {
            GCSettings.LatencyMode = GCLatencyMode.Batch; // try to keep the GC out of our stuff
            var addandGCCount = ObjectAddress.GetAddress(o);
            gate.DoSomething(addandGCCount.Value, addandGCCount.Key);
        }
        finally
        {
            GCSettings.LatencyMode = old;
        }
    }
}

Main 메서드의 LoaderOptimization.MultiDomain 옵션을 제거하면 CrossDomainData data;를 공유할 수 없어 다음과 같은 예외가 발생합니다.

Unhandled Exception: System.InvalidCastException: [A]CrossDomainData cannot be cast to [B]CrossDomainData. Type A originates from 'ConsoleApp1, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null' in the context 'Default' at location 'C:\temp\ConsoleApp1\bin\Debug\ConsoleApp1.exe'. Type B originates from 'ConsoleApp1, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null' in the context 'Default' at location 'C:\temp\ConsoleApp1\bin\Debug\ConsoleApp1.exe'.
   at DomainGate.DoSomething(Int32 gcCount, IntPtr objAddress) in C:\temp\ConsoleApp1\Program.cs:line 54
   at DomainGate.DoSomething(Int32 gcCount, IntPtr objAddress)
   at DomainGate.Send(DomainGate gate, Object o) in C:\temp\ConsoleApp1\Program.cs:line 75
   at Program.Main(String[] args) in C:\temp\ConsoleApp1\Program.cs:line 30





77. callback을 해제하지 않은 상태에서 Timer에 대한 별도의 참조가 없다면 삭제될까요?

다음의 예제 코드에서 확인할 수 있듯이,

using System;
using System.Threading;

public class Program
{
    public static void Main(string[] args)
    {
        var stateTimer = new Timer(t => Console.WriteLine("Timer!"), null, 50, 50);

        Thread.Sleep(70);
        stateTimer = null;
        GC.Collect();
        GC.WaitForPendingFinalizers();

        Thread.Sleep(1500);
    }
}

/* 출력 결과 (한 번만 출력)
Timer!
*/

삭제됩니다.





78. delegate {...}와 delegate() {...}의 차이점

전자의 경우는 어떤 인자라도 받을 수 있지만 후자의 경우에는 인자가 없어야만 합니다. 확인은 다음의 코드로.

using System;

namespace Program
{
    public delegate void Foo();
    public delegate void Bar(int x);

    public class Program
    {
        public static void Main(string[] args)
        {
            Foo a = delegate { };
            Bar b = delegate { };

            Foo c = delegate () { };

            // Error CS1593 Delegate 'Bar' does not take 0 arguments
            Bar d = delegate () { };
        }
    }
}





79. C# 어셈블리에서 DllMain 함수를 가질 수 있을까?

사실 C/C++에서의 DllMain과 정확히 일치하는 함수를 구현할 수는 없지만 그와 유사한 역할을 하는 메서드는 IL로 구현할 수 있습니다.

.assembly extern mscorlib
{
  .publickeytoken = (B7 7A 5C 56 19 34 E0 89 )                         // .z\V.4..
  .ver 4:0:0:0
}
.assembly 'f66978ab-21bd-416d-b1fd-2cc8d4467cef'
{
  .hash algorithm 0x00008004
  .ver 0:0:0:0
}
.module 'f66978ab-21bd-416d-b1fd-2cc8d4467cef.dll'
// MVID: {15C95762-8B92-4BE6-9ABD-A280C050496E}
.imagebase 0x10000000
.file alignment 0x00000200
.stackreserve 0x00100000
.subsystem 0x0003       // WINDOWS_CUI
.corflags 0x00000001    //  ILONLY
// Image base: 0x015B0000

.method public hidebysig specialname rtspecialname void .cctor() cil managed
{
    .maxstack  8
    IL_0000:  nop
    IL_0001:  ldstr      "Module initializer!"
    IL_0006:  call       void [mscorlib]System.Console::WriteLine(string)
    IL_000b:  nop
    IL_000c:  ret
}

.class public auto ansi beforefieldinit Program.Program
       extends [mscorlib]System.Object
{
  .method public hidebysig static void  Main(string[] args) cil managed
  {
    // 
    .entrypoint
    .maxstack  8
    IL_0000:  nop
    IL_000c:  ret
  } // end of method Program::Main

  .method public hidebysig specialname rtspecialname 
          instance void  .ctor() cil managed
  {
    // 
    .maxstack  8
    IL_0000:  ldarg.0
    IL_0001:  call       instance void [mscorlib]System.Object::.ctor()
    IL_0006:  nop
    IL_0007:  ret
  } // end of method Program::.ctor

} // end of class Program.Program

ilasm.exe로 컴파일하고 실행하면 "Module initializer"를 출력하는 것을 볼 수 있습니다.

C:\temp> ilasm test.il /quiet
test.il(21) : warning : Non-static global method '.cctor', made static

C:\temp> test
Module initializer!

.NET Reflector 등으로 확인해 보면 <Module> 타입의 cctor로 등록된 것을 확인할 수 있습니다.

static <Module>()
{
    Console.WriteLine("Module initializer!");
}

예전에 저도 Module 타입에 대해 다룬 적이 있었죠. ^^

닷넷 - <Module> 클래스의 용도
; https://www.sysnet.pe.kr/2/0/11335





80. Nullable<T> 타입을 직접 구현할 수 있을까?

Nullable 타입은 특별하게 대우를 받기 때문에 그와 동일한 타입을 C#으로 구현할 수는 없습니다. 대신 struct로 유사하게 구현할 수는 있는데요.

using System;

namespace Program
{
    public class Program
    {
        public static void Main(string[] args)
        {
            Nullable<int> i = 5;
            Console.WriteLine(i.GetType()); // System.Int32
            OwnNullable<int> j = 6;
            Console.WriteLine(j.GetType()); // OwnNullable`1[System.Int32]

            i = null;

            // Error CS0037 Cannot convert null to 'OwnNullable<int>' because it is a non - nullable value type
            // j = null;
        }
    }
}

struct OwnNullable<T>
    where T : struct
{
    private T field;
    public OwnNullable(T value)
    {
        field = value;
    }

    public static implicit operator OwnNullable<T>(T value)
    {
        return new OwnNullable<T>(value);
    }
}

보는 바와 같이 GetType의 결과도 런타임의 배려가 있다는 것을 확인할 수 있습니다. 게다가 순수 struct이기 때문에 null 대입이 안 되는 등의 문제가 있습니다.





81. 제한된 접근 제한자가 지정된 타입을 보다 공개된 접근 제한자를 갖는 타입에 상속할 수 있을까?

일반적으로는 다음과 같이 오류가 발생하지만,

public class Foo
{
    private class IBar
    {
    }

    // Error CS0060 Inconsistent accessibility: base class 'Foo.IBar' is less accessible than class 'Foo.Bar'
    //public class Bar : IBar
    //{
    //}
}

인터페이스로는 가능합니다.

public class Foo
{
    private interface IBar
    {
    }

    public class Bar : IBar
    {
    }
}

(첨부 파일은 이 글의 예제 코드를 포함합니다.)



[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 4/4/2023]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




1  2  3  4  5  6  7  8  9  10  11  12  [13]  14  15  ...
NoWriterDateCnt.TitleFile(s)
13297정성태3/26/20234350Windows: 235. Win32 - Code Modal과 UI Modal
13296정성태3/25/20233692Windows: 234. IsDialogMessage와 협업하는 WM_GETDLGCODE Win32 메시지 [1]파일 다운로드1
13295정성태3/24/20233957Windows: 233. Win32 - modeless 대화창을 modal처럼 동작하게 만드는 방법파일 다운로드1
13294정성태3/22/20234126.NET Framework: 2105. LargeAddressAware 옵션이 적용된 닷넷 32비트 프로세스의 가용 메모리 - 두 번째
13293정성태3/22/20234195오류 유형: 853. dumpbin - warning LNK4048: Invalid format file; ignored
13292정성태3/21/20234314Windows: 232. C/C++ - 일반 창에도 사용 가능한 IsDialogMessage파일 다운로드1
13291정성태3/20/20234720.NET Framework: 2104. C# Windows Forms - WndProc 재정의와 IMessageFilter 사용 시의 차이점
13290정성태3/19/20234223.NET Framework: 2103. C# - 윈도우에서 기본 제공하는 FindText 대화창 사용법파일 다운로드1
13289정성태3/18/20233419Windows: 231. Win32 - 대화창 템플릿의 2진 리소스를 읽어들여 자식 윈도우를 생성하는 방법파일 다운로드1
13288정성태3/17/20233518Windows: 230. Win32 - 대화창의 DLU 단위를 pixel로 변경하는 방법파일 다운로드1
13287정성태3/16/20233688Windows: 229. Win32 - 대화창 템플릿의 2진 리소스를 읽어들여 윈도우를 직접 띄우는 방법파일 다운로드1
13286정성태3/15/20234147Windows: 228. Win32 - 리소스에 포함된 대화창 Template의 2진 코드 해석 방법
13285정성태3/14/20233740Windows: 227. Win32 C/C++ - Dialog Procedure를 재정의하는 방법파일 다운로드1
13284정성태3/13/20233941Windows: 226. Win32 C/C++ - Dialog에서 값을 반환하는 방법파일 다운로드1
13283정성태3/12/20233481오류 유형: 852. 파이썬 - TypeError: coercing to Unicode: need string or buffer, NoneType found
13282정성태3/12/20233819Linux: 58. WSL - nohup 옵션이 필요한 경우
13281정성태3/12/20233720Windows: 225. 윈도우 바탕화면의 아이콘들이 넓게 퍼지는 경우 [2]
13280정성태3/9/20234471개발 환경 구성: 670. WSL 2에서 호스팅 중인 TCP 서버를 외부에서 접근하는 방법
13279정성태3/9/20234013오류 유형: 851. 파이썬 ModuleNotFoundError: No module named '_cffi_backend'
13278정성태3/8/20233972개발 환경 구성: 669. WSL 2의 (init이 아닌) systemd 지원 [1]
13277정성태3/6/20234636개발 환경 구성: 668. 코드 사인용 인증서 신청 및 적용 방법(예: Digicert)
13276정성태3/5/20234316.NET Framework: 2102. C# 11 - ref struct/ref field를 위해 새롭게 도입된 scoped 예약어
13275정성태3/3/20234669.NET Framework: 2101. C# 11의 ref 필드 설명
13274정성태3/2/20234260.NET Framework: 2100. C# - ref 필드로 ref struct 타입을 허용하지 않는 이유
13273정성태2/28/20233958.NET Framework: 2099. C# - 관리 포인터로서의 ref 예약어 의미
13272정성태2/27/20234216오류 유형: 850. SSMS - mdf 파일을 Attach 시킬 때 Operating system error 5: "5(Access is denied.)" 에러
1  2  3  4  5  6  7  8  9  10  11  12  [13]  14  15  ...