PaddleClas学习3——使用PPLCNet模型对车辆朝向进行识别(c++)

news2025/1/10 2:57:15

使用PPLCNet模型对车辆朝向进行识别

  • 1 准备环境
  • 2 准备模型
    • 2.1 模型导出
    • 2.2 修改配置文件
  • 3 编译
    • 3.1 使用CMake生成项目文件
    • 3.2 编译
    • 3.3 执行
    • 3.4 添加后处理程序
      • 3.4.1 postprocess.h
      • 3.4.2 postprocess.cpp
      • 3.4.3 在cls.h中添加函数声明
      • 3.4.4 在cls.cpp中添加函数定义
      • 3.4.5 在main.cpp中调用
  • 4 模型预测
    • 4.1 测试结果
    • 4.2 与python预测结果对比

1 准备环境

参考上一篇:Windows PaddleSeg c++部署

2 准备模型

2.1 模型导出

对上一篇 使用PPLCNet模型对车辆朝向进行识别 训练得到模型进行转换。将该模型转为 inference 模型只需运行如下命令:

python tools\export_model.py -c .\ppcls\configs\PULC\vehicle_attribute\PPLCNet_x1_0.yaml -o Global.pretrained_model=output/PPLCNet_x1_0/best_model -o Global.save_inference_dir=./deploy/models/class_vehicle_attribute_infer

训练得到的模型
图2.1 训练得到的模型
在这里插入图片描述
图2.2 导出的模型

2.2 修改配置文件

deploy/configs/PULC/vehicle_attribute/inference_vehicle_attribute.yaml
修改Global下的infer_imgsinference_model_dir

Global:
  infer_imgs: "./images/PULC/vehicle_attribute/0002_c002_00030670_0.jpg"
  inference_model_dir: "./models/class_vehicle_attribute_infer"
  batch_size: 1
  use_gpu: True
  enable_mkldnn: True
  cpu_num_threads: 10
  #benchmark: False
  enable_benchmark: False
  use_fp16: False
  ir_optim: True
  use_tensorrt: False
  gpu_mem: 8000
  enable_profile: False

3 编译

工程整体目录结构如下:

G:/paddle/c++
  ├── paddle_inference
G:/paddle
  ├── PaddleClas-release-2.5

3.1 使用CMake生成项目文件

在这里插入图片描述

3.2 编译

用Visual Studio 2022打开cpp\build\clas_system.sln,将编译模式设置为Release,点击生成->生成解决方案,在cpp\build\Release文件夹内生成clas_system.exe

3.3 执行

进入到build/Release目录下,将准备的模型和图片放到clas_system.exe同级目录,build/Release目录结构如下:

Release
├──clas_system.exe                # 可执行文件
├──images         				  # 测试图片
    ├── PULC
        ├── vehicle_attribute
        	├── 0002_c002_00030670_0.jpg
├──configs         				  # 配置文件
    ├── PULC
        ├── vehicle_attribute
        	├── inference_vehicle_attribute.yaml
├──models      					  # 推理用到的模型
    ├── class_vehicle_attribute_infer
    	├── inference.pdmodel          # 预测模型的拓扑结构文件
    	├── inference.pdiparams        # 预测模型的权重文件
    	└── inference.pdiparams.info   # 参数额外信息,一般无需关注
├──*.dll                          # dll文件

3.4 添加后处理程序

3.4.1 postprocess.h

// postprocess.h
#include <iostream>
#include <vector>

namespace PaddleClas {

	class VehicleAttribute {

	public:
		float color_threshold = 0.5;
		float type_threshold = 0.5;
		float direction_threshold = 0.5;

		std::vector<std::string> color_list = { "yellow", "orange", "green", "gray", "red", "blue", "white",
			"golden", "brown", "black" };
		std::vector<std::string> type_list = { "sedan", "suv", "van", "hatchback", "mpv", "pickup", "bus",
			"truck", "estate" };
		std::vector<std::string> direction_list = { "forward", "sideward", "backward" };

		std::string run(std::vector<float>& pred_data);
	};
}

3.4.2 postprocess.cpp

// postprocess.cpp

#include "include/postprocess.h"
#include <string>
namespace PaddleClas {
	std::string VehicleAttribute::run(std::vector<float>& pred_data) {
		int color_num = 10;
		int type_num = 9;
		int direction_num = 3;

		int index_color = std::distance(&pred_data[0], std::max_element(&pred_data[0], &pred_data[0] + 10));//左闭右开
		int index_type = std::distance(&pred_data[0] + 10, std::max_element(&pred_data[0] + 10, &pred_data[0] + 19));
		int index_direction = std::distance(&pred_data[0] + 19, std::max_element(&pred_data[0] + 19, &pred_data[0] + 22));

		std::string color_info, type_info, direction_info;
		if (pred_data[index_color] >= this->color_threshold) {
			color_info = "Color: (" + color_list[index_color] + ", pro: " + std::to_string(pred_data[index_color]) + ")";
		}
		if (pred_data[index_type + 10] >= this->type_threshold) {
			type_info = "Type: (" + type_list[index_type] + ", pro: " + std::to_string(pred_data[index_type + 10]) + ")";
		}
		if (pred_data[index_direction + 19] >= this->direction_threshold) {
			direction_info = "Direction: (" + direction_list[index_direction] + ", pro: " + std::to_string(pred_data[index_direction + 19]) + ")";
		}

		std::string pred_res = color_info + type_info + direction_info;
		pred_res += "pred: ";
		for (int i = 0; i < pred_data.size(); i++) {

			if (i < 10) {
				if (pred_data[i] > color_threshold) {
					pred_res += "1, ";
				}
				else {
					pred_res += "0, ";
				}
			}
			else if (i < 19) {
				if (pred_data[i] > type_threshold) {
					pred_res += "1, ";
				}
				else {
					pred_res += "0, ";
				}
			}
			else {
				if (pred_data[i] > direction_threshold) {
					pred_res += "1, ";
				}
				else {
					pred_res += "0, ";
				}
			}
		}
		return pred_res;
	}
}//namespace

3.4.3 在cls.h中添加函数声明

// Run predictor for vehicle attribute
void Run(cv::Mat& img, std::vector<float>& out_data, std::string &pred_res,
    std::vector<double>& times);

3.4.4 在cls.cpp中添加函数定义

void Classifier::Run(cv::Mat& img, std::vector<float>& out_data, std::string& pred_res,
    std::vector<double>& times){
    cv::Mat srcimg;
    cv::Mat resize_img;
    img.copyTo(srcimg);

    auto preprocess_start = std::chrono::system_clock::now();
    this->resize_op_.Run(img, resize_img, this->resize_size_);

    //this->resize_op_.Run(img, resize_img, this->resize_short_size_);

    //this->crop_op_.Run(resize_img, this->crop_size_);

    this->normalize_op_.Run(&resize_img, this->mean_, this->std_, this->scale_);
    std::vector<float> input(1 * 3 * resize_img.rows * resize_img.cols, 0.0f);
    this->permute_op_.Run(&resize_img, input.data());

    auto input_names = this->predictor_->GetInputNames();
    auto input_t = this->predictor_->GetInputHandle(input_names[0]);
    input_t->Reshape({ 1, 3, resize_img.rows, resize_img.cols });
    auto preprocess_end = std::chrono::system_clock::now();

    auto infer_start = std::chrono::system_clock::now();
    input_t->CopyFromCpu(input.data());
    this->predictor_->Run();

    auto output_names = this->predictor_->GetOutputNames();
    auto output_t = this->predictor_->GetOutputHandle(output_names[0]);
	std::vector<int> output_shape = output_t->shape();
	int out_num = std::accumulate(output_shape.begin(), output_shape.end(), 1,
		std::multiplies<int>());

	out_data.resize(out_num);
	output_t->CopyToCpu(out_data.data());
	auto infer_end = std::chrono::system_clock::now();

	auto postprocess_start = std::chrono::system_clock::now();
	pred_res = this->vehicle_attribute_op.run(out_data);
	auto postprocess_end = std::chrono::system_clock::now();

	std::chrono::duration<float> preprocess_diff =
		preprocess_end - preprocess_start;
	times[0] = double(preprocess_diff.count() * 1000);
	std::chrono::duration<float> inference_diff = infer_end - infer_start;
    double inference_cost_time = double(inference_diff.count() * 1000);
    times[1] = inference_cost_time;
    std::chrono::duration<float> postprocess_diff =
        postprocess_end - postprocess_start;
    times[2] = double(postprocess_diff.count() * 1000);
}

3.4.5 在main.cpp中调用

EFINE_string(config,
"./configs/PULC/vehicle_attribute/inference_vehicle_attribute.yaml", "Path of yaml file");
DEFINE_string(c,
"", "Path of yaml file");

int main(int argc, char** argv) {

    google::ParseCommandLineFlags(&argc, &argv, true);
    std::string yaml_path = "";
    if (FLAGS_config == "" && FLAGS_c == "") {
        std::cerr << "[ERROR] usage: " << std::endl
            << argv[0] << " -c $yaml_path" << std::endl
            << "or:" << std::endl
            << argv[0] << " -config $yaml_path" << std::endl;
        exit(1);
    }
    else if (FLAGS_config != "") {
        yaml_path = FLAGS_config;
    }
    else {
        yaml_path = FLAGS_c;
    }
    ClsConfig config(yaml_path);
    config.PrintConfigInfo();

    std::string path(config.infer_imgs);

    std::vector <std::string> img_files_list;
    if (cv::utils::fs::isDirectory(path)) {
        std::vector <cv::String> filenames;
        cv::glob(path, filenames);
        for (auto f : filenames) {
            img_files_list.push_back(f);
        }
    }
    else {
        img_files_list.push_back(path);
    }

    std::cout << "img_file_list length: " << img_files_list.size() << std::endl;

    Classifier classifier(config);

    std::vector<double> cls_times = { 0, 0, 0 };
    std::vector<double> cls_times_total = { 0, 0, 0 };
    double infer_time;
    std::vector<float> out_data;
    std::string result;
    int warmup_iter = 5;
    bool label_output_equal_flag = true;
    for (int idx = 0; idx < img_files_list.size(); ++idx) {
        std::string img_path = img_files_list[idx];
        cv::Mat srcimg = cv::imread(img_path, cv::IMREAD_COLOR);
        if (!srcimg.data) {
            std::cerr << "[ERROR] image read failed! image path: " << img_path
                << "\n";
            exit(-1);
        }

        cv::cvtColor(srcimg, srcimg, cv::COLOR_BGR2RGB);
        classifier.Run(srcimg, out_data, result, cls_times);

        std::cout << "Current image path: " << img_path << std::endl;
        infer_time = cls_times[0] + cls_times[1] + cls_times[2];
        std::cout << "Current total inferen time cost: " << infer_time << " ms."
            << std::endl;
        std::cout << "Current inferen result: " << result << " ."
            << std::endl;
        if (idx >= warmup_iter) {
            for (int i = 0; i < cls_times.size(); ++i)
                cls_times_total[i] += cls_times[i];
        }
    }
    if (img_files_list.size() > warmup_iter) {

        infer_time = cls_times_total[0] + cls_times_total[1] + cls_times_total[2];
        std::cout << "average time cost in all: "
            << infer_time / (img_files_list.size() - warmup_iter) << " ms."
            << std::endl;
    }

    std::string presion = "fp32";
    if (config.use_fp16)
        presion = "fp16";
    if (config.benchmark) {
        AutoLogger autolog("Classification", config.use_gpu, config.use_tensorrt,
            config.use_mkldnn, config.cpu_threads, 1,
            "1, 3, 224, 224", presion, cls_times_total,
            img_files_list.size());
        autolog.report();
    }
    return 0;
}

4 模型预测

4.1 测试结果

在这里插入图片描述
图 4.1 输入图像
在这里插入图片描述
图4.2 预测结果

4.2 与python预测结果对比

python deploy\python\predict_cls.py -c .\deploy\configs\PULC\vehicle_attribute\inference_vehicle_attribute.yaml -o Global.pretrained_model=output/PPLCNet_x1_0/best_model

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1312803.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

邮件群发平台推荐:提升邮件营销效果的关键

邮件群发平台是外贸、跨境电商、出海企业常用的一种营销方式。他以低成本、高投资回报比获得了众多企业的青睐。现市场上有各种功能各异、价格不同的邮件群发平台&#xff0c;企业在抉择的时候也会犯难。所以&#xff0c;邮件群发平台哪个好用呢&#xff1f;Zoho Campaigns就是…

Vue3-11- 【v-for】循环数组

v-for的基本介绍 v-for 是一个指令&#xff0c; 它是用来在 html 模板中实现循环的。它可以循环 普通的数组、也可以直接循环一个范围值&#xff0c;也可以循环对象的每个属性。v-for 的语法介绍 <div v-for"(item,index) in arrayName" : key"index"…

lua安装

lua安装 1.Lua介绍 特点&#xff1a;轻量、小巧。C语言开发。开源。 设计的目的&#xff1a;嵌入到应用程序当中&#xff0c;提供灵活的扩展和定制化的功能。 luanginx&#xff0c;luaredis。 2.windows安装lua windows上安装lua&#xff1a; 检查机器上是否有lua C:\U…

scala笔记

函数字面量 字面量包括整形字面量、浮点数子面量、布尔型字面量、字符字面量、字符串字面量、符号字面量、函数字面量和元组字面量 除了函数字面量我们比较陌生以外&#xff0c;其他几种字面量都很容易理解 val counter: Int > Int {(value) > value 1}匿名函数 val…

UE4/UE5 日志插件(基于spdlog)

1 解决问题 对于高频日志序列化到本地的需求&#xff0c;spdlog肯定完美满足。 源码地址&#xff1a;https://github.com/gabime/spdlog 博主下载的版本为 spdlog-1.12.0&#xff0c;各位大佬可以根绝自己爱好选择。 2 过程介绍 大概目录&#xff1a; SpdlogLibC目录下是对…

牛客网SQL训练3—SQL必知必会

文章目录 一、检索数据二、排序检索数据三、过滤数据四、高级数据过滤五、用通配符进行过滤六、创建计算字段七、使用函数处理数据八、汇总数据九、分组数据十、使用子查询十一、联结表十二、创建高级联结十三、组合查询 一、检索数据 【题目1&#xff1a;从 Customers 表中检…

常见的Linux基本指令

目录 什么是Linux&#xff1f; Xshell如何远程控制云服务器 Xshell远程连接云服务器 Linux基本指令 用户管理指令 pwd指令 touch指令 mkdir指令 ls指令 cd指令 rm指令 man命令 cp指令 mv指令 cat指令 head指令 ​编辑 tail指令 ​编辑echo指令 find命令 gr…

【教程】Autojs脚本实现暂停和超时重启功能的思路和示例代码

转载请注明出处&#xff1a;小锋学长生活大爆炸[xfxuezhang.cn] 背景介绍 autojs本身不支持暂停脚本&#xff0c;现有网上大部分最直接的做法就是在每条语句后面添加检查是否暂停。当脚本功能和代码量非常打的时候&#xff0c;每一条语句后面都加检测&#xff0c;未免不太现实。…

【SpringBoot】Starter的使用与案例讲解

&#x1f389;&#x1f389;欢迎来到我的CSDN主页&#xff01;&#x1f389;&#x1f389; &#x1f3c5;我是Java方文山&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; &#x1f31f;推荐给大家我的专栏《SpringBoot》。&#x1f3af;&#x1f3af; &…

Linux主机自动注册NPS客户端(脚本化)

参考官方对API使用方法的定义&#xff1a;https://ehang-io.github.io/nps/#/ 1、首先必须要在配置文件中开启 auth_key 并配置一个合适的密钥 2、修改脚本中的可变量参数&#xff0c;以适配自己的环境 #!/bin/bash # 脚本使用说明&#xff1a;# 脚本名称&#xff1a;npc_cr…

I Doc View在线文档预览系统cms.json存在RCE漏洞

文章目录 产品简介漏洞概述指纹识别漏洞利用修复建议 产品简介 i Doc View是一个在线文档解析应用&#xff0c;旨在提供便捷的文件查看和编辑服务。 漏洞概述 iDocView是一个在线文档I Doc View在线文档预览系统cmd.json 处存在命令执行漏洞&#xff0c;攻击者可通过此漏洞获…

【产品经理】需求池和版本树

在这个人人都是产品经理的时代&#xff0c;每位入行的产品人进阶速度与到达高度各有不同。本文作者结合自身三年产品行业的经历&#xff0c;根据案例拆解产品行业的极简研发过程、需求池、版本树、产品自我优化等相关具体方法论。 一、产品研发的极简过程 1. 产品概述 产品就…

第二证券:股票买卖五档什么意思?

股票生意五档是股票生意中的一个常见术语。它是指股票生意盘中最上面的五个报价。股票生意盘是股票商场上的生意报价汇总&#xff0c;其间卖盘代表了其时商场中卖方的报价&#xff0c;买盘代表了其时商场中买方的报价。 股票生意五档通常是指股票生意盘中最上面的五个报价&…

AR眼镜光学方案_AR眼镜整机硬件定制

增强现实(Augmented Reality&#xff0c;AR)技术通过将计算机生成的虚拟物体或其他信息叠加到真实世界中&#xff0c;实现对现实的增强。AR眼镜作为实现AR技术的重要设备&#xff0c;具备虚实结合、实时交互的特点。为了实现透视效果&#xff0c;AR眼镜需要同时显示真实的外部世…

差分法详解

前言 差分算法适用于一些需要对数组和序列进行增减、查询和更新操作的问题&#xff0c;可以提高计算效率和降低存储空间的需求。今天我将带大家学习如何使用差分法&#xff0c;会以例题来带大家使用差分法以增进理解。话不多说让我们开始吧&#xff01; 文章目录 一维差分尾声…

3D Web轻量引擎HOOPS Communicator如何实现对大模型的渲染支持?

除了读取轻松外&#xff0c;HOOPS Communicator对超大模型的支持效果也非常好&#xff0c;它可以支持30GB的包含70万个零件和3.5亿个三角面的Catia装配模型&#xff01; 那么它是如何来实现对大模型的支持呢&#xff1f; 我们将从以下几个方面与大家分享&#xff1a;最低帧率…

算法的时间复杂度是什么?

算法的时间复杂度是什么&#xff1f; 时间复杂度的概念 时间复杂度是用来估算出程序的运行时间的。我们通常会估计算法的操作单元数量&#xff0c;来代表程序消耗的时间。 随着数据规模n的增大&#xff0c;算法执行时间的增长率和f(n)的增长率相同&#xff0c;称作算法的渐近…

线性回归在数据库中的应用

简介 今天看到微信群有人问&#xff0c;如何知道数据库一年的磁盘增量&#xff1f;如果没有研究过统计学&#xff0c;IT人员对于这个问题就只能靠经验了去断定了。没经验的往往都是回复扩容越大越好。当然未来的事情我们是无法预料的。本博主就通过简单的线性回归做一个计算&am…

12. IO

1.File类 • File 类代表与平台无关的文件和目录。 • File 能新建、删除、重命名文件和目录&#xff0c;但 File 不能访问文件内容本身。如果需要访问文件内容本身&#xff0c;则需要使用输入/输出流。 1).File的常用方法 在这里插入图片描述 2).遍历给定目录所有文件 …

QT-坦克大战游戏

QT-坦克大战游戏 一、演示效果二、关键程序三、下载链接 一、演示效果 二、关键程序 #include "score.h" Score::Score(){health30; maxHealthhealth;QLabel *label1 new QLabel(this);label1->setFrameStyle(QFrame::Plain | QFrame::Box);label1->setStyle…