淺陋見解,僅供參考。拋磚引玉,多加指教!
          石正
          一個計(jì)算機(jī)技術(shù)愛好者! 雖然我擁有了系統(tǒng)分析師的證書,但我仍然是一個計(jì)算機(jī)技術(shù)愛好者!
          posts - 119,comments - 73,trackbacks - 0
          為了對avi進(jìn)行讀寫,微軟提供了一套API,總共50個函數(shù),他們的用途主要有兩類,一個是avi文件的操作,一類是數(shù)據(jù)流streams的操作。

            1、打開和關(guān)閉文件

            AVIFileOpen ,AVIFileAddRef, AVIFileRelease

            2、從文件中讀取文件信息

            通過AVIFileInfo可以獲取avi文件的一些信息,這個函數(shù)返回一個AVIFILEINFO結(jié)構(gòu),通過AVIFileReadData可以用來獲取AVIFileInfo函數(shù)得不到的信息。這些信息也許不包含在文件的頭部,比如擁有file的公司和個人的名稱。

            3、寫入文件信息

            可以通過AVIFileWriteData函數(shù)來寫入文件的一些額外信息。

            4、打開和關(guān)閉一個流

            打開一個數(shù)據(jù)流就跟打開文件一樣,你可以通過 AVIFileGetStream函數(shù)來打開一個數(shù)據(jù)流,這個函數(shù)創(chuàng)建了一個流的接口,然后在該接口中保存了一個句柄。

            如果你想操作文件的某一個單獨(dú)的流,你可以采用AVIStreamOpenFromFile函數(shù),這個函數(shù)綜合了AVIFileOpen和AVIFileGetStream函數(shù)。

            如果你想操作文件中的多個數(shù)據(jù)流,你就要首先AVIFileOpen,然后AVIFileGetStream。

            可以通過AVIStreamAddRef來增加stream接口的引用。

            通過AVIStreamRelease函數(shù)來關(guān)閉數(shù)據(jù)流。這個函數(shù)用來減少streams的引用計(jì)數(shù),當(dāng)計(jì)數(shù)減少為0時(shí),刪除。

            5、從流中讀取數(shù)據(jù)和信息

            AVIStreamInfo函數(shù)可以獲取數(shù)據(jù)的一些信息,該函數(shù)返回一個AVISTREAMINFO結(jié)構(gòu),該結(jié)構(gòu)包含了數(shù)據(jù)的類型壓縮方法,建議的buffersize,回放的rate,以及一些description。

            如果數(shù)據(jù)流還有一些其它的額外的信息,你可以通過AVIStreamReadData函數(shù)來獲取。應(yīng)用程序分配一個內(nèi)存,傳遞給這個函數(shù),然后這個函數(shù)會通過這個內(nèi)存返回?cái)?shù)據(jù)流的信息,額外的信息可能包括數(shù)據(jù)流的壓縮和解壓縮的方法,你可以通過AVIStreamDataSize宏來回去需要申請內(nèi)存塊的大小。

            可以通過AVIStreamReadFormat函數(shù)獲取數(shù)據(jù)流的格式信息。這個函數(shù)通過指定的內(nèi)存返回?cái)?shù)據(jù)流的格式信息,比如對于視頻流,這個buffer包含了一個BIMAPINFO結(jié)構(gòu),對于音頻流,內(nèi)存塊包含了WAVEFORMATEX或者PCMAVEFORMAT結(jié)構(gòu)。你可以通過給AVIStreamReadFormat傳遞一個空buffer就可以獲取buffer的大小。也可以通過AVIStreamFormatSize宏。

            可以通過AVIStreamRead函數(shù)來返回多媒體的數(shù)據(jù)。這個函數(shù)將數(shù)據(jù)復(fù)制到應(yīng)用程序提供的內(nèi)存中,對于視頻流,這個函數(shù)返回圖像禎,對于音頻流,這個函數(shù)返回音頻的sample數(shù)據(jù)。可以通過給AVIStreamRead傳遞一個NULL的buffer來獲取需要的buffer的大小。也可以通過AVIStreamSampleSize宏來獲取buffer的大小。

            有些AVI數(shù)據(jù)流句柄可能需要在啟動數(shù)據(jù)流的前要做一下準(zhǔn)備工作,此時(shí),我們可以調(diào)用AVIStreamBeginStreaming函數(shù)來告知AVI數(shù)據(jù)流handle來申請分配它需要的一些資源。在完畢后,調(diào)用AVIStreamEndStreamming函數(shù)來釋放資源。

            6、操作壓縮的視頻數(shù)據(jù)

            如果你要演示一禎或者幾禎壓縮視頻圖像時(shí),你可以調(diào)用AVIStreamRead函數(shù),將獲取的數(shù)據(jù)傳遞給DrawDib函數(shù)來顯示圖像。這些函數(shù)可以顯示壓縮和未壓縮的圖像。

            AVIFile也提供了一個函數(shù)AVIStreamGetFrameOpen,來獲取未壓縮的視頻禎,這個函數(shù)創(chuàng)建了內(nèi)存來獲取未壓縮的數(shù)據(jù)。也可以通過AVIStreamGetFrame函數(shù)來解壓縮一個單獨(dú)的視頻禎。這個函數(shù)可以解壓縮某一禎圖像,然后將數(shù)據(jù)以一個BIMAPINFOHEADER結(jié)構(gòu)返回。當(dāng)你調(diào)用完AVIStreamGetFrame函數(shù)后,要調(diào)用AVIStreamGetFrameClose函數(shù)釋放上一個函數(shù)申請的資源。

            7、根據(jù)已存在的數(shù)據(jù)流創(chuàng)建文件

            創(chuàng)建一個包含多個數(shù)據(jù)流的文件的方法就是整合多個數(shù)據(jù)流,將其寫入一個新文件。這些數(shù)據(jù)流可以是內(nèi)存中的數(shù)據(jù),也可以是存在于另一個文件中。

            我們可以用AVISave這個函數(shù)來build一個文件。這個函數(shù)可以創(chuàng)建一個文件,并且將指定的多個數(shù)據(jù)流按照指定的順序?qū)懭胛募阋部梢酝ㄟ^AVISaveV函數(shù)來創(chuàng)建一個新的文件,這個函數(shù)的功能和AVISave的功能一樣,主要區(qū)別是AVISaveV采用的數(shù)據(jù)流數(shù)組,而AVISave是單個的數(shù)據(jù)流,多次保存。

            我們可以調(diào)用AVISaveOptions函數(shù)來顯示一個對話框,可以讓用戶來選擇壓縮方式。

            我們可以在調(diào)用AVISave和AVISaveV函數(shù)時(shí)指定一個回調(diào)函數(shù),用來顯示avi文件的生成進(jìn)度,可以讓用戶隨時(shí)地取消生成avi文件。

            我們可以調(diào)用GetSaveFileNamePreview函數(shù)來顯示保存的對話框讓用戶選擇保存的文件名。

            通過AVIMakeFileFromStreams函數(shù)我們可以創(chuàng)建一個虛擬的文件句柄,其他的avi函數(shù)可以通過這個虛擬的文件句柄來操作文件中的數(shù)據(jù)流,操作完畢要記得調(diào)用AVIFileRelease釋放。

          8、向文件寫入一個數(shù)據(jù)流

            我們可以通過AVIFileCreateStream函數(shù)來在一個新文件或者已經(jīng)存在的文件中創(chuàng)建一個數(shù)據(jù)流。這個函數(shù)根據(jù)AVISTREAMINFO結(jié)構(gòu)定義了新的數(shù)據(jù)流,并為新的數(shù)據(jù)流創(chuàng)建一個接口,返回接口的指針。

            在寫入新的數(shù)據(jù)前,一定要指定流的格式信息,通過AVIStreamSetFormat函數(shù),當(dāng)設(shè)置一個視頻流的時(shí)候,一定要使用BIMAPINFO結(jié)構(gòu)來設(shè)置,音頻就用WAVEFORMAT。

            然后我們就可以通過AVIStreamWrite函數(shù)將我們的多媒體數(shù)據(jù)寫入數(shù)據(jù)流了。這個函數(shù)將應(yīng)用程序提供的內(nèi)存數(shù)據(jù)復(fù)制到指定的流。缺省的avi handler將數(shù)據(jù)寫入流的最后。

            如果你有其他額外的信息需要寫入流,你可以調(diào)用AVIFileWriteData或者AVIStreamWriteData,最后記得在完成數(shù)據(jù)寫入后,要調(diào)用AVIStreamRelease。

            9、數(shù)據(jù)流中的禎的位置

            尋找起始禎:

            可以通過AVIStreamStart函數(shù)來獲取第一禎包含的sample number。也可以通過AVIStreamInfo函數(shù)來獲取這個信息,這個函數(shù)的AVISTREAMINFO結(jié)構(gòu)中包含了dwStart,可以通過AVIStreamStartTime宏來獲取第一個sample。

            可以通過AVIStreamLength函數(shù)來獲取流的長度。這個函數(shù)返回流中的sample的數(shù)目。也可以通過AVIStreamInfo函數(shù)來獲取這些信息,可以通過AVIStreamLengthTime宏來獲取流的長度,毫秒。

            在視頻流中,一個sample對應(yīng)著一禎圖像,所以,有時(shí)這些sample中沒有視頻數(shù)據(jù),如果你調(diào)用AVIStreamRead函數(shù)來數(shù)據(jù),可能返回NULL,也可以通過AVIStreamFindSample通過指定FIND_ANY標(biāo)志來查找指定的sample。

            查找關(guān)鍵禎

            通過AVIStreamFindSample函數(shù)查找符合要尋找的sample,然后可以通過下面的宏判斷是否關(guān)鍵禎。

            在time和sample間互相切換。

            AVIStreamSampleToTime這個函數(shù)可以將smaple轉(zhuǎn)換成毫秒。對于視頻,這個值代表的是這個禎開始播放的時(shí)間。

            在了解了上面的知識后,我們對avi的文件結(jié)構(gòu)以及如何操作avi文件心里就明白了,下面我們可以開始我們的編程了。我們要做兩件事情:

            1、如何將一組靜態(tài)的bmp位圖合成一個avi的視頻文件;

            2、如何將一個未壓縮的avi文件解析成一幅幅位圖。

            示例程序界面如下:


            下面的函數(shù)演示了如何將一個文件夾下面的所有bmp文件都保存為一個avi文件,函數(shù)的第一個參數(shù)是要生成的AVI的文件名,第二個參數(shù)是存放bmp文件的文件夾名,這個函數(shù)會枚舉該文件夾下的所有bmp文件,合成一個AVI文件。


          void Cbmp2aviDlg::AVItoBmp(CString strAVIFileName, CString strBmpDir)
          {
           // TODO: 在此添加控件通知處理程序代碼
           AVIFileInit();
           PAVIFILE avi;
           int res=AVIFileOpen(&avi, strAVIFileName, OF_READ, NULL);
           int n = GetLastError();
           if (res!=AVIERR_OK)
           {
            //an error occures
            if (avi!=NULL)
             AVIFileRelease(avi);
            return ;
           }
           AVIFILEINFO avi_info;
           AVIFileInfo(avi, &avi_info, sizeof(AVIFILEINFO));
           PAVISTREAM pStream;
           res=AVIFileGetStream(avi, &pStream, streamtypeVIDEO /*video stream*/,
             0 /*first stream*/);
           if (res!=AVIERR_OK)
           {
            if (pStream!=NULL)
             AVIStreamRelease(pStream);
             AVIFileExit();
            return ;
           }

           //do some task with the stream
           int iNumFrames;
           int iFirstFrame;
           iFirstFrame=AVIStreamStart(pStream);
           if (iFirstFrame==-1)
           {
            //Error getteing the frame inside the stream
            if (pStream!=NULL)
             AVIStreamRelease(pStream);
            AVIFileExit();
            return ;
           }
           iNumFrames=AVIStreamLength(pStream);
           if (iNumFrames==-1)
           {
            //Error getteing the number of frames inside the stream
            if (pStream!=NULL)
             AVIStreamRelease(pStream);
            AVIFileExit();
            return ;
           }

           //getting bitmap from frame
           BITMAPINFOHEADER bih;
           ZeroMemory(&bih, sizeof(BITMAPINFOHEADER));

           bih.biBitCount=24; //24 bit per pixel
           bih.biClrImportant=0;
           bih.biClrUsed = 0;
           bih.biCompression = BI_RGB;
           bih.biPlanes = 1;
           bih.biSize = 40;
           bih.biXPelsPerMeter = 0;
           bih.biYPelsPerMeter = 0;
           //calculate total size of RGBQUAD scanlines (DWORD aligned)
           bih.biSizeImage = (((bih.biWidth * 3) + 3) & 0xFFFC) * bih.biHeight ;

           PGETFRAME pFrame;
           pFrame=AVIStreamGetFrameOpen(pStream, NULL );

           AVISTREAMINFO streaminfo;
           AVIStreamInfo(pStream,&streaminfo,sizeof(AVISTREAMINFO));

           //Get the first frame
           BITMAPINFOHEADER bih2;
           long lsize = sizeof(bih2);
           int index=0;
           for (int i=iFirstFrame; i<iNumFrames; i++)
           {
            index= i-iFirstFrame;
            BYTE* pDIB = (BYTE*) AVIStreamGetFrame(pFrame, index); //
            AVIStreamReadFormat(pStream,index,&bih2,&lsize);
            BITMAPFILEHEADER stFileHdr;

            BYTE* Bits=new BYTE[bih2.biSizeImage];
            AVIStreamRead(pStream,index,1,Bits,bih2.biSizeImage,NULL,NULL);
            //RtlMoveMemory(Bits, pDIB + sizeof(BITMAPINFOHEADER), bih2.biSizeImage);

            bih2.biClrUsed =0;
            stFileHdr.bfOffBits=sizeof(BITMAPFILEHEADER)+sizeof(BITMAPINFOHEADER);
            stFileHdr.bfSize=sizeof(BITMAPFILEHEADER);
            stFileHdr.bfType=0x4d42;

            CString FileName;
            FileName.Format("Frame-%05d.bmp", index);
            CString strtemp = strBmpDir;
            strtemp += "\\";
            strtemp += FileName;
            FILE* fp=_tfopen(strtemp ,_T("wb"));
            fwrite(&stFileHdr,1,sizeof(BITMAPFILEHEADER),fp);
            fwrite(&bih2,1,sizeof(BITMAPINFOHEADER),fp);
            int ff = fwrite(Bits,1,bih2.biSizeImage,fp);
            int e = GetLastError();
            fclose(fp);
            /////
            delete Bits;
            //CreateFromPackedDIBPointer(pDIB, index);
           }

           AVIStreamGetFrameClose(pFrame);

           //close the stream after finishing the task
           if (pStream!=NULL)
            AVIStreamRelease(pStream);
           AVIFileExit();
          }

            下面的這個函數(shù)演示了如何將AVI文件中的每一楨圖像單獨(dú)取出來,保存為bmp文件。函數(shù)的頭一個參數(shù)是avi文件名,第二個參數(shù)是存放bmp文件的文件夾。

          //生成avi
          void Cbmp2aviDlg::BMPtoAVI(CString szAVIName, CString strBmpDir)
          {
           CFileFind finder;
           strBmpDir += _T("\\*.*");
           AVIFileInit();
           AVISTREAMINFO strhdr;
           PAVIFILE pfile;
           PAVISTREAM ps;
           int nFrames =0;
           HRESULT hr;

           BOOL bFind = finder.FindFile(strBmpDir);
           while(bFind)
           {
            bFind = finder.FindNextFile();
            if(!finder.IsDots() && !finder.IsDirectory())
            {
             CString str = finder.GetFilePath();
             FILE *fp = fopen(str,"rb");
             BITMAPFILEHEADER bmpFileHdr;
             BITMAPINFOHEADER bmpInfoHdr;
             fseek( fp,0,SEEK_SET);
             fread(&bmpFileHdr,sizeof(BITMAPFILEHEADER),1, fp);
             fread(&bmpInfoHdr,sizeof(BITMAPINFOHEADER),1, fp);

             BYTE *tmp_buf = NULL;
             if(nFrames ==0 )
             {
              AVIFileOpen(&pfile,szAviName,OF_WRITE | OF_CREATE,NULL);
              _fmemset(&strhdr, 0, sizeof(strhdr));
              strhdr.fccType = streamtypeVIDEO;// stream type
              strhdr.fccHandler = 0;
              strhdr.dwScale = 1;
              strhdr.dwRate = 15; // 15 fps
              strhdr.dwSuggestedBufferSize = bmpInfoHdr.biSizeImage ;
              SetRect(&strhdr.rcFrame, 0, 0, bmpInfoHdr.biWidth, bmpInfoHdr.biHeight);

              // And create the stream;
              hr = AVIFileCreateStream(pfile,&ps,&strhdr);
              // hr = AVIStreamSetFormat(ps,nFrames,&bmpInfoHdr,sizeof(bmpInfoHdr));
             }
             tmp_buf = new BYTE[bmpInfoHdr.biWidth * bmpInfoHdr.biHeight * 3];
             fread(tmp_buf, 1, bmpInfoHdr.biWidth * bmpInfoHdr.biHeight * 3, fp);
             hr = AVIStreamSetFormat(ps,nFrames,&bmpInfoHdr,sizeof(bmpInfoHdr));
             hr = AVIStreamWrite(ps, // stream pointer
                nFrames , // time of this frame
                1, // number to write
                (LPBYTE) tmp_buf,
                bmpInfoHdr.biSizeImage , // size of this frame
                AVIIF_KEYFRAME, // flags....
                NULL,
                NULL);

             nFrames ++;
             fclose(fp);
            }
           }

           AVIStreamClose(ps);

           if(pfile != NULL)
            AVIFileRelease(pfile);
           AVIFileExit();
          }

            結(jié)束語:

            以上代碼在 vc 6.0 和windows xp平臺調(diào)試通過。這兩個函數(shù)你可以直接在你的程序中使用,更詳細(xì)的代碼可以參見隨著本文附上的示例源碼。這里我要指出的是,這個AVI文件和bmp互相轉(zhuǎn)換過程中,avi中的視頻數(shù)據(jù)都是存放的是沒有壓縮的數(shù)據(jù),如果你要分解AVI文件是經(jīng)過壓縮編碼,比如,DVSD,MPEG4編碼,首先你要采用相應(yīng)的解碼器對視頻數(shù)據(jù)解碼,然后將解碼過的數(shù)據(jù)保存為bmp文件。好了,關(guān)于avi文件的介紹就到這里結(jié)束了。

          posted on 2007-10-24 10:30 石正 閱讀(7542) 評論(16)  編輯  收藏

          FeedBack:
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2007-11-05 17:04 | 劃過天空
          我用你的代碼寫了,怎么生成的圖片都是空白的阿  回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2007-11-27 20:00 | mosina
          不錯,學(xué)習(xí)中。多謝  回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2008-01-02 15:59 | VC學(xué)習(xí)者
          怎樣用API寫這個程序呢?謝謝  回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2008-01-05 15:51 | mafei
          我用你的代碼寫了,怎么生成的圖片都是空白的阿
          能否將源程序給出啊,
          謝謝
          email: mafei0603@163.com

          謝謝
            回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2009-08-14 12:46 | 工學(xué)
          兩個函數(shù)的說明顛倒了,不知道誰轉(zhuǎn)誰的,一錯皆錯,呼  回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2009-12-04 16:57 | 韓增義
          能不能也給我一份你的源碼呢?  回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2009-12-04 16:58 | 韓增義
          我的郵箱是hanzengyi000@163.com  回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2010-09-17 15:00 | dfdf
          怎么現(xiàn)在沒有下載鏈接了???  回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作 [未登錄]
          2010-12-09 22:55 |
          931318912@qq.com我的郵箱,謝謝  回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2011-07-12 23:36 | 阿斯達(dá)
          yzwkin@163.com 代碼可以給我嗎?謝謝  回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2011-07-12 23:36 | 阿斯達(dá)
          阿大蘇打撒旦撒旦撒旦撒旦撒打算的  回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2012-04-01 17:57 | Jia
          非常好的文章,謝謝,辛苦了!  回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作 [未登錄]
          2012-05-09 00:34 | su
          有源代碼嗎??qq:603552120  回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2014-04-24 16:39 | 學(xué)習(xí)
          FindFile 與合并 應(yīng)該放在2個函數(shù)里   回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2015-11-24 18:53 | wodeball
          純屬誤導(dǎo)呀。。。。。不夠完整  回復(fù)  更多評論
            
          # re: vc++實(shí)現(xiàn)avi文件的操作
          2015-11-24 20:51 | wodeball
          hr = AVIStreamWrite(ps, // stream pointer
                nFrames , // time of this frame
                1, // number to write
                (LPBYTE) tmp_buf,
          //改為 bmpInfoHdr.biWidth * bmpInfoHdr.biHeight * 3
                //bmpInfoHdr.biSizeImage , // size of this frame
                AVIIF_KEYFRAME, // flags....
                NULL,
                NULL);
          查了原因,因?yàn)橛行┑膱D那個大小為0,所以做成的AVI都是空的。  回復(fù)  更多評論
            

          只有注冊用戶登錄后才能發(fā)表評論。


          網(wǎng)站導(dǎo)航:
           
          淺陋見解,僅供參考。拋磚引玉,多加指教!
          主站蜘蛛池模板: 错那县| 浪卡子县| 成武县| 横峰县| 定边县| 皮山县| 富裕县| 沁水县| 光山县| 南安市| 靖江市| 时尚| 万源市| 胶州市| 家居| 阳朔县| 视频| 象山县| 陵水| 凤城市| 远安县| 略阳县| 斗六市| 铁岭市| 庄河市| 吉水县| 纳雍县| 察雅县| 平罗县| 桃园县| 阳高县| 屏东市| 峨边| 旬邑县| 自贡市| 孟津县| 上虞市| 和田市| 青神县| 颍上县| 苗栗县|