Microsoft MVP성태의 닷넷 이야기
.NET Framework: 288. FFmpeg.exe를 이용한 C# 동영상 인코더 예제 [링크 복사], [링크+제목 복사],
조회: 41014
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)
(시리즈 글이 7개 있습니다.)
.NET Framework: 288. FFmpeg.exe를 이용한 C# 동영상 인코더 예제
; https://www.sysnet.pe.kr/2/0/1210

개발 환경 구성: 622. vcpkg로 ffmpeg를 빌드하는 경우 생성될 구성 요소 제어하는 방법
; https://www.sysnet.pe.kr/2/0/12900

개발 환경 구성: 623. ffmpeg.exe를 사용해 비디오 파일의 이미지를 PGM(Portable Gray Map) 파일 포맷으로 출력하는 방법
; https://www.sysnet.pe.kr/2/0/12912

개발 환경 구성: 626. ffmpeg.exe를 사용해 비디오 파일을 MPEG1 포맷으로 변경하는 방법
; https://www.sysnet.pe.kr/2/0/12923

개발 환경 구성: 634. ffmpeg.exe - 기존 동영상 컨테이너에 다중 스트림을 추가하는 방법
; https://www.sysnet.pe.kr/2/0/12947

개발 환경 구성: 636. ffmpeg.exe를 이용해 planar 포맷의 데이터를 packed 형식으로 변환하는 방법?
; https://www.sysnet.pe.kr/2/0/12957

개발 환경 구성: 639. ffmpeg.exe - Intel Quick Sync Video(qsv)를 이용한 인코딩
; https://www.sysnet.pe.kr/2/0/12973




FFmpeg.exe를 이용한 C# 동영상 인코더 예제

참고로, 이번 내용은 다음의 글을 읽고 작성한 것입니다.

Creating a Video Converter Using VB.NET
; http://www.codeguru.com/vb/gen/vb_multimedia/article.php/c19683

마침 질문 주신 분도 있고 해서,

FFmpeg 사용해 보신 분 좀 도와주세요
; http://mblog.devpia.com/link/?no=2124351

VB.NET으로 된 예제를 C#으로 작성해 보았습니다. 그런데, 음... 구 버전의 ffmpeg.exe를 써서 그런지 저대로 코딩하거나 글에 포함된 소스코드를 다운로드해도 정상적으로 동작하지 않습니다.

그래서, 제 경우에는 위의 것을 그대로 따라하지 않고 최대한 단순화 시키면서 '동작 가능한 버전'으로 만들어 보았습니다.




우선, ffmpeg.exe 파일을 구해야 하는데 다음의 사이트에서 다운로드 할 수 있습니다.

FFmpeg
; http://ffmpeg.org/

소스 코드를 직접 빌드해도 되지만, FFmpeg 윈도우 빌드를 다음의 경로에서 다운로드하는 것도 가능합니다.

FFmpeg Windows Builds
; http://ffmpeg.zeranoe.com/builds/ 

이 글에서는 위의 사이트에서 다운로드한 32/64-bit builds(static) 버전으로 진행하겠습니다.

FFmpeg git-dd1fb65 64-bit Static (Latest)(2011-12-22)
; http://ffmpeg.zeranoe.com/builds/win64/static/ffmpeg-git-dd1fb65-win64-static.7z

FFmpeg git-dd1fb65 32-bit Static (Latest)(2011-12-22)
; http://ffmpeg.zeranoe.com/builds/win32/static/ffmpeg-git-dd1fb65-win32-static.7z

위의 ffmpeg.exe를 구했으면 이제 나머지는 아주 초보적인 코딩에 불과합니다. 왜냐하면, ffmpeg.exe를 외부 프로세스로 호출하고 변환할 것이므로 단순히 명령행 인자만 적절하게 구성해서 전달해 주면 되기 때문입니다.

우선 명령행에서 간단하게 테스트해 보는 것이 좋은데요.

대충 예제를 맞추기 위해서 다음과 같은 옵션들을 선택할 수 있었습니다.

-b:v video bitrate
64k, 96k, 128k, 192k, 256k, 320k, 340k

-s frame size
wvga, ... hd420,...

-ar audio sampling frequency
11025
22050
44100

-qscale Use fixed quality scale (VBR). The meaning of q is codec-dependent.
To have a constant quality (but a variable bitrate), use the option ’-qscale n’ when ’n’ is between 1 (excellent quality) and 31 (worst quality).

이 외의 좀 더 다양한 옵션에 대해서는 다음의 링크를 참고하세요.

ffmpeg Documentation
; http://ffmpeg.org/ffmpeg.html


그래서, 플래시 파일을 MP4로 변환하는 것은 다음과 같이 해줄 수 있습니다.

ffmpeg.exe -i "D:\temp\sample.flv" -b:v 340k -y "D:\temp\sample.mp4" -s hd480 -ar 44100

C#에서는 Process.Start로 실행시켜 주기만 하면 끝이겠지요. ^^

ProcessStartInfo psiProcInfo = new ProcessStartInfo();

StreamReader srFFMPEG;
string strFFMPEGCmd = " -i \"" + srcFile + "\" -ar 44100 " + videoRateOption + videoSizeOption + " -y \"" + dstFile + "\"";

psiProcInfo.FileName = Application.StartupPath + ((IntPtr.Size == 8) ? "\\x64" : "\\x86") + "\\ffmpeg.exe";
psiProcInfo.Arguments = strFFMPEGCmd;
psiProcInfo.UseShellExecute = false;
psiProcInfo.WindowStyle = ProcessWindowStyle.Hidden;
psiProcInfo.RedirectStandardError = true;
psiProcInfo.RedirectStandardOutput = true;

prcFFMPEG.StartInfo = psiProcInfo;

prcFFMPEG.Start();

그런데... 혹시 '진행율'을 표시하는 것이 가능할까요? 일단, 외부 프로세스를 실행하는 것이라서 별도의 이벤트가 제공되지 않는 한 불가능합니다. 이를 위해서는 FFmpeg 라이브러리 수준으로 사용해서 직접 API를 제어하는 것으로 가능할 텐데요.

그래도, 일단 아쉬운 대로 FFmpeg.exe의 출력 결과를 보면 어느 정도 예측하는 것은 가능합니다. 예를 들어 아래의 실행 결과를 보면,

D:\temp>ffmpeg.exe  -i "D:\temp\videoplayback.flv" -b:v 340k -y "D:\temp\videoplayback.mp4"  -s hd480 -ar 44100 -pass 2
...[생략]...
Input #0, flv, from 'D:\temp\videoplayback.flv':
  Metadata:
    starttime       : 0
    totalduration   : 238
    totaldatarate   : 478
    bytelength      : 14186363
    canseekontime   : true
    sourcedata      : B4A7DD5C6MH1324909622848868
    purl            :
    pmsg            :
  Duration: 00:03:57.50, start: 0.000000, bitrate: 477 kb/s
    Stream #0:0: Video: h264 (Main), yuv420p, 640x360, 348 kb/s, 29.97 tbr, 1k tbn, 59.94 tbc
    Stream #0:1: Audio: aac, 44100 Hz, stereo, s16, 131 kb/s
...[생략]...
Press [q] to stop, [?] for help
frame= 7119 fps=166 q=-1.0 Lsize=   13720kB time=00:03:57.47 bitrate= 473.3kbits/s
video:9777kB audio:3711kB global headers:0kB muxing overhead 1.722711%
...[생략]...

Input #0 FLV 파일의 Duration을 03:57.50으로 출력해 주고 있기 때문에 이 값을 보관하고 있다가 이 후에 인코딩 시에 출력해 주는 time=... 값을 비교해서 백분율을 내면 진행 상태를 출력해 줄 수 있는 것입니다.

첨부된 파일은 위의 코드를 포함한 예제 프로젝트이고, 클릭원스로도 올려놓았으니 원하시는 분들은 간단하게 테스트하실 수 있을 것입니다. (참고로, 입력 파일은 플래시 확장자(.flv)로 제한했습니다.)

FLV 동영상 변환
; https://www.sysnet.pe.kr/temp/app/vencoder/VEncoder.application

64비트 PC에서는 64비트 버전의 ffmpeg.exe를 사용하고, 32비트 PC에서는 32비트 버전의 ffmpeg.exe를 사용해서 인코딩합니다.

ffmpeg_cs_1.png

그 유명한 "다음 팟 인코더"도 결국 FFmpeg 라이브러리를 이용한 것이기 때문에, 위의 응용 프로그램과 비교해서 동영상 인코딩 품질은 동일합니다. ^^






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

[연관 글]






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

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

비밀번호

댓글 작성자
 



2011-12-30 10시47분
[남산골] 친절한 답변 감사합니다. 저 위 싸이트들 들락날락하면서 1주일 이상을 머리싸매고 봤지만 솔직히 아직도 제대로 이해못합니다 ㅠㅠ. 저런걸 금방보고 코드까지 작성할 수 있다니 정말 고수시네요. 어찌어찌해서 인코더는 만들었는데 저 것보고 더 연구해 봐야겠네요. 나중에 또 궁금한 것 있으면 찾아오겠습니다. 올해도 하루 밖에 안남았네요. 좋은 연말 보내시고 새해 복 많이 받으세요(^.^)
[guest]
2012-01-01 09시18분
[네모] 키라라 인코더 가 VB.NET 입니다.. 소스포지에 가보면 소스도 있으니 이걸 개조하던지 참고해보세요....
[guest]
2015-01-22 11시45분
[개발자] 우와 잘정리되어 있네요 많은 도움이 되었습니다 감사 합니다.
[guest]
2015-04-23 07시56분
[하수] 안녕하세요 질문있습니다. ogv파일으로 변환할 수 있도록 추가하고싶은데 어디에다가 추가하는지 잘 모르겠습니다. 가르쳐주세요
[guest]
2015-04-23 03시07분
글쎄요. 저도 웹상에 공개된 방법 이외에는 잘 모릅니다. 아래의 글에 보니, ogv를 대상으로 하는 예제가 실려 있으니 참고하세요.

http://paulrouget.com/e/converttohtml5video/

참고 삼아 여기에도 적어둡니다. ^^

* OGG/Theora
ffmpeg -i input.mov -acodec libvorbis -ac 2 -ab 96k -ar 44100 -b 345k -s 640x360 output.ogv
정성태
2015-08-21 07시44분
동영상의 기본적인 이해(컨테이너 포맷이란?, 동영상의 변환이란?)
; http://blog.naver.com/brian423/174055572
정성태
2016-06-07 06시48분
[초보자] 안녕하세요 게시물 잘봤습니다!
혹시 png나 jpg같은 이미지 파일들을 동영상으로 제작하려면 어떤식으로 구현을 해야할까요?... 가르쳐주세요
[guest]
2016-06-07 07시09분
아 연속된 이미지를 영상으로 만들고 싶습니다 답변부탁드립니다!
박정훈
2016-06-07 07시32분
@박정훈 아래의 사용법을 보세요.

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

Create a video slideshow from images
; https://trac.ffmpeg.org/wiki/Create%20a%20video%20slideshow%20from%20images

----------------------------------------

How to build FFmpeg 7 from source and what you can do with its new features
; https://www.codeproject.com/Tips/5380437/How-to-build-FFmpeg-7-from-source-and-what-you-can
정성태

... 31  32  33  34  35  36  37  38  39  40  [41]  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
12623정성태5/2/20219979.NET Framework: 1054. C# 9 최상위 문에 STAThread 사용 [1]파일 다운로드1
12622정성태5/2/20216726오류 유형: 713. XSD 파일을 포함한 프로젝트 - The type or namespace name 'TypedTableBase<>' does not exist in the namespace 'System.Data'
12621정성태5/1/202110488.NET Framework: 1053. C# - 특정 레지스트리 변경 시 알림을 받는 방법 [1]파일 다운로드1
12620정성태4/29/202112432.NET Framework: 1052. C# - 왜 구조체는 16 바이트의 크기가 적합한가? [1]파일 다운로드1
12619정성태4/28/202112935.NET Framework: 1051. C# - 구조체의 크기가 16바이트가 넘어가면 힙에 할당된다? [2]파일 다운로드1
12618정성태4/27/202111389사물인터넷: 58. NodeMCU v1 ESP8266 CP2102 Module을 이용한 WiFi UDP 통신 [1]파일 다운로드1
12617정성태4/26/20219217.NET Framework: 1050. C# - ETW EventListener의 Keywords별 EventId에 따른 필터링 방법파일 다운로드1
12616정성태4/26/20219056.NET Framework: 1049. C# - ETW EventListener를 상속받았을 때 초기화 순서파일 다운로드1
12615정성태4/26/20217089오류 유형: 712. Microsoft Live 로그인 - 계정을 선택하는(Pick an account) 화면에서 진행이 안 되는 문제
12614정성태4/24/20219933개발 환경 구성: 570. C# - Azure AD 인증을 지원하는 ASP.NET Core/5+ 웹 애플리케이션 예제 구성 [4]파일 다운로드1
12613정성태4/23/20218994.NET Framework: 1048. C# - ETW 이벤트의 Keywords에 속한 EventId 구하는 방법 (2) 관리 코드파일 다운로드1
12612정성태4/23/20219123.NET Framework: 1047. C# - ETW 이벤트의 Keywords에 속한 EventId 구하는 방법 (1) PInvoke파일 다운로드1
12611정성태4/22/20218381오류 유형: 711. 닷넷 EXE 실행 오류 - Mixed mode assembly is build against version 'v2.0.50727' of the runtime
12610정성태4/22/20218276.NET Framework: 1046. C# - 컴파일 시점에 참조할 수 없는 타입을 포함한 이벤트 핸들러를 Reflection을 이용해 구독하는 방법파일 다운로드1
12609정성태4/22/20219715.NET Framework: 1045. C# - 런타임 시점에 이벤트 핸들러를 만들어 Reflection을 이용해 구독하는 방법파일 다운로드1
12608정성태4/21/202110577.NET Framework: 1044. C# - Generic Host를 이용해 .NET 5로 리눅스 daemon 프로그램 만드는 방법 [9]파일 다운로드1
12607정성태4/21/20219074.NET Framework: 1043. C# - 실행 시점에 동적으로 Delegate 타입을 만드는 방법파일 다운로드1
12606정성태4/21/202113223.NET Framework: 1042. C# - enum 값을 int로 암시적(implicit) 형변환하는 방법? [2]파일 다운로드1
12605정성태4/18/20219142.NET Framework: 1041. C# - AssemblyID, ModuleID를 관리 코드에서 구하는 방법파일 다운로드1
12604정성태4/18/20217723VS.NET IDE: 163. 비주얼 스튜디오 속성 창의 "Build(빌드)" / "Configuration(구성)"에서의 "활성" 의미
12603정성태4/16/20218615VS.NET IDE: 162. 비주얼 스튜디오 - 상속받은 컨트롤이 디자인 창에서 지원되지 않는 문제
12602정성태4/16/20219812VS.NET IDE: 161. x64 DLL 프로젝트의 컨트롤이 Visual Studio의 Designer에서 보이지 않는 문제 [1]
12601정성태4/15/20218876.NET Framework: 1040. C# - REST API 대신 github 클라이언트 라이브러리를 통해 프로그래밍으로 접근
12600정성태4/15/20219053.NET Framework: 1039. C# - Kubeconfig의 token 설정 및 인증서 구성을 자동화하는 프로그램
12599정성태4/14/20219794.NET Framework: 1038. C# - 인증서 및 키 파일로부터 pfx/p12 파일을 생성하는 방법파일 다운로드1
12598정성태4/14/20219913.NET Framework: 1037. openssl의 PEM 개인키 파일을 .NET RSACryptoServiceProvider에서 사용하는 방법 (2)파일 다운로드1
... 31  32  33  34  35  36  37  38  39  40  [41]  42  43  44  45  ...