从视频中提取帧以便使用opencv

时间:2016-09-14 15:01:29

标签: c++ opencv video frame

我正在开发一个程序来检测视频或图像中的对象。 它适用于图像,但现在我想用它与视频。我使用特定的文件夹来选择图像,所以我想在检测之前保存该文件夹中的视频帧。 变量视频和salvataggio是alredy设置的。 在下面的代码中,我导航抛出文件夹以分析视频:

DIR *dir;
dir = opendir(video.c_str());
string vidName;
struct dirent *ent;
if (dir != NULL) {
        while ((ent = readdir (dir)) != NULL) {
            vidName= ent->d_name;
            if(vidName.compare(".")!= 0 && vidName.compare("..")!= 0)
            {
            //string vidPath(neg + vidName);
                estraiframe(video, vidName, salvataggio);
            }
        }
        closedir (dir);

}
else {
    cout<<"directory "<< video << " not present"<<endl;
}
}

函数estraiframe将帧保存在输出文件夹中。

void estraiframe(string path, string vidName, string output){

string vidPath(path + vidName);
VideoCapture cap(vidPath);
if( !cap.isOpened()){
        cout << "Cannot open the video file" << endl;
        return;
}

double count = cap.get(CV_CAP_PROP_FRAME_COUNT);
double rate = cap.get(CV_CAP_PROP_FPS);
int counter = 0;
for (int i=1; i< count; i+=rate*5)
{


cap.set(CV_CAP_PROP_POS_FRAMES,i);

Mat frame;
cap.read(frame);

counter++;
string nomeframe = to_string(counter) + "-frame_from"+vidName+".jpg";
string percorso (output+nomeframe);
cout << percorso;
imwrite(percorso,frame);
}
}

显然它可以工作但是在最后一帧之后它给了我以下错误:

  

断言stream_index&lt; ogg-&gt; nstreams在libavformat / oggdec.c:898失败   我锁定它但我找不到错误在哪里

2 个答案:

答案 0 :(得分:0)

您的视频包含各种流索引,例如3个音频和1个视频流将产生4个流索引。我建议您检查视频所包含的流量,并打印ogg-&gt; nstreams以检查是否相符。查看第898行oggdec.c中的源代码

static int ogg_read_seek(AVFormatContext *s, int stream_index,
                         int64_t timestamp, int flags)
{
    struct ogg *ogg       = s->priv_data;
    struct ogg_stream *os = ogg->streams + stream_index;
    int ret;

    av_assert0(stream_index < ogg->nstreams); /* Line 898 */

你显然已经离开这里了。

答案 1 :(得分:0)

显然我设法解决它,我修改了提取框架的功能。

void estraiframe(string path, string vidName, string output){
string vidPath(path + vidName);
VideoCapture cap(vidPath);
if( !cap.isOpened()){
        cout << "Cannot open the video file" << endl;
        return;
}

double rate = cap.get(CV_CAP_PROP_FPS);
int counter = 0;
Mat frame;
int i=1;

while(1)
{
     cap.read ( frame);
     if( frame.empty()) break;
     counter++;

     if (counter == rate*5*i){
     i++;
     string nomeframe = to_string(counter) + "-frame_from"+vidName+".jpg";
     string percorso (output+nomeframe);
     imwrite(percorso, frame);
     }

     char key = waitKey(10);
     if ( key == 27) break;
}
}