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

윈도우 - 싱글 스레드는 컨텍스트 스위칭이 없을까요?

아래와 같은 질문이 있군요. ^^

싱글스레드 프로그램도 컨텍스트 스위칭이 생길 수 있나요?
; https://www.sysnet.pe.kr/3/0/5755

답글에서도 언급했지만, 스레드가 컨텍스트 스위칭이 발생하는가에 대해서는 "싱글"이냐, "멀티"냐가 중요한 것이 아닙니다. 그건 운영체제의 스레드 스케줄링이 결정할 사항입니다.

그렇다면, 단순히 무한 루프를 도는 스레드가 정말 다른 코어에서 실행되는지 확인해 볼까요? ^^

윈도우의 경우 GetCurrentProcessorNumber API를 이용하면,

GetCurrentProcessorNumber function (processthreadsapi.h)
; https://learn.microsoft.com/en-us/windows/win32/api/processthreadsapi/nf-processthreadsapi-getcurrentprocessornumber

코드는 대략 다음과 같이 작성할 수 있습니다.

using System.Runtime.InteropServices;

internal class Program
{
    [DllImport("kernel32.dll")]
    private static extern int GetCurrentProcessorNumber();

    static void Main(string[] args)
    {
        HashSet<int> list = new HashSet<int>();

        while (true)
        {
            int coreIdx = GetCurrentProcessorNumber();
            if (list.Contains(coreIdx) == false)
            {
                list.Add(coreIdx);
                Console.WriteLine($"[{DateTime.Now:T}] {coreIdx}");
            }
        }
    }
}

16개 코어를 가진 제 컴퓨터에서 실행해 보면 이런 결과를 얻게 됩니다.

[오후 7:43:34] 2
[오후 7:43:34] 15
[오후 7:43:34] 0
[오후 7:43:34] 3
[오후 7:43:34] 4
[오후 7:43:35] 1
[오후 7:43:35] 5
[오후 7:43:35] 8
[오후 7:43:35] 7
[오후 7:43:35] 13
[오후 7:43:35] 11
[오후 7:43:35] 12
[오후 7:43:35] 6
[오후 7:43:36] 10
[오후 7:43:36] 14
[오후 7:43:36] 9

2초도 안 돼서, 단일 스레드의 무한 루프가 16개의 코어 모두에서 실행된 것을 알 수 있습니다.

만약 이런 결과가 싫다면, C# - IdealProcessor와 ProcessorAffinity의 차이점 글에서 설명한 대로 ProcessorAffinity를 지정하면 됩니다.

using System.Diagnostics;
using System.Runtime.InteropServices;
using System.Runtime.Versioning;

[assembly: SupportedOSPlatform("windows")]

internal class Program
{
    [DllImport("kernel32.dll")]
    private static extern int GetCurrentProcessorNumber();
    [DllImport("kernel32.dll")]
    static extern long GetCurrentThreadId();

    static void Main(string[] args)
    {
        HashSet<int> list = new HashSet<int>();
        SetIdealProcessor(3);

        while (true)
        {
            // ...[생략]...
        }
    }

    static void SetIdealProcessor(int cpuNumber)
    {
        if (cpuNumber >= Environment.ProcessorCount)
        {
            cpuNumber = 0;
        }

        foreach (ProcessThread pthread in Process.GetCurrentProcess().Threads)
        {
            if (pthread.Id == GetCurrentThreadId())
            {
                pthread.ProcessorAffinity = new IntPtr(1 << (cpuNumber));
                break;
            }
        }
    }
}

위의 프로그램을 실행하면, 이제 "3"만 출력되고 다른 코어의 숫자는 볼 수 없습니다.

[오후 7:49:55] 3

그런데, 여기서 오해하면 안 되는 사실이 있는데요, 위와 같이 1개의 코어에 고정이 되었다고 해서 context switching이 없는 것은 아닙니다. 이것은 Process Explorer와 같은 도구를 이용해 확인할 수 있는데요,

thread_ctx_switch_1.png

그러니까, 1개의 코어라고 해도 해당 스레드가 실행될 quantum 시간을 소비했으면 다른 스레드에 양보를 하게 됩니다. 따라서 현재의 스레드 문맥이 저장되고, 이후 다시 스케줄링 되면 문맥이 복원돼 실행을 계속합니다.




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 11/24/2022]

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

비밀번호

댓글 작성자
 




... 31  32  [33]  34  35  36  37  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
12818정성태8/31/20217461Windows: 198. 윈도우 - 작업 관리자에서 (tensorflow 등으로 인한) GPU 연산 부하 보는 방법
12817정성태8/31/202110094스크립트: 25. 파이썬 - 윈도우 환경에서 directml을 이용한 tensorflow의 AMD GPU 사용 방법
12816정성태8/30/202115451스크립트: 24. 파이썬 - tensorflow 2.6 NVidia GPU 사용 방법 [2]
12815정성태8/30/20218544개발 환경 구성: 602. WSL 2 - docker-desktop-data, docker-desktop (%LOCALAPPDATA%\Docker\wsl\data\ext4.vhdx) 파일을 다른 디렉터리로 옮기는 방법
12814정성태8/30/202110859.NET Framework: 1110. C# 11 - 인터페이스 내에 정적 추상 메서드 정의 가능 (DIM for Static Members) [2]파일 다운로드1
12813정성태8/29/20219060.NET Framework: 1109. C# 10 - (11) Lambda 개선파일 다운로드1
12812정성태8/28/20218684.NET Framework: 1108. C# 10 - (10) 개선된 #line 지시자
12811정성태8/27/20218915Linux: 44. 윈도우 개발자를 위한 리눅스 fork 동작 방식 설명 (파이썬 코드)
12810정성태8/27/20217718.NET Framework: 1107. .NET Core/5+에서 동적 컴파일한 C# 코드를 (Breakpoint도 활용하며) 디버깅하는 방법 - #line 지시자파일 다운로드1
12809정성태8/26/20218385.NET Framework: 1106. .NET Core/5+에서 C# 코드를 동적으로 컴파일/사용하는 방법 [1]파일 다운로드1
12808정성태8/25/20219621오류 유형: 758. go: ...: missing go.sum entry; to add it: go mod download ...
12807정성태8/25/20219603.NET Framework: 1105. C# 10 - (9) 비동기 메서드가 사용할 AsyncMethodBuilder 선택 가능파일 다운로드1
12806정성태8/24/20217244개발 환경 구성: 601. PyCharm - 다중 프로세스 디버깅 방법
12805정성태8/24/20218474.NET Framework: 1104. C# 10 - (8) 분해 구문에서 기존 변수의 재사용 가능파일 다운로드1
12804정성태8/24/20219188.NET Framework: 1103. C# 10 - (7) Source Generator V2 APIs
12803정성태8/23/20218920개발 환경 구성: 600. pip cache 디렉터리 옮기는 방법
12802정성태8/23/20219137.NET Framework: 1102. .NET Conf Mini 21.08 - WinUI 3 따라해 보기 [1]
12801정성태8/23/20218650.NET Framework: 1101. C# 10 - (6) record class 타입의 ToString 메서드를 sealed 처리 허용파일 다운로드1
12800정성태8/22/20218879개발 환경 구성: 599. PyCharm - (반대로) 원격 프로세스가 PyCharm에 디버그 연결하는 방법
12799정성태8/22/20218885.NET Framework: 1100. C# 10 - (5) 속성 패턴의 개선파일 다운로드1
12798정성태8/21/202110297개발 환경 구성: 598. PyCharm - 원격 프로세스를 디버그하는 방법
12797정성태8/21/20217979Windows: 197. TCP의 MSS(Maximum Segment Size) 크기는 고정된 것일까요?
12796정성태8/21/20218618.NET Framework: 1099. C# 10 - (4) 상수 문자열에 포맷 식 사용 가능파일 다운로드1
12795정성태8/20/20219234.NET Framework: 1098. .NET 6에 포함된 신규 BCL API - 스레드 관련
12794정성태8/20/20218722스크립트: 23. 파이썬 - WSGI를 만족하는 최소한의 구현 코드 및 PyCharm에서의 디버깅 방법 [1]
12793정성태8/20/20219396.NET Framework: 1097. C# 10 - (3) 개선된 변수 초기화 판정파일 다운로드1
... 31  32  [33]  34  35  36  37  38  39  40  41  42  43  44  45  ...