너무 늦었지만 유용한 조사가 될 수 있습니다.
컴파일 된 코드 ( IL ) 의 내부 구조는 다음과 같습니다.
public static async Task<int> GetTestData()
{
return 12;
}
IL에서는 다음과 같이됩니다.
.method private hidebysig static class [mscorlib]System.Threading.Tasks.Task`1<int32>
GetTestData() cil managed
{
.custom instance void [mscorlib]System.Runtime.CompilerServices.AsyncStateMachineAttribute::.ctor(class [mscorlib]System.Type) = ( 01 00 28 55 73 61 67 65 4C 69 62 72 61 72 79 2E
53 74 61 72 74 54 79 70 65 2B 3C 47 65 74 54 65
73 74 44 61 74 61 3E 64 5F 5F 31 00 00 )
.custom instance void [mscorlib]System.Diagnostics.DebuggerStepThroughAttribute::.ctor() = ( 01 00 00 00 )
.maxstack 2
.locals init ([0] class UsageLibrary.StartType/'<GetTestData>d__1' V_0,
[1] valuetype [mscorlib]System.Runtime.CompilerServices.AsyncTaskMethodBuilder`1<int32> V_1)
IL_0000: newobj instance void UsageLibrary.StartType/'<GetTestData>d__1'::.ctor()
IL_0005: stloc.0
IL_0006: ldloc.0
IL_0007: call valuetype [mscorlib]System.Runtime.CompilerServices.AsyncTaskMethodBuilder`1<!0> valuetype [mscorlib]System.Runtime.CompilerServices.AsyncTaskMethodBuilder`1<int32>::Create()
IL_000c: stfld valuetype [mscorlib]System.Runtime.CompilerServices.AsyncTaskMethodBuilder`1<int32> UsageLibrary.StartType/'<GetTestData>d__1'::'<>t__builder'
IL_0011: ldloc.0
IL_0012: ldc.i4.m1
IL_0013: stfld int32 UsageLibrary.StartType/'<GetTestData>d__1'::'<>1__state'
IL_0018: ldloc.0
IL_0019: ldfld valuetype [mscorlib]System.Runtime.CompilerServices.AsyncTaskMethodBuilder`1<int32> UsageLibrary.StartType/'<GetTestData>d__1'::'<>t__builder'
IL_001e: stloc.1
IL_001f: ldloca.s V_1
IL_0021: ldloca.s V_0
IL_0023: call instance void valuetype [mscorlib]System.Runtime.CompilerServices.AsyncTaskMethodBuilder`1<int32>::Start<class UsageLibrary.StartType/'<GetTestData>d__1'>(!!0&)
IL_0028: ldloc.0
IL_0029: ldflda valuetype [mscorlib]System.Runtime.CompilerServices.AsyncTaskMethodBuilder`1<int32> UsageLibrary.StartType/'<GetTestData>d__1'::'<>t__builder'
IL_002e: call instance class [mscorlib]System.Threading.Tasks.Task`1<!0> valuetype [mscorlib]System.Runtime.CompilerServices.AsyncTaskMethodBuilder`1<int32>::get_Task()
IL_0033: ret
}
비동기 및 작업 방법없이 :
public static int GetTestData()
{
return 12;
}
됩니다 :
.method private hidebysig static int32 GetTestData() cil managed
{
.maxstack 1
.locals init ([0] int32 V_0)
IL_0000: nop
IL_0001: ldc.i4.s 12
IL_0003: stloc.0
IL_0004: br.s IL_0006
IL_0006: ldloc.0
IL_0007: ret
}
이 방법의 큰 차이점을 알 수 있습니다. 비동기 메서드 내에서 await를 사용하지 않고 비동기 메서드 (예 : API 호출 또는 이벤트 처리기) 사용에 신경 쓰지 않는 경우 좋은 아이디어는이를 일반 동기화 메서드로 변환합니다 (애플리케이션 성능을 절약 함).
업데이트 :
Microsoft Docs https://docs.microsoft.com/en-us/dotnet/standard/async-in-depth의 추가 정보도 있습니다 .
비동기 메서드는 본문에 await 키워드가 있어야합니다. 이것은 명심해야 할 중요합니다. await가 비동기 메서드의 본문에 사용되지 않는 경우 C # 컴파일러는 경고를 생성하지만 코드는 일반적인 메서드 인 것처럼 컴파일 및 실행됩니다. 비동기 메서드에 대해 C # 컴파일러에 의해 생성 된 상태 머신이 아무 것도 수행하지 못하기 때문에 이것은 매우 비효율적입니다.
async
됩니까?