Microsoft MVP성태의 닷넷 이야기
Windows: 213. 윈도우 - 싱글 스레드는 컨텍스트 스위칭이 없을까요? [링크 복사], [링크+제목 복사],
조회: 13353
글쓴 사람
정성태 (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

비밀번호

댓글 작성자
 




... 46  47  48  49  50  51  52  [53]  54  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
12610정성태4/22/202115348.NET Framework: 1046. C# - 컴파일 시점에 참조할 수 없는 타입을 포함한 이벤트 핸들러를 Reflection을 이용해 구독하는 방법파일 다운로드1
12609정성태4/22/202117831.NET Framework: 1045. C# - 런타임 시점에 이벤트 핸들러를 만들어 Reflection을 이용해 구독하는 방법파일 다운로드1
12608정성태4/21/202118369.NET Framework: 1044. C# - Generic Host를 이용해 .NET 5로 리눅스 daemon 프로그램 만드는 방법 [9]파일 다운로드1
12607정성태4/21/202115589.NET Framework: 1043. C# - 실행 시점에 동적으로 Delegate 타입을 만드는 방법파일 다운로드1
12606정성태4/21/202121414.NET Framework: 1042. C# - enum 값을 int로 암시적(implicit) 형변환하는 방법? [2]파일 다운로드1
12605정성태4/18/202116797.NET Framework: 1041. C# - AssemblyID, ModuleID를 관리 코드에서 구하는 방법파일 다운로드1
12604정성태4/18/202114715VS.NET IDE: 163. 비주얼 스튜디오 속성 창의 "Build(빌드)" / "Configuration(구성)"에서의 "활성" 의미
12603정성태4/16/202116348VS.NET IDE: 162. 비주얼 스튜디오 - 상속받은 컨트롤이 디자인 창에서 지원되지 않는 문제
12602정성태4/16/202117433VS.NET IDE: 161. x64 DLL 프로젝트의 컨트롤이 Visual Studio의 Designer에서 보이지 않는 문제 [1]
12601정성태4/15/202116471.NET Framework: 1040. C# - REST API 대신 github 클라이언트 라이브러리를 통해 프로그래밍으로 접근
12600정성태4/15/202116715.NET Framework: 1039. C# - Kubeconfig의 token 설정 및 인증서 구성을 자동화하는 프로그램
12599정성태4/14/202117490.NET Framework: 1038. C# - 인증서 및 키 파일로부터 pfx/p12 파일을 생성하는 방법파일 다운로드1
12598정성태4/14/202118052.NET Framework: 1037. openssl의 PEM 개인키 파일을 .NET RSACryptoServiceProvider에서 사용하는 방법 (2)파일 다운로드1
12597정성태4/13/202117670개발 환경 구성: 569. csproj의 내용을 공통 설정할 수 있는 Directory.Build.targets / Directory.Build.props 파일
12596정성태4/12/202116977개발 환경 구성: 568. Windows의 80 포트 점유를 해제하는 방법
12595정성태4/12/202116731.NET Framework: 1036. SQL 서버 - varbinary 타입에 대한 문자열의 CAST, CONVERT 변환을 C# 코드로 구현
12594정성태4/11/202116199.NET Framework: 1035. C# - kubectl 명령어 또는 REST API 대신 Kubernetes 클라이언트 라이브러리를 통해 프로그래밍으로 접근 [1]파일 다운로드1
12593정성태4/10/202117209개발 환경 구성: 567. Docker Desktop for Windows - kubectl proxy 없이 k8s 대시보드 접근 방법
12592정성태4/10/202116742개발 환경 구성: 566. Docker Desktop for Windows - k8s dashboard의 Kubeconfig 로그인 및 Skip 방법
12591정성태4/9/202120619.NET Framework: 1034. C# - byte 배열을 Hex(16진수) 문자열로 고속 변환하는 방법 [2]파일 다운로드1
12590정성태4/9/202116791.NET Framework: 1033. C# - .NET 4.0 이하에서 Console.IsInputRedirected 구현 [1]
12589정성태4/8/202118028.NET Framework: 1032. C# - Environment.OSVersion의 문제점 및 윈도우 운영체제의 버전을 구하는 다양한 방법 [1]
12588정성태4/7/202119752개발 환경 구성: 565. PowerShell - New-SelfSignedCertificate를 사용해 CA 인증서 생성 및 인증서 서명 방법
12587정성태4/6/202121031개발 환경 구성: 564. Windows 10 - ClickOnce 배포처럼 사용할 수 있는 MSIX 설치 파일 [1]
12586정성태4/5/202117900오류 유형: 710. Windows - Restart-Computer / shutdown 명령어 수행 시 Access is denied(E_ACCESSDENIED)
12585정성태4/5/202116853개발 환경 구성: 563. 기본 생성된 kubeconfig 파일의 내용을 새롭게 생성한 인증서로 구성하는 방법
... 46  47  48  49  50  51  52  [53]  54  55  56  57  58  59  60  ...