Microsoft MVP성태의 닷넷 이야기
.NET Framework: 781. C# - OpenCvSharp 사용 시 포인터를 이용한 속도 향상 [링크 복사], [링크+제목 복사],
조회: 12011
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

(시리즈 글이 10개 있습니다.)
.NET Framework: 707. OpenCV 응용 프로그램을 C#으로 구현 - OpenCvSharp
; https://www.sysnet.pe.kr/2/0/11402

.NET Framework: 708. C# - OpenCvSharp을 이용한 동영상(avi, mp4, ...) 처리
; https://www.sysnet.pe.kr/2/0/11403

.NET Framework: 709. C# - OpenCvSharp을 이용한 동영상(avi, mp4, ...) 처리 + Direct2D
; https://www.sysnet.pe.kr/2/0/11404

.NET Framework: 710. C# - OpenCvSharp을 이용한 Webcam 영상 처리 + Direct2D
; https://www.sysnet.pe.kr/2/0/11405

.NET Framework: 711. C# - OpenCvSharp의 Mat 데이터 조작 방법
; https://www.sysnet.pe.kr/2/0/11406

.NET Framework: 723. C# - OpenCvSharp 사용 시 C/C++을 이용한 속도 향상 (for 루프 연산)
; https://www.sysnet.pe.kr/2/0/11422

VC++: 123. 내가 만든 코드보다 OpenCV의 속도가 월등히 빠른 이유
; https://www.sysnet.pe.kr/2/0/11423

.NET Framework: 781. C# - OpenCvSharp 사용 시 포인터를 이용한 속도 향상
; https://www.sysnet.pe.kr/2/0/11567

개발 환경 구성: 447. Visual Studio Code에서 OpenCvSharp 개발 환경 구성
; https://www.sysnet.pe.kr/2/0/11971

Graphics: 38. C# - OpenCvSharp.VideoWriter에 BMP 파일을 1초씩 출력하는 예제
; https://www.sysnet.pe.kr/2/0/12485




C# - OpenCvSharp 사용 시 포인터를 이용한 속도 향상

아래의 글을 쓴 이후,

내가 만든 코드보다 OpenCV의 속도가 월등히 빠른 이유
; https://www.sysnet.pe.kr/2/0/11423

그래도 C#의 성능을 어떻게 좀 높일 수 있을까... 생각하다가 그냥 포인터 연산을 사용해 보기로 했습니다. 지난 글에서도 봤듯이 포인터 연산이라고 해서 무조건 빠른 것은 아닙니다.

C# - System.Span<T> 성능
; https://www.sysnet.pe.kr/2/0/11535

위의 결과에도 나오지만 단순 배열의 경우 포인터 연산과 비교해도 속도에서 결코 뒤지지 않습니다. 단지, OpenCvSharp의 경우에는 개별 요소를 제네릭 메서드로 경유하는 것이기 때문에 포인터를 직접 사용하는 것이 더 나을 수 있겠다는 생각이었습니다.




변환 코드는 다음의 제네릭 버전(At, Set)인 C# 메서드를,

static int Convert2(Mat srcMat, Mat kernel, Window window)
{
    int iMin, iVal;

    using (Mat dstMat = srcMat.Clone())
    {
        for (int i = 0; i < srcMat.Rows - 2; i++)
        {
            for (int j = 0; j < srcMat.Cols - 2; j++)
            {
                iMin = 0xFFF;

                for (int ii = 0; ii < kernel.Rows; ii++)
                {
                    for (int jj = 0; jj < kernel.Cols; jj++)
                    {
                        if (kernel.At<byte>(ii, jj) != 0)
                        {
                            iVal = srcMat.At<byte>(i + ii, j + jj);
                            if (iMin > iVal)
                            {
                                iMin = iVal;
                            }
                        }
                    }
                }

                dstMat.Set<byte>(i + 1, j + 1, (byte)iMin);
            }
        }
    }

    return 0;
}

포인터 연산으로 바꾼 것입니다.

static unsafe int Convert3(Mat srcMat, Mat kernel, Window window)
{
    int iMin, iVal;

    byte* kernelPtr = kernel.DataPointer;
    long kernelStep = kernel.Step();
    int kernelElemSize = kernel.ElemSize();

    byte* srcPtr = srcMat.DataPointer;
    long srcStep = srcMat.Step();
    int srcElemSize = srcMat.ElemSize();

    using (Mat dstMat = srcMat.Clone())
    {
        byte *dstPtr = dstMat.DataPointer;
        long dstStep = dstMat.Step();
        int dstElemSize = dstMat.ElemSize();

        for (int i = 0; i < srcMat.Rows - 2; i++)
        {
            for (int j = 0; j < srcMat.Cols - 2; j++)
            {
                iMin = 0xFFF;

                for (int ii = 0; ii < kernel.Rows; ii++)
                {
                    for (int jj = 0; jj < kernel.Cols; jj++)
                    {
                        if (*(kernelPtr + (ii) * kernelStep + (jj) * kernelElemSize) != 0)
                        {
                            iVal = *(srcPtr + (i + ii) * srcStep + (j + jj) * srcElemSize);
                            if (iMin > iVal)
                            {
                                iMin = iVal;
                            }
                        }
                    }
                }

                *(dstPtr + ((i + 1) * dstStep + (j + 1) * dstElemSize)) = (byte)iMin;
            }
        }

        if (window != null)
        {
            window.ShowImage(dstMat);
        }
    }

    return 0;
}

테스트 결과는, OpenCvSharp의 제네릭 메서드 버전이 얼마나 낮은 성능을 보이는지 극명하게 나타내고 있습니다.

[CPU i5-4670 4-core]

OpenCvSharp 제네릭 At, Set :  26,550ms
C# unsafe ptr              :   1,341ms
C# unsafe ptr parallel     :     285ms
C++                        :      51ms

엄청난 차이입니다. ^^; 물론, 그래도 C++의 51ms에 비하면 많이 느리지만 26초 걸리던 것을 1초 정도로 줄였으니 현실적으로 봤을 때 가벼운 목적으로 제작하는 프로그램이라면 C++의 힘을 빌리지 않아도 될 수준까지는 내려갔습니다.

따라서 OpenCvSharp에서 제네릭 메서드 버전의 사용은 지양하는 것이 좋습니다.

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




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







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

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

비밀번호

댓글 작성자
 




... 61  62  63  [64]  65  66  67  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12042정성태10/26/201911135오류 유형: 573. OneDrive 하위에 위치한 Documents, Desktop 폴더에 대한 권한 변경 시 "Unable to display current owner"
12041정성태10/23/201911144오류 유형: 572. mstest.exe - The load test results database could not be opened.
12040정성태10/23/201911397오류 유형: 571. Unhandled Exception: System.Net.Mail.SmtpException: Transaction failed. The server response was: 5.2.0 STOREDRV.Submission.Exception:SendAsDeniedException.MapiExceptionSendAsDenied
12039정성태10/22/20199813스크립트: 16. cmd.exe의 for 문에서는 ERRORLEVEL이 설정되지 않는 문제
12038정성태10/17/20199371오류 유형: 570. SQL Server 2019 RC1 - SQL Client Connectivity SDK 설치 오류
12037정성태10/15/201915309.NET Framework: 867. C# - Encoding.Default 값을 바꿀 수 있을까요?파일 다운로드1
12036정성태10/14/201916335.NET Framework: 866. C# - 고성능이 필요한 환경에서 GC가 발생하지 않는 네이티브 힙 사용파일 다운로드1
12035정성태10/13/201912478개발 환경 구성: 461. C# 8.0의 #nulable 관련 특성을 .NET Framework 프로젝트에서 사용하는 방법 [2]파일 다운로드1
12034정성태10/12/201911824개발 환경 구성: 460. .NET Core 환경에서 (프로젝트가 아닌) C# 코드 파일을 입력으로 컴파일하는 방법 [1]
12033정성태10/11/201915513개발 환경 구성: 459. .NET Framework 프로젝트에서 C# 8.0/9.0 컴파일러를 사용하는 방법
12032정성태10/8/201911986.NET Framework: 865. .NET Core 2.2/3.0 웹 프로젝트를 IIS에서 호스팅(Inproc, out-of-proc)하는 방법 - AspNetCoreModuleV2 소개
12031정성태10/7/20199409오류 유형: 569. Azure Site Extension 업그레이드 시 "System.IO.IOException: There is not enough space on the disk" 예외 발생
12030정성태10/5/201915677.NET Framework: 864. .NET Conf 2019 Korea - "닷넷 17년의 변화 정리 및 닷넷 코어 3.0" 발표 자료 [1]파일 다운로드1
12029정성태9/27/201915781제니퍼 .NET: 29. Jennifersoft provides a trial promotion on its APM solution such as JENNIFER, PHP, and .NET in 2019 and shares the examples of their application.
12028정성태9/26/201911577.NET Framework: 863. C# - Thread.Suspend 호출 시 응용 프로그램 hang 현상을 해결하기 위한 시도파일 다운로드1
12027정성태9/26/20198845오류 유형: 568. Consider app.config remapping of assembly "..." from Version "..." [...] to Version "..." [...] to solve conflict and get rid of warning.
12026정성태9/26/201912516.NET Framework: 862. C# - Active Directory의 LDAP 경로 및 정보 조회
12025정성태9/25/201910865제니퍼 .NET: 28. APM 솔루션 제니퍼, PHP, .NET 무료 사용 프로모션 2019 및 적용 사례 (8) [1]
12024정성태9/20/201912301.NET Framework: 861. HttpClient와 HttpClientHandler의 관계 [2]
12023정성태9/18/201912720.NET Framework: 860. ServicePointManager.DefaultConnectionLimit와 HttpClient의 관계파일 다운로드1
12022정성태9/12/201915769개발 환경 구성: 458. C# 8.0 (Preview) 신규 문법을 위한 개발 환경 구성 [3]
12021정성태9/12/201927658도서: 시작하세요! C# 8.0 프로그래밍 [4]
12020정성태9/11/201914493VC++: 134. SYSTEMTIME 값 기준으로 특정 시간이 지났는지를 판단하는 함수
12019정성태9/11/20199842Linux: 23. .NET Core + 리눅스 환경에서 Environment.CurrentDirectory 접근 시 주의 사항
12018정성태9/11/20198841오류 유형: 567. IIS - Unrecognized attribute 'targetFramework'. Note that attribute names are case-sensitive. (D:\lowSite4\web.config line 11)
12017정성태9/11/201911877오류 유형: 566. 비주얼 스튜디오 - Failed to register URL "http://localhost:6879/" for site "..." application "/". Error description: Access is denied. (0x80070005)
... 61  62  63  [64]  65  66  67  68  69  70  71  72  73  74  75  ...