Microsoft MVP성태의 닷넷 이야기
VC++: 148. Golang - 채널에 따른 다중 작업 처리 [링크 복사], [링크+제목 복사],
조회: 12720
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

Golang - 채널에 따른 다중 작업 처리

Go 언어로 프로그램을 하다 보면, 자칫 goroutine을 남발해 어느 순간이 되면 프로그램의 실행 경로가 어떻게 되는지 감이 안 오는 경우가 있습니다. 물론, 비동기 처리를 훌륭하게 그려낼 수 있는 뇌구조를 가진 개발자라면 상관없겠지만, 저처럼 ^^ 일반인은 역시 동기 처리가 편할 수밖에 없습니다.

그래서 최대한 실행 경로를 구조적으로 유지하는 것이 중요한데요, 가령 channel에 따른 실행 처리를 할 때 각각의 경우 별로 goroutine을 사용할 수도 있겠지만,

package main

import (
    "bufio"
    "fmt"
    "os"
    "time"
)

func main() {
    go WorkPer1Sec()
    go WorkPer5Sec()

    // 프로세스 종료를 막기 위해.
    bufio.NewReader(os.Stdin).ReadLine()
}

func WorkPer5Sec() {

    // 5초마다 수행할 작업
    for {
        <-time.After(5 * time.Second)
        fmt.Println("5sec elapsed")
    }
}

func WorkPer1Sec() {

    // 1초마다 수행할 작업
    for {
        <-time.After(1 * time.Second)
        fmt.Println("1sec elapsed")
    }
}

select를 이용해 한데 모으는 것도 가능합니다.

func main() {

    go WorkPer()

    bufio.NewReader(os.Stdin).ReadLine()
}

func WorkPer() {

    time5 := time.Tick(5 * time.Second)
    time1 := time.Tick(1 * time.Second)

    for {

        select {
        case <-time5:
            fmt.Print("5, ")
        case <-time1:
            fmt.Print("1, ")
        }
    }
}

/* 출력 결과
1, 1, 1, 1, 5, 1, 1, 1, 1, 1, 5, 1, 1, 1, 1, 1, 5, 1, 1, 1, 1, 1, 
*/

대신 이런 경우에는 채널이 하나의 goroutine 내에서 처리되기 때문에 동일한 CPU 스레드에서 동작한다는 특징이 있습니다. 이로 인해 분리된 채널에서의 공유 변수에 대한 동기화는 필요 없지만 스레드 점유에 따른 간섭은 발생할 수 있습니다. 가령, 다음과 같이 5초 채널에서 (I/O 발생이 아닌) CPU 소비가 필요한 동작을 하게 되면,

func WorkPer() {

    time5 := time.Tick(5 * time.Second)
    time1 := time.Tick(1 * time.Second)

    for {

        select {
        case <-time5:
            fmt.Print("5, ")

            now := time.Now()
            for {
                var elapsed = time.Since(now)
                if elapsed.Seconds() > 5 { // 일부러 5초가 걸리는 CPU 작업을 수행
                    break
                }
            }
        case <-time1:
            fmt.Print("1, ")
        }

    }
}

화면에는 '1' 채널의 작업에 영향을 줘 다음과 같은 식의 예측할 수 없는 출력이 발생합니다.

...[생략]..., 5, 1, 1, 5, 1, 1, 5, 5, 1, 1, 5, 5, 1, 1, 5

만약 저런 상황이라면, 각각의 channel 작업을 분리해 goroutine으로 실행하시면 됩니다.

func WorkPer() {

    time5 := time.Tick(5 * time.Second)
    time1 := time.Tick(1 * time.Second)

    for {

        select {
        case <-time5:
            go Work5()

        case <-time1:
            go Work1()
        }

    }
}

func Work1() {
    fmt.Print("1, ")
}

func Work5() {
    fmt.Print("5, ")

    now := time.Now()
    for {
        var elapsed = time.Since(now)
        if elapsed.Seconds() > 5 {
            break
        }
    }
}

/* 출력 결과
1, 1, 1, 1, 5, 1, 1, 1, 1, 1, 5, 1, 1, 1, 1, 1, 5, 1, 1, 1, 1, 1, 
*/

어찌 보면 처음부터 goroutine으로 작업한 것과 별반 다를 바가 없지만, 뭔가 비동기적으로 발생하는 이벤트를 한 곳에서 확인하는 것이 관리적인 차원에서 더 나을 수 있습니다.




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







[최초 등록일: ]
[최종 수정일: 9/9/2021]

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)
13693정성태7/24/20247249개발 환경 구성: 717. Visual Studio - C# 프로젝트에서 레지스트리에 등록하지 않은 COM 개체 참조 및 사용 방법파일 다운로드1
13692정성태7/24/20248043디버깅 기술: 199. Windbg - 리눅스에서 뜬 닷넷 응용 프로그램 덤프 파일에 포함된 DLL의 Export Directory 탐색
13691정성태7/23/20247401디버깅 기술: 198. Windbg - 스레드의 Win32 Message Queue 정보 조회
13690정성태7/23/20247020오류 유형: 919. Visual C++ 리눅스 프로젝트 - error : ‘u8’ was not declared in this scope
13689정성태7/22/20248554디버깅 기술: 197. Windbg - PE 포맷의 Export Directory 탐색
13688정성태7/21/20247641닷넷: 2281. C# - Lock / Wait 상태에서도 일부 Win32 메시지 처리파일 다운로드1
13687정성태7/19/20248065닷넷: 2280. C# - PostThreadMessage로 보낸 메시지를 Windows Forms에서 수신하는 방법파일 다운로드1
13686정성태7/19/20247856오류 유형: 918. Visual Studio - ATL Simple Object 추가 시 error C2065: 'IDR_...': undeclared identifier
13685정성태7/19/20248017스크립트: 66. Windows 디렉터리 경로를 WSL의 /mnt 포맷으로 구하는 방법 - 두 번째 이야기
13684정성태7/19/20248191닷넷: 2279. C# - 문자열 보간식 사례 (예: 조건 연산자 사용)
13683정성태7/18/20247669오류 유형: 917. ClrMD - Linux 환경의 .NET 5 덤프 분석 시 hang 현상
13682정성태7/18/20247871닷넷: 2278. WPF - 스레드에 종속되는 DependencyObject파일 다운로드1
13681정성태7/17/20247485닷넷: 2277. C# 13 - (2) 메서드 그룹의 자연 타입 개선 (메서드 추론 개선)파일 다운로드1
13680정성태7/16/20247865닷넷: 2276. C# - Method Group, Natural Type, function_type파일 다운로드1
13679정성태7/16/20246947Linux: 76. Linux - C++ (getaddrinfo 등을 담고 있는) libnss 정적 링크
13678정성태7/15/20247086VS.NET IDE: 191. Visual Studio 2022 - .NET 5 프로젝트를 Docker Support로 실행했을 때 오류
13677정성태7/15/20247175오류 유형: 916. MSBuild - CheckEolTargetFramework (warning NETSDK1138)
13676정성태7/14/20247346Linux: 75. gdb에서 glibc의 함수에 Breakpoint 걸기
13675정성태7/13/20249137C/C++: 166. C/C++ - DLL에서 template 함수를 export하는 방법 [1]파일 다운로드1
13674정성태7/13/20248010오류 유형: 915. Unhandled Exception: Microsoft.Diagnostics.NETCore.Client.ServerNotAvailableException: Unable to connect to Process
13673정성태7/11/20248461닷넷: 2275. C# 13 - (1) 신규 이스케이프 시퀀스 '\e'파일 다운로드1
13672정성태7/10/20247170닷넷: 2274. IIS - (프로세스 종료 없는) AppDomain Recycle
13671정성태7/10/20247270오류 유형: 914. Package ca-certificates is not installed.
13669정성태7/9/20247388오류 유형: 913. C# - AOT StaticExecutable 정적 링킹 시 빌드 오류
13668정성태7/8/20247419개발 환경 구성: 716. Hyper-V - Ubuntu 22.04 Generation 2 유형의 VM 설치
13667정성태7/7/20246625닷넷: 2273. C# - 리눅스 환경에서의 Hyper-V Socket 연동 (AF_VSOCK)파일 다운로드1
1  2  3  4  5  6  7  8  9  [10]  11  12  13  14  15  ...