Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

C# - ImageCodecInfo 사용 시 System.Drawing.Image와 System.Drawing.Bitmap에 따른 Save 성능 차이

Bitmap을 이용해 JPG 파일로 저장하는 코드를 좀 더 범용적으로 사용하려고 Image로 바꿨더니 성능이 낮아졌습니다. 이상하군요, 정말 그런지 다음과 같은 코드로 테스트를 할 수 있습니다.

using System.Diagnostics;
using System.Drawing;
using System.Drawing.Imaging;

internal class Program
{
    static void Main(string[] args)
    {
        Image img = Bitmap.FromFile("C:\\temp\\SingAgain_No_48.png");
        Bitmap bmp = new Bitmap(img);

        ImageCodecInfo? jpgEncoder = GetEncoder(ImageFormat.Jpeg);

        System.Drawing.Imaging.Encoder myEncoder = System.Drawing.Imaging.Encoder.Quality;
        EncoderParameters myEncoderParameters = new EncoderParameters(1);
        EncoderParameter myEncoderParameter = new EncoderParameter(myEncoder, 80L);
        myEncoderParameters.Param[0] = myEncoderParameter;

        Action<int, string, Action<object, ImageCodecInfo?, EncoderParameters>, object, ImageCodecInfo?, EncoderParameters> action =
            (loopCount, title, work, arg0, arg1, arg2) =>
        {

            for (int i = 0; i < loopCount; i++)
            {
                Stopwatch st = new Stopwatch();
                st.Start();
                work(arg0, arg1, arg2);
                st.Stop();

                if (loopCount != 1)
                {
                    Console.WriteLine(title + " : " + st.ElapsedMilliseconds);
                }
            }
        };

        action(1, "Bitmap", UseBitmap, bmp, jpgEncoder, myEncoderParameters);
        action(5, "Bitmap", UseBitmap, bmp, jpgEncoder, myEncoderParameters);

        action(1, "Image", UseImage, img, jpgEncoder, myEncoderParameters);
        action(5, "Image", UseImage, img, jpgEncoder, myEncoderParameters);
    }

    static void UseBitmap(object bitmapObj, ImageCodecInfo? encoder, EncoderParameters encoderParameter)
    {
        Bitmap? bitmap = bitmapObj as Bitmap;
        MemoryStream jpgStream = new MemoryStream();
        bitmap?.Save(jpgStream, encoder, encoderParameter);
    }

    static void UseImage(object imgObj, ImageCodecInfo? encoder, EncoderParameters encoderParameter)
    {
        Image? img = imgObj as Image;
        MemoryStream jpgStream = new MemoryStream();
        img?.Save(jpgStream, encoder, encoderParameter);
    }

    private static ImageCodecInfo? GetEncoder(ImageFormat format)
    {
        ImageCodecInfo[] codecs = ImageCodecInfo.GetImageEncoders();
        foreach (ImageCodecInfo codec in codecs)
        {
            if (codec.FormatID == format.Guid)
            {
                return codec;
            }
        }

        return null;
    }
}

출력 결과를 보면,

// .NET Framework 4.8 + x64 Release + 이미지 1922 x 1082
Bitmap : 12
Bitmap : 12
Bitmap : 12
Bitmap : 12
Bitmap : 12
Image : 36
Image : 35
Image : 34
Image : 34
Image : 34

// .NET 7 + x64 Release + 이미지 1922 x 1082
Bitmap : 10
Bitmap : 11
Bitmap : 10
Bitmap : 10
Bitmap : 10
Image : 44
Image : 43
Image : 42
Image : 42
Image : 42

Bitmap으로 변환했을 때 3배 이상 빠릅니다. 재미있는 것은, Image를 사용하는 코드를 Bitmap으로 다시 변환하도록 작업만 해도,

static void UseImage(object imgObj, ImageCodecInfo? encoder, EncoderParameters encoderParameter)
{
    Image? img = imgObj as Image;
    Bitmap? bitmap = new Bitmap(img);
    MemoryStream jpgStream = new MemoryStream();
    bitmap?.Save(jpgStream, encoder, encoderParameter);
}

속도가 빨라집니다.

// .NET Framework 4.8 + x64 Release + 이미지 1922 x 1082
Bitmap : 22
Bitmap : 23
Bitmap : 22
Bitmap : 23
Bitmap : 23
Image : 35
Image : 35
Image : 34
Image : 34
Image : 35

// .NET 7 + x64 Release + 이미지 1922 x 1082
Bitmap : 20
Bitmap : 20
Bitmap : 19
Bitmap : 19
Bitmap : 19
Image : 42
Image : 42
Image : 42
Image : 44
Image : 42

즉, 단순히 Image에서 Bitmap으로 변환하는 데 들어가는 비용이 전부가 아니라는 것입니다. 어쨌든, 위의 경우를 따졌을 때 Image를 그냥 사용하기보다는 Bitmap으로 바꾼 후 포맷 처리를 하는 것이 성능상 도움이 됩니다.

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




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







[최초 등록일: ]
[최종 수정일: 6/22/2023]

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

비밀번호

댓글 작성자
 




... 136  137  138  139  140  [141]  142  143  144  145  146  147  148  149  150  ...
NoWriterDateCnt.TitleFile(s)
1563정성태1/3/201429634오류 유형: 201. ASP.NET 웹 사이트를 IIS 7 이상의 환경에서 호스팅할 때 500 오류 발생
1562정성태1/2/201434309.NET Framework: 402. 카카오톡 PC 버전 실행 시 개발자 컴퓨터에서 Themida 오류 나는 문제 - 두 번째 이야기
1561정성태1/1/201443998오류 유형: 200. 카카오톡 PC 버전 실행 시 개발자 컴퓨터에서 Themida 오류 나는 문제 [2]
1560정성태1/1/201422509오류 유형: 199. Hyper-V - Checkpoint 생성 오류
1559정성태12/30/201325864.NET Framework: 401. windbg에서 확인해 보는 관리 힙의 인스턴스 구조 [2]
1558정성태12/29/201335493.NET Framework: 400. 눈으로 확인하는 LayoutKind 옵션 효과 [2]파일 다운로드1
1557정성태12/27/201324413.NET Framework: 399. LayoutKind 옵션에 대해 [2]
1556정성태12/27/201344679개발 환경 구성: 208. IIS + Node.js 환경 구성 [8]
1555정성태12/27/201331181개발 환경 구성: 207. IIS의 80 포트를 공유하는 응용 프로그램 만드는 방법 [1]
1554정성태12/26/201336551Windows: 78. 마음에 드는 윈도우 8.1 태블릿 - 델 베뉴 8 프로 5830 [4]
1553정성태12/26/201323615개발 환경 구성: 206. JNBridgePro와 한글 인코딩 문제파일 다운로드1
1552정성태12/25/201328849개발 환경 구성: 205. JNBridgePro를 이용해 C#에서 Java메서드 호출 테스트파일 다운로드1
1551정성태12/24/201324003.NET Framework: 398. tech-days 미니 토요세미나 - 3회 C#편 PPT 자료파일 다운로드1
1550정성태12/13/201326405Windows: 77. Windows 8 - 잠시 사용을 안하는 경우 화면 잠김 상태로 빠지는 문제
1549정성태12/13/201329967VC++: 73. IIS - ISAPI 필터 제작하는 방법 [2]
1548정성태12/10/201322526오류 유형: 198. C# - 제네릭 covariance/contravariance 사용할 때 컴파일 오류가 발생한다면?
1547정성태12/10/201332806.NET Framework: 397. C# - OCX 컨트롤에 구현된 메서드에 배열을 in, out으로 전달하는 방법파일 다운로드2
1546정성태11/28/201326147.NET Framework: 396. C# - 프로퍼티로 정의하면 필드보다 느릴까요? - windbg / ollydbg [3]
1545정성태11/28/201329974.NET Framework: 395. C# - 프로퍼티로 정의하면 필드보다 느릴까요? [3]
1544정성태11/27/201326571개발 환경 구성: 204. Visual Studio Online "Monaco" 서비스와 github 연동
1543정성태11/27/201331250오류 유형: 197. error MSB8008: Specified platform toolset (v120) is not installed or invalid. [1]
1542정성태11/27/201336922오류 유형: 196. The procedure entry point InitializeCriticalSectionEx could not be located in the dynamic link library KERNEL32.dll
1541정성태11/22/201338197.NET Framework: 394. async/await 사용 시 hang 문제가 발생하는 경우 [7]파일 다운로드1
1540정성태11/20/201326486개발 환경 구성: 203. Azure - WEB SITES 서비스 소개 [4]
1539정성태11/19/201330454VS.NET IDE: 83. 형상 관리 서버 운영을 대신해 주는 Visual Studio 온라인 서비스
1538정성태11/19/201331324오류 유형: 195. 웹 사이트의 모든 정적 컨텐츠 요청에 대해 "Internal Server Error" 응답
... 136  137  138  139  140  [141]  142  143  144  145  146  147  148  149  150  ...