在python中调用OpenposeDemo.exe遇到的问题

在复现st-gcn时,需要用到Openpose提取视频中的骨骼信息,但是使用的时候我却遇到了很多问题。

准备工作

下载、编译openpose,这个网上也有很多教程,这里我不赘述,参考openpose安装到实战(win10)

调用Openpsoe

import os
import shutil
from pathlib import Path
import json
from .utils import get_frames


def extract_skeleon(path_data, path_out, label, index,
                    openposeCmd='D:/githouse/openpose/build/x64/Release/OpenPoseDemo.exe',
                    model='COCO'):
    '''
    利用OpenPoseDemo.exe处理指定路径下的视频文件,生成.json的骨骼文件以及处理后的视频
    param
        path_data: 视频路径
        path_out: 保存的结果的路径
        label: 视频标签
        index: 标签索引
    return
    '''
    videos = os.listdir(path_data)
    path_out_snippets = path_out + '/snippets'
    if not os.path.exists(path_out_snippets):
        os.makedirs(path_out_snippets)
    path_out_sequence = path_out + '/data'
    if not os.path.exists(path_out_sequence):
        os.makedirs(path_out_sequence)
    for video in videos:
        path_video = '{}/{}'.format(path_data, video)

        video_name = video.split('.')[0]

        ## 姿态估计,输出处理后的视频
        path_out_video = '{}/{}'.format(path_out_snippets, video_name)
        openpose_args = dict(
            video=path_video,
            write_json=path_out_video,
            display=0,
            render_pose=0,
            model_pose=model)
        command_line = openposeCmd + ' '
        command_line += ' '.join(['--{} {}'.format(k, v) for k, v in openpose_args.items()])
        shutil.rmtree(path_out_video, ignore_errors=True)
        print('执行指令:', command_line)
        os.system(command_line)

        ## 生成骨骼数据文件
        video = get_frames(path_video)

        height, width, _ = video[0].shape

        # 这里可以修改label, label_index
        video_info = json_pack(path_out_video, video_name, width, height, label, index)
        path_out_json = path_out_sequence+'/{}.json'.format(video_name)
        with open(path_out_json, 'w') as outfile:
            json.dump(video_info, outfile)
        if len(video_info['data']) == 0:
            print('{} Can not find pose estimation results.'.format(video))
        else:
            print('{} pose estimation complete.'.format(video))
        return


def json_pack(snippets_dir, video_name, frame_width, frame_height, label='unknown', label_index=-1):
    '''
    将 视频提取的 *.json数据打包成dict
    param
        snippets_dir:
        video_name:
        frame_width:
        frame_height:
        label:
        label_index:
    return:
        video_info:dict{
            data:list(dict{
                skeleton:list
                pose:2d坐标[x,y]
                score:坐标的置信分数
            })
            label:
            label_index:
        }
    '''
    sequence_info = []
    p = Path(snippets_dir)
    for path in p.glob(video_name + '*.json'):
        json_path = str(path)
        print(path)
        frame_id = int(path.stem.split('_')[-2])
        frame_data = {'frame_index': frame_id}
        data = json.load(open(json_path))
        skeletons = []
        for person in data['people']:
            score, coordinates = [], []
            skeleton = {}
            keypoints = person['pose_keypoints_2d']
            for i in range(0, len(keypoints), 3):
                coordinates += [keypoints[i]/frame_width, keypoints[i + 1]/frame_height]
                score += [keypoints[i + 2]]
            skeleton['pose'] = coordinates
            skeleton['score'] = score
            skeletons += [skeleton]
        frame_data['skeleton'] = skeletons
        sequence_info += [frame_data]

    video_info = dict()
    video_info['data'] = sequence_info
    video_info['label'] = label
    video_info['label_index'] = label_index

    return video_info

问题

问题1:路径有空格

但是我遇到了一个问题,输出提示我

Can not find pose estimation results

我将指令复制,打开cmd执行,同样报错,指令如下:

D:/githouse/openpose/build/x64/Release/OpenPoseDemo.exe --video E:/DataSet/test data/resize/boxing/person01_boxing_d1_uncomp(003320-021320).avi --write_json E:/DataSet/test data/skeleton/boxing/snippets/person01_boxing_d1_uncomp(003320-021320) --display 0 --render_pose 0 --model_pose COCO

根据命令行的提示我找到了原因,路径中有空格,导致路径无法被识别

问题2:找不到*.dll

我将数据集路径以及代码修改后,再次运行还是出现了

Can not find pose estimation results

同样我在cmd中执行,这次提示找不到glob.dll,...,反正就是找不到很多.dll文件,经过一番操作,我发现这些文件在D:/githouse/openpose/build/bin中找到了,于是我把这些文件复制到OpenposeDemo.exe所在的目录,我这里是:

D:/githouse/openpose/build/x64/Release/OpenPoseDemo.exe

然后该问题就没出现了

问题3:工作路径没有设置正确

解决了前两个问题,运行还是报错,这次命令行给我三个可能的原因:

(1)openpose需要的包没有下载

(2)工作路径没有在model文件夹所在路径

(3)文件路径有空格

一眼看1,3是不可能的,只能是2了,于是我在pycharm中把工作路径设置为model文件夹的父路径,即openpose的安装路径。

问题解决了,运行成功,nice。这里附上全部代码

utils.py

import os
import skvideo
# skvideo.setFFmpegPath('S:/ffmpeg/bin')
import skvideo.io
import numpy as np
import cv2

def parse_info(video_info, num_person_in=5, num_person_out=2):
    data_numpy = np.zeros((3, len(video_info['data']), 18, num_person_in))
    for frame_info in video_info['data']:
        frame_index = frame_info['frame_index']
        for m, skeleton_info in enumerate(frame_info["skeleton"]):
            if m >= num_person_in:
                break
            pose = skeleton_info['pose']
            score = skeleton_info['score']
            data_numpy[0, frame_index, :, m] = pose[0::2]
            data_numpy[1, frame_index, :, m] = pose[1::2]
            data_numpy[2, frame_index, :, m] = score

    # centralization
    data_numpy[0:2] = data_numpy[0:2] - 0.5
    data_numpy[0][data_numpy[2] == 0] = 0
    data_numpy[1][data_numpy[2] == 0] = 0

    sort_index = (-data_numpy[2, :, :, :].sum(axis=1)).argsort(axis=1)
    for t, s in enumerate(sort_index):
        data_numpy[:, t, :, :] = data_numpy[:, t, :, s].transpose((1, 2,
                                                                    0))
    data_numpy = data_numpy[:, :, :, :num_person_out]

    label = video_info['label_index']
    return data_numpy, label

def get_frames(video_path):
    vread = skvideo.io.vread(video_path)
    video = []
    for frame in vread:
        video.append(frame)
    return video

def play(video_path, fps=30):
    cap = cv2.VideoCapture(video_path)
    while(cap.isOpened()):
        ret, frame = cap.read()
        gray = cv2.cvtColor(frame, cv2.COLOR_BGR2GRAY)
        cv2.imshow('frame', gray)
        if cv2.waitKey(1000/fps) & 0xFF == ord('q'):
            break
    cap.release()
    cv2.destroyAllWindows()

def resize(path_data, path_out, size = (340,256)):
    '''
    将输入路径下的视频大小调整为指定大小,默认为340x256,并将修改后的视频保存到指定路径
    param
        path_data:
        path_out:
    '''
    videos = os.listdir(path_data)
    size_str = '{}x{}'.format(size[0], size[1])
    for video in videos:
        video_path = '{}/{}'.format(path_data, video)
        out_path = '{}/{}'.format(path_out, video)
        writer = skvideo.io.FFmpegWriter(out_path,
                outputdict={'-f': 'mp4', '-vcodec': 'libx264', '-s': size_str, '-r': '30'})

        reader = skvideo.io.FFmpegReader(video_path)
        for frame in reader.nextFrame():
            writer.writeFrame(frame)
        writer.close()
        print('{} resize success'.format(video))

skeleton.py

import os
import shutil
from pathlib import Path
import json
from .utils import get_frames


def extract_skeleon(path_data, path_out, label, index,
                    openposeCmd='D:/githouse/openpose/build/x64/Release/OpenPoseDemo.exe',
                    model='COCO'):
    '''
    利用OpenPoseDemo.exe处理指定路径下的视频文件,生成.json的骨骼文件以及处理后的视频
    param
        path_data: 视频路径
        path_out: 保存的结果的路径
        label: 视频标签
        index: 标签索引
    return
    '''
    videos = os.listdir(path_data)
    path_out_snippets = path_out + '/snippets'
    if not os.path.exists(path_out_snippets):
        os.makedirs(path_out_snippets)
    path_out_sequence = path_out + '/data'
    if not os.path.exists(path_out_sequence):
        os.makedirs(path_out_sequence)
    for video in videos:
        path_video = '{}/{}'.format(path_data, video)

        video_name = video.split('.')[0]

        ## 姿态估计,输出处理后的视频
        path_out_video = '{}/{}'.format(path_out_snippets, video_name)
        openpose_args = dict(
            video=path_video,
            write_json=path_out_video,
            display=0,
            render_pose=0,
            model_pose=model)
        command_line = openposeCmd + ' '
        command_line += ' '.join(['--{} {}'.format(k, v) for k, v in openpose_args.items()])
        shutil.rmtree(path_out_video, ignore_errors=True)
        print('执行指令:', command_line)
        os.system(command_line)

        ## 生成骨骼数据文件
        video = get_frames(path_video)

        height, width, _ = video[0].shape

        # 这里可以修改label, label_index
        video_info = json_pack(path_out_video, video_name, width, height, label, index)
        path_out_json = path_out_sequence+'/{}.json'.format(video_name)
        with open(path_out_json, 'w') as outfile:
            json.dump(video_info, outfile)
        if len(video_info['data']) == 0:
            print('{} Can not find pose estimation results.'.format(video))
        else:
            print('{} pose estimation complete.'.format(video))
        return


def json_pack(snippets_dir, video_name, frame_width, frame_height, label='unknown', label_index=-1):
    '''
    将 视频提取的 *.json数据打包成dict
    param
        snippets_dir:
        video_name:
        frame_width:
        frame_height:
        label:
        label_index:
    return:
        video_info:dict{
            data:list(dict{
                skeleton:list
                pose:2d坐标[x,y]
                score:坐标的置信分数
            })
            label:
            label_index:
        }
    '''
    sequence_info = []
    p = Path(snippets_dir)
    for path in p.glob(video_name + '*.json'):
        json_path = str(path)
        print(path)
        frame_id = int(path.stem.split('_')[-2])
        frame_data = {'frame_index': frame_id}
        data = json.load(open(json_path))
        skeletons = []
        for person in data['people']:
            score, coordinates = [], []
            skeleton = {}
            keypoints = person['pose_keypoints_2d']
            for i in range(0, len(keypoints), 3):
                coordinates += [keypoints[i]/frame_width, keypoints[i + 1]/frame_height]
                score += [keypoints[i + 2]]
            skeleton['pose'] = coordinates
            skeleton['score'] = score
            skeletons += [skeleton]
        frame_data['skeleton'] = skeletons
        sequence_info += [frame_data]

    video_info = dict()
    video_info['data'] = sequence_info
    video_info['label'] = label
    video_info['label_index'] = label_index

    return video_info

preprocess.py


#!/usr/bin/env python
# coding:gbk
import os
from utils.skeleton import extract_skeleon



def PreProcess(path_data, path_resize, path_out):
    '''
    处理指定路径下的视频数据集,得到调整大小后的视频和提取的骨骼数据
    param
        path_data: 原视频数据路径
        path_resize: 调整大小后的视频路径
        path_out: 提取的骨骼数据路径
    '''
    video_list = os.listdir(path_data)
    video_num = len(video_list)
    for process_index in range(video_num):
        action = video_list[process_index]
        label = action
        index = process_index

        ## 1. 将视频调整到指定大小(默认340x256)
        path_data_action = '{}/{}'.format(path_data, action)
        path_resize_action = '{}/{}'.format(path_resize,action)
        # if not os.path.exists(path_resize_action):
        #     os.makedirs(path_resize_action)
        # resize(path_data_action, path_resize_action)


        # 2. 利用openpose提取每段视频骨骼点数据
        path_out_action = '{}/{}'.format(path_out,action)
        if not os.path.exists(path_out_action):
            os.makedirs(path_out_action)
        extract_skeleon(path_resize_action, path_out_action, label, index)

if __name__ == '__main__':
    path_data = 'E:/DataSet/test_data/origin'
    path_resize = 'E:/DataSet/test_data/resize'
    path_skeleton = 'E:/DataSet/test_data/skeleton'
    PreProcess(path_data, path_resize, path_skeleton)


  • 10
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Python,可以使用`subprocess`模块来调用和执行一个`.exe`文件。 首先需要导入`subprocess`模块: ```python import subprocess ``` 接下来,可以使用`subprocess.call()`函数来调用一个`.exe`文件,并打开它: ```python subprocess.call(['path/to/exe_file.exe']) ``` 其,`'path/to/exe_file.exe'`是你要调用的`.exe`文件的路径。 如果希望在执行完`.exe`文件后获取它的输出信息,可以使用`subprocess.check_output()`函数: ```python output = subprocess.check_output(['path/to/exe_file.exe']) print(output) ``` 如果你需要传递参数给`.exe`文件,也可以在调用时传递一个参数列表: ```python subprocess.call(['path/to/exe_file.exe', 'arg1', 'arg2', 'arg3']) ``` 其,`'arg1'`、`'arg2'`、`'arg3'`等是你要传递给`.exe`文件的参数。 这样,就可以通过Python调用一个`.exe`文件并打开它了。 ### 回答2: 在Python,可以使用`subprocess`模块调用一个.exe文件并打开它。 首先,需要导入`subprocess`模块: ```python import subprocess ``` 然后,可以使用`subprocess.run()`方法来执行一个命令。通过指定`shell=True`参数可以使用命令行来执行命令,包括打开一个.exe文件。在`subprocess.run()`方法,可以传入一个命令字符串作为参数,如下: ```python subprocess.run('path_to_exe_file.exe', shell=True) ``` 其,`path_to_exe_file.exe`表示你要打开的.exe文件的完整路径。 如果你的.exe文件需要一些输入参数,可以将它们添加到命令字符串: ```python subprocess.run('path_to_exe_file.exe input1 input2', shell=True) ``` 其,`input1`和`input2`是你的.exe文件所需的输入参数。 另外,如果你需要获取.exe文件的输出结果,可以使用`subprocess.run()`方法的`capture_output=True`参数,并使用`.stdout`属性来获取输出内容: ```python result = subprocess.run('path_to_exe_file.exe', shell=True, capture_output=True) output = result.stdout.decode('utf-8') print(output) ``` 其,`output`变量将包含你的.exe文件的输出结果。 最后,如果你需要等待打开的.exe文件执行完毕后再继续执行后续代码,可以使用`subprocess.run()`方法的`wait()`方法: ```python subprocess.run('path_to_exe_file.exe', shell=True).wait() ``` 这样,程序将会等待指定的.exe文件执行完毕后再继续执行后续代码。 ### 回答3: 在Python,可以使用subprocess模块来调用一个.exe文件并打开它。 下面是一个简单的示例,演示如何使用Python调用并打开一个.exe文件: ```python import subprocess # 定义要调用的.exe文件路径 exe_path = "path_to_your_exe_file.exe" # 使用subprocess调用.exe文件 subprocess.call(exe_path) ``` 以上代码,首先导入了subprocess模块。然后,通过设置`exe_path`变量,指定了要调用的.exe文件的路径。最后,使用`subprocess.call()`函数调用并打开了.exe文件。 请确保将`exe_path`替换为实际的.exe文件的路径。如果你希望传递一些参数给.exe文件,可以在`subprocess.call()`函数添加额外的参数。 需要注意的是,调用一个.exe文件可能会导致你的操作系统执行一个新的进程,这个进程可能会打开一个新的窗口来展示.exe文件的界面。因此,如果你在运行Python脚本时,窗口没有显示或迅速闪现一下,可能是因为.exe文件的界面打开并关闭得很快, 或者.exe文件可能确实没有图形界面。你可以尝试在命令行直接运行.exe文件,以验证它是否有图形界面。 希望以上回答对你有所帮助!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值