Microsoft MVP성태의 닷넷 이야기
.NET Framework: 369. ThreadPool.QueueUserWorkItem의 실행 지연 [링크 복사], [링크+제목 복사],
조회: 29519
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 5개 있습니다.)
(시리즈 글이 5개 있습니다.)
.NET Framework: 369. ThreadPool.QueueUserWorkItem의 실행 지연
; https://www.sysnet.pe.kr/2/0/1455

.NET Framework: 919. C# - 닷넷에서의 진정한 비동기 호출을 가능케 하는 I/O 스레드 사용법
; https://www.sysnet.pe.kr/2/0/12250

.NET Framework: 922. C# - .NET ThreadPool의 Local/Global Queue
; https://www.sysnet.pe.kr/2/0/12253

.NET Framework: 2010. C# - ThreadPool.SetMaxThreads 사용법
; https://www.sysnet.pe.kr/2/0/13058

.NET Framework: 2011. C# - CLR ThreadPool의 I/O 스레드에 작업을 맡기는 방법
; https://www.sysnet.pe.kr/2/0/13059




ThreadPool.QueueUserWorkItem의 실행 지연

최근에 아주 재미있는 질문을 하나 봤습니다. ^^

동적 background workers 생성 시간 지연 발생 이슈
; http://social.msdn.microsoft.com/Forums/ko-KR/visualcsharpko/thread/59f6af80-2903-44b0-8f07-e3c78029a246/

질문을 다시 한번 정리해 보면, 다음과 같이 BackgroundWorker 객체를 생성했을 때,

using System;
using System.ComponentModel;
using System.Windows.Forms;
using System.Threading;
using System.Diagnostics;

namespace WindowsFormsApplication1
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            for (int i = 0; i < 20; i++)
            {
                BackgroundWorker bgw = new BackgroundWorker();
                bgw.DoWork += new DoWorkEventHandler(bgw_DoWork);
                bgw.WorkerSupportsCancellation = true;
                bgw.RunWorkerAsync(DateTime.Now);
            }
        }

        void bgw_DoWork(object sender, DoWorkEventArgs e)
        {
            DateTime started = (DateTime)e.Argument;
            TimeSpan elapsed = DateTime.Now - started;

            Trace.WriteLine(Thread.CurrentThread.ManagedThreadId + ": Work started after " + elapsed);
            Thread.Sleep(1000 * 60);
        }
    }
}

이를 실행시켜 보면, BackgroundWorker.RunWorkerAsync 호출 후 bgw_DoWork가 호출되기까지 지연시간이 있음을 확인할 수 있습니다.

16: Work started after 00:00:00.0231812
11: Work started after 00:00:00.0231812
13: Work started after 00:00:00.0231812
14: Work started after 00:00:00.0231812
12: Work started after 00:00:00.0231812
6: Work started after 00:00:00.0241812
10: Work started after 00:00:00.0231812
15: Work started after 00:00:00.0231812
17: Work started after 00:00:01.0011160
18: Work started after 00:00:01.5013938
19: Work started after 00:00:02.0026778
20: Work started after 00:00:02.4813795
21: Work started after 00:00:02.9821267
22: Work started after 00:00:03.4832162
23: Work started after 00:00:03.9835948
24: Work started after 00:00:04.4842889
25: Work started after 00:00:04.9850062
26: Work started after 00:00:05.9859442
27: Work started after 00:00:06.9870558
28: Work started after 00:00:07.9884043

보는 바와 같이 위의 실행 결과에서 15번 스레드까지는 빠르게 bgw_DoWork가 호출되는 반면 그 이후로는 1초 이상의 지연이 누적되면서 발생하고 있습니다. 오~~~ 재미있지 않나요? ^^

더욱 재미있는 것은 bgw_DoWork 메소드에 포함된 Thread.Sleep(1000 * 60); 코드를 제거하면 저런 현상이 없다는 점입니다.

// Thread.Sleep(1000 * 60); 코드를 제거했을 때의 실행 결과
10: Work started after 00:00:00.0009996
11: Work started after 00:00:00.0330125
6: Work started after 00:00:00.0019996
...[생략: 모든 결과가 밀리초 내에서 실행]...
16: Work started after 00:00:00.1193636
12: Work started after 00:00:00.0079991




BackgroundWorker는 내부적으로 DoWork의 실행을 닷넷의 기본 스레드 풀을 이용해 실행합니다. 따라서, 위의 현상은 BackgroundWorker에 한정된 것이 아니고 ThreadPool.QueueUserWorkItem을 이용해서도 재현할 수 있습니다.

for (int i = 0; i < 20; i++)
{
    DateTime started = DateTime.Now;

    ThreadPool.QueueUserWorkItem(
        (obj) =>
        {
            TimeSpan elapsed = DateTime.Now - (DateTime)obj;
            Trace.WriteLine(Thread.CurrentThread.ManagedThreadId + ": Work started after " + elapsed);

            Thread.Sleep(1000 * 60);
        }, started);
}

다음은 실행 결과입니다.

12: Work started after 00:00:00.0070009
10: Work started after 00:00:00.0040013
15: Work started after 00:00:00.0140017
13: Work started after 00:00:00.0060018
14: Work started after 00:00:00.0140017
6: Work started after 00:00:00.0029826
11: Work started after 00:00:00.0040013
16: Work started after 00:00:00.0140017
17: Work started after 00:00:01.0021976
18: Work started after 00:00:01.5034076
19: Work started after 00:00:02.0042271
20: Work started after 00:00:02.5047204
21: Work started after 00:00:03.0050134
22: Work started after 00:00:03.5056883
23: Work started after 00:00:04.0073806
24: Work started after 00:00:04.5074312
25: Work started after 00:00:05.0083132
26: Work started after 00:00:06.0105310
27: Work started after 00:00:07.0122110
28: Work started after 00:00:08.0123949

정말 ^^ BackgroundWorker와 실행 지연 현상이 유사하게 나타나는 것을 확인할 수 있고, 마찬가지로 Sleep 코드를 뺀 경우의 결과도 유사합니다. 좀 더 확대해서 테스트해 보면, .NET 4.0의 Task도 ThreadPool을 기반으로 하기 때문에 마찬가지 현상이 발생합니다.

for (int i = 0; i < 20; i++)
{
    DateTime started = DateTime.Now;

    Task.Factory.StartNew((obj) =>
    {
        TimeSpan elapsed = DateTime.Now - (DateTime)obj;
        Trace.WriteLine(Thread.CurrentThread.ManagedThreadId + ": Work started after " + elapsed);

        Thread.Sleep(1000 * 60);
    }, started);
}

그런데, 왜 이런 현상이 나타나는 것일까요? 알 수 없습니다. ^^ 마이크로소프트의 내부 직원만이 이에 대해 정확한 답을 할 수 있겠지요. (혹은, 직접 BCL 코드를 분석해 보시면 됩니다.)

단지, 예상해 볼 수 있는 것은 ThreadPool의 용도가 단타성 용도로 적합하고 스레드를 늘리는 것에 신중(!)하다는 점입니다. ThreadPool에 있는 모든 스레드가 점유되었을 때 빠르게 스레드의 수를 늘리기보다는 가능한 기존 스레드의 재활용을 하려고 노력한다는 것인데요. 따라서, 해당 작업이 장시간 소요되는 것(Time-consuming operations)이라면 ThreadPool을 이용하기 보다는 Thread 객체를 만들어 사용하는 것이 더 효율적입니다.

하지만, 아이러니하게도 BackgroundWorker는 UI 반응성을 떨어뜨리지 않기 위해 장시간 소요되는 작업을 맡기는 용도로 만들어진 것이기 때문에 ThreadPool의 동작과 어울리지 않고, 오히려 내부적으로 개별 Thread를 만들어서 처리해야 하지 않았을까... 하는 생각이 듭니다. 또는 마이크로소프트 측에서 설마 BackgroundWorker를 이렇게 많이 사용하리라고는 짐작하지 못했을 수도 있고.




어쨌든, 문제가 ThreadPool과 연관이 있다는 것을 알았다면 해결책도 나온 것이나 다름없습니다. 실행 결과를 보면, 대략 처음 8개의 DoWork 메소드는 빠르게 실행되다가 이후부터 느려지는 것을 목격하게 되는데요. 왜 하필 8개일까요? ^^

그 이유는, ThreadPool이 기본적으로 유지하는 최소 스레드가 수가 8개이기 때문입니다.

int workerT, portT;
ThreadPool.GetMinThreads(out workerT, out portT);

Trace.WriteLine(string.Format("Min: worker - {0}, port - {1}", workerT, portT));
// 출력 결과
// Min: worker - 8, port - 8 (Environment.ProcessorCount에 해당)

아하~~~ 그럼 기본 스레드 수를 응용 프로그램에서 예상되는 최대치 언저리로 만들어주면 되겠군요. 예를 들어 이 글의 예제에서는 20개가 사용되었으니까, 다음과 같은 식으로 해주면 해결이 됩니다.

int workerT, portT;
int minRequire = 25; // 다른 용도로도 사용될 수 있으므로 안전하게 +5

ThreadPool.GetMinThreads(out workerT, out portT);

if (workerT < minRequire)
{
    int inc = minRequire - workerT;
    workerT += inc;
}

ThreadPool.SetMinThreads(workerT, portT);

이렇게 SetMinThreads를 호출하고 나면 ThreadPool.QueueUserWorkItem, Task.Factory.StartNew, BackgroundWorker의 모든 수행 결과가 개선되는 것을 확인할 수 있습니다.

10: Work started after 00:00:00.0129996
13: Work started after 00:00:00.0200018
14: Work started after 00:00:00.0260009
...[생략: 모든 결과가 밀리 초 내에서 실행]...
16: Work started after 00:00:00.0470040
15: Work started after 00:00:00.0300031
12: Work started after 00:00:00.0330018

첨부된 파일은 위의 결과를 테스트 해볼 수 있는 간단한 프로젝트입니다.

(그나저나... IT 분야에서도 "닥터 하우스"에 나오는 것처럼 "진단학과"가 있었으면 좋겠군요. ^^)




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

[연관 글]






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

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

비밀번호

댓글 작성자
 



2013-06-04 12시48분
[ing™] 제가 알고 있기로는 ThreadPool이 기본적으로 유지하는 최소 스레드가 수는 현재 CPU의 특성에 따라 다르게 되는 것으로 알고 있습니다. CPU Core(하이퍼쓰레드 포함) 갯수대로 초기 스레드가 수가 결정이 된답니다.
[guest]
2013-06-04 01시10분
넵. ^^ 이 글의 테스트 결과에 의하면, CPU Core(HT 포함) 수만큼 초기 스레드가 결정되고 그것을 초과해서 요구했을 때 0.5초마다 CPU Core(HT 포함) 수만큼 생성되다가, 다시 그 수를 넘으면 1초마다 생성되는 규칙이 있는 것 같습니다. (CPU Core 수만큼의 규칙대로 0.5초씩 늘어나는 것 같기도 하고요.)

혹시, 이런 규칙이 설명된 공식 문서가 어디 있는지 아시나요?
정성태
2013-07-17 11시50분
[ing™] 스택에서 본건 있지만 공식 문서가 어디에 있는지는 저도 잘 모르겠습니다.
그리고 여러가지로 테스트를 해 본결과 쓰레드풀에서는 기본적으로 재 사용을 하지만 쓰레드 안에 Waiting 하는(DB Query or Sleep, HTTPClient load) 작업이 1초이상 걸리면 새로운 쓰레드를 생성하고 있습니다.
[guest]
2015-11-03 01시04분
이와 관련된 WCF에서의 현상을 다음의 질문에서 볼 수 있습니다.

BottleNeck/delay when Calling WCF service
; http://stackoverflow.com/questions/33402316/bottleneck-delay-when-calling-wcf-service/
정성태

... [151]  152  153  154  155  156  157  158  159  160  161  162  163  164  165  ...
NoWriterDateCnt.TitleFile(s)
1308정성태7/3/201227289.NET Framework: 330. IEnumerator는 언제나 읽기 전용일까?파일 다운로드1
1307정성태6/30/201229628개발 환경 구성: 154. Sysnet, Azure를 만나다. [5]
1306정성태6/29/201230065제니퍼 .NET: 22. 눈으로 확인하는 connectionManagement의 maxconnection 설정값 [4]
1305정성태6/28/201228409오류 유형: 157. IIS 6 - WCF svc 호출 시 404 Not Found 발생
1304정성태6/27/201229211개발 환경 구성: 153. sysnet 첨부 파일을 Azure Storage에 마이그레이션 [3]파일 다운로드1
1303정성태6/26/201228789개발 환경 구성: 152. sysnet DB를 SQL Azure 데이터베이스로 마이그레이션
1302정성태6/25/201230725개발 환경 구성: 151. Azure 웹 사이트에 사용자 도메인 네임 연결하는 방법
1301정성태6/20/201227046오류 유형: 156. KB2667402 윈도우 업데이트 실패 및 마이크로소프트 Answers 웹 사이트 대응
1300정성태6/20/201233567.NET Framework: 329. C# - Rabin-Miller 소수 생성방법을 이용하여 RSACryptoServiceProvider의 개인키를 직접 채워보자 [1]파일 다운로드2
1299정성태6/18/201234191제니퍼 .NET: 21. 제니퍼 닷넷 - Ninject DI 프레임워크의 성능 분석 [2]파일 다운로드2
1298정성태6/14/201235831VS.NET IDE: 72. Visual Studio에서 pfx 파일로 서명한 경우, 암호는 어디에 저장될까? [2]
1297정성태6/12/201232331VC++: 63. 다른 프로세스에 환경 변수 설정하는 방법파일 다운로드1
1296정성태6/5/201228877.NET Framework: 328. 해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법 - 두 번째 이야기 [4]파일 다운로드1
1295정성태6/5/201226295.NET Framework: 327. RSAParameters와 System.Numerics.BigInteger 이야기파일 다운로드1
1294정성태5/27/201250371.NET Framework: 326. 유니코드와 한글 - 유니코드와 닷넷을 이용한 한글 처리 [7]파일 다운로드2
1293정성태5/24/201230816.NET Framework: 325. System.Drawing.Bitmap 데이터를 Parallel.For로 처리하는 방법 [2]파일 다운로드1
1292정성태5/24/201224534.NET Framework: 324. First-chance exception에 대해 조건에 따라 디버거가 멈추게 할 수는 없을까? [1]파일 다운로드1
1291정성태5/23/201231441VC++: 62. 배열 초기화를 위한 기계어 코드 확인 [2]
1290정성태5/18/201236326.NET Framework: 323. 관리자 권한이 필요한 작업을 COM+에 대행 [7]파일 다운로드1
1289정성태5/17/201240646.NET Framework: 322. regsvcs.exe로 어셈블리 등록 시 시스템 변경 사항 [5]파일 다운로드2
1288정성태5/17/201227635.NET Framework: 321. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (3) - Type Library파일 다운로드1
1287정성태5/17/201230437.NET Framework: 320. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (2) - .NET 4.0 + .NET 2.0 [2]
1286정성태5/17/201239320.NET Framework: 319. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (1) - .NET 2.0 + x86/x64/AnyCPU [5]
1285정성태5/16/201234455.NET Framework: 318. gacutil.exe로 어셈블리 등록 시 시스템 변경 사항파일 다운로드1
1284정성태5/15/201226896오류 유형: 155. Windows Phone 연결 상태에서 DRIVER POWER STATE FAILURE 블루 스크린 뜨는 현상
1283정성태5/12/201234519.NET Framework: 317. C# 관점에서의 Observer 패턴 구현 [1]파일 다운로드1
... [151]  152  153  154  155  156  157  158  159  160  161  162  163  164  165  ...