• libtorch c++ 搭建分类网络进行训练和预测


    目录

    1. vgg.h

    2. vgg.cpp

    3. 预训练权重

    3.1 保存pytorch版的预训练权重

    3.2 训练

    3.2.1 打印权重参数信息

     3.2.2 权重初始化

     3.2.3 测试预训练权重是否可用

    3.2.4 训练函数

    3.2.5 主函数

    4. 测试

    4.1 主函数 

     4.2 predic函数


    1. vgg.h

    这里以vgg分类网络为例,vgg网络简单:多层卷积提取特征,最大池化下采样,全连接层进行分类,最后接一个sotmax归一化输出。

    1. #ifndef VGG_H
    2. #define VGG_H
    3. #include
    4. #include
    5. #include
    6. // 二维卷积参数配置
    7. // 注意:关键字inline必须与函数定义放在一起才能使函数成为内联函数,仅仅将inline放在函数声明前面不起任何作用。
    8. inline torch::nn::Conv2dOptions conv_options(int64_t in_planes, int64_t out_planes, int64_t kerner_size,
    9. int64_t stride = 1, int64_t padding = 0, bool with_bias = false) {
    10. torch::nn::Conv2dOptions conv_options = torch::nn::Conv2dOptions(in_planes, out_planes, kerner_size);
    11. conv_options.stride(stride);
    12. conv_options.padding(padding);
    13. conv_options.bias(with_bias);
    14. return conv_options;
    15. }
    16. // 最大池化参数配置
    17. inline torch::nn::MaxPool2dOptions maxpool_options(int kernel_size, int stride){
    18. torch::nn::MaxPool2dOptions maxpool_options(kernel_size);
    19. maxpool_options.stride(stride);
    20. return maxpool_options;
    21. }
    22. // 工具函数:生成多个卷积层,作为特征提取层
    23. // 参数cfg: 中间特征层的通道数,batch_norm:是否使用bn
    24. torch::nn::Sequential make_features(std::vector<int> &cfg, bool batch_norm);
    25. // 创建VGG类,必须继承Module类
    26. // vgg网络:卷积层+pooling层+全连接层
    27. class VGGImpl: public torch::nn::Module
    28. {
    29. public:
    30. VGGImpl(std::vector<int>& cfg, int num_classes = 1000, bool batch_norm = false); // 1个构造函数,初始化各个层操作的参数
    31. torch::Tensor forward(torch::Tensor x); // 1个forward函数
    32. private:
    33. torch::nn::Sequential features_{nullptr}; // 卷积层
    34. torch::nn::AdaptiveAvgPool2d avgpool{nullptr}; // pooling
    35. torch::nn::Sequential classifier; // 全连接层
    36. };
    37. TORCH_MODULE(VGG);
    38. #endif // VGG_H

    2. vgg.cpp

    1. #include "vgg.h"
    2. // 工具函数:生成多个卷积层,作为特征提取层
    3. // 参数cfg: 中间特征层的通道数,batch_norm:是否使用bn
    4. torch::nn::Sequential make_features(std::vector<int> &cfg, bool batch_norm){
    5. torch::nn::Sequential features;
    6. int in_channels = 3;
    7. for(auto v : cfg){ // v是通道数
    8. if(v==-1){ // 遇到-1,则接一个池化层
    9. features->push_back(torch::nn::MaxPool2d(maxpool_options(2,2)));
    10. }
    11. else{
    12. // conv2 + bn + relu
    13. auto conv2d = torch::nn::Conv2d(conv_options(in_channels,v,3,1,1)); // k=3,s=1,p=1
    14. features->push_back(conv2d);
    15. if(batch_norm){
    16. features->push_back(torch::nn::BatchNorm2d(torch::nn::BatchNorm2dOptions(v))); // or torch::nn::BatchNorm2d(v)
    17. }
    18. features->push_back(torch::nn::ReLU(torch::nn::ReLUOptions(true))); // relu层
    19. in_channels = v;
    20. }
    21. }
    22. return features;
    23. }
    24. // 初始化私有成员: features_,avgpool,classifier
    25. VGGImpl::VGGImpl(std::vector<int> &cfg, int num_classes, bool batch_norm){
    26. features_ = make_features(cfg,batch_norm); // 1.初始化卷积层(包含了池化层)
    27. avgpool = torch::nn::AdaptiveAvgPool2d(torch::nn::AdaptiveAvgPool2dOptions(7)); // 2.初始化池化层
    28. classifier->push_back(torch::nn::Linear(torch::nn::LinearOptions(512 * 7 * 7, 4096))); // 3.初始化全连接层
    29. classifier->push_back(torch::nn::ReLU(torch::nn::ReLUOptions(true)));
    30. classifier->push_back(torch::nn::Dropout());
    31. classifier->push_back(torch::nn::Linear(torch::nn::LinearOptions(4096, 4096)));
    32. classifier->push_back(torch::nn::ReLU(torch::nn::ReLUOptions(true)));
    33. classifier->push_back(torch::nn::Dropout());
    34. classifier->push_back(torch::nn::Linear(torch::nn::LinearOptions(4096, num_classes))); // 两个linear+relu+dropout, 再加一个linear.
    35. features_ = register_module("features",features_); // module必须注册
    36. classifier = register_module("classifier",classifier);
    37. }
    38. torch::Tensor VGGImpl::forward(torch::Tensor x){
    39. x = features_->forward(x); // 1,先卷积(包含了池化层),注册的module需要通过forward使用
    40. x = avgpool(x); // 2,再池化,直接使用
    41. x = torch::flatten(x,1); // 3,二维变一维
    42. x = classifier->forward(x); // 4,再接全连接层
    43. return torch::log_softmax(x, 1); // 5,最后softmax
    44. }

    3. 预训练权重

    使用预训练权重训练模型,可以更快的收敛。为了使c++训练模型时能够使用pytorch版的预训练权重,c++代码搭建的vgg必须和pytorch一致。感兴趣的可以看下官方的pytorch代码torchvision.models.vgg16_bn

    3.1 保存pytorch版的预训练权重

    注意要使用jit.trace函数。

    1. import torch
    2. from torchvision.models import vgg16, vgg16_bn
    3. # 在c++中搭建一个和pytorch下完全一致的vgg16bn。如果不一致的话其实不影响正常的模型训练和预测,
    4. # 但是影响初始化状态,模型加载从ImageNet数据集训练好的权重以后,训练收敛的速度和收敛后的精度都会好很多。
    5. model = vgg16_bn(pretrained=True)
    6. model = model.to(torch.device("cpu"))
    7. model.eval()
    8. var = torch.ones((1, 3, 224, 224))
    9. # 保存pytorch模型的权重不能直接用torch.save保存模型,这样存下来的模型不能被c++加载。我们利用部署时常用的torch.jit.script模型来保存。
    10. traced_script_module = torch.jit.trace(model, var)
    11. traced_script_module.save("vgg16bn.pt")
    12. # 这样,模型的卷积层,归一化层,线性层的权重就保存到.pt文件中了。

    3.2 训练

    基于VGG类,再打包一次得到Classifier类。

    classification.h 

    1. #ifndef CLASSIFICATION_H
    2. #define CLASSIFICATION_H
    3. #include
    4. #include
    5. #include
    6. class Classifier
    7. {
    8. private:
    9. torch::Device device = torch::Device(torch::kCPU); // 默认使用cpu
    10. VGG vgg = VGG{nullptr}; // 自定义网络对象
    11. public:
    12. Classifier(int gpu_id = 0); // 构造函数,初始化device私用成员
    13. void Initialize(int num_classes, std::string pretrained_path); // 加载预训练权重
    14. void Train(int epochs, int batch_size, float learning_rate, std::string train_val_dir, std::string image_type, std::string save_path);
    15. int Predict(cv::Mat &image); // preprocess + infer + postprocess
    16. void LoadWeight(std::string weight); // 加载权重
    17. };
    18. #endif // CLASSIFICATION_H

    3.2.1 打印权重参数信息

    先看看权重参数的特点。

    1. # pytorch
    2. import torch
    3. from torchvision.models import vgg16, vgg16_bn
    4. model = vgg16_bn(pretrained=True)
    5. for k, v in model.named_parameters():
    6. print(k)
    7. // c++
    8. std::vector<int> cfg_dd = {64, 64, -1, 128, 128, -1, 256, 256, 256, -1, 512, 512, 512, -1, 512, 512, 512, -1};
    9. auto vgg_dd = VGG(cfg_dd,1000,true); // 直接实例化VGG即可
    10. auto dictdd = vgg_dd->named_parameters();
    11. for (auto n = dictdd.begin(); n != dictdd.end(); n++) // 打印出模型每一层(有权重的层,不包括类似激活函数层)的名称
    12. {
    13. std::cout<<(*n).key()<<std::endl;
    14. }

    左边是pytorch打印的信息,右边是c++打印的信息,可以发现c++的少了部分bias 权重,这是因为有些卷积没有设置bias(可以设置)。初始化的时候,根据权重参数名称初始化右边对应的值。

     3.2.2 权重初始化

    利用pytorch版的预训练权重(前面有保存下来),初始化c++网络权重参数。

    classification.cpp

    1. // 使用预训练权重,初始化自定义的模型权重
    2. void Classifier::Initialize(int _num_classes, std::string _pretrained_path){
    3. std::vector<int> cfg_d = {64, 64, -1, 128, 128, -1, 256, 256, 256, -1, 512, 512, 512, -1, 512, 512, 512, -1};
    4. auto net_pretrained = VGG(cfg_d,1000,true); // 注意这里是1000个类别,实例化预训练网络,为了取出权重
    5. vgg = VGG(cfg_d,_num_classes,true); // 注意这里的类别个数
    6. torch::load(net_pretrained, _pretrained_path); // 预训练的网络载入预训练的权重。
    7. torch::OrderedDict pretrained_dict = net_pretrained->named_parameters();
    8. torch::OrderedDict model_dict = vgg->named_parameters();
    9. // 将训练的权重值提取出来,其中后面的分类层参数值丢弃不用。
    10. for (auto n = pretrained_dict.begin(); n != pretrained_dict.end(); n++)
    11. {
    12. if (strstr((*n).key().data(), "classifier")) { // 不使用分类层的权重
    13. continue;
    14. }
    15. model_dict[(*n).key()] = (*n).value();
    16. }
    17. torch::autograd::GradMode::set_enabled(false); // make parameters copying possible
    18. auto new_params = model_dict; // implement this
    19. auto params = vgg->named_parameters(true /*recurse*/);
    20. auto buffers = vgg->named_buffers(true /*recurse*/);
    21. for (auto& val : new_params) { // 功能:将屏蔽分类层后的新权重,复制到自定义的vgg网络模型权重中。
    22. auto name = val.key();
    23. auto* t = params.find(name); // 查看自定义网络中是否有该权重参数名
    24. if (t != nullptr) {
    25. t->copy_(val.value()); // 有,则将预训练得到的val复制到自定义网络参数中。
    26. }
    27. else {
    28. t = buffers.find(name); // 没有,则看看buffer里面有没有,都没有则跳过。
    29. if (t != nullptr) {
    30. t->copy_(val.value());
    31. }
    32. }
    33. }
    34. torch::autograd::GradMode::set_enabled(true);
    35. try
    36. {
    37. vgg->to(device);
    38. }
    39. catch (const std::exception&e)
    40. {
    41. std::cout << e.what() << std::endl;
    42. }
    43. return;
    44. }

     3.2.3 测试预训练权重是否可用

    1. #include
    2. int main(int argc, char *argv[])
    3. {
    4. //2,加载预训练权重,测试权重是否可用。
    5. std::vector<int> cfg_16bn = { 64, 64, -1, 128, 128, -1, 256, 256, 256, -1, 512, 512, 512, -1, 512, 512, 512, -1 };
    6. auto vgg16bn = VGG(cfg_16bn, 1000, true);
    7. torch::load(vgg16bn, "E:\\code\\c++\\LibtorchTutorials-main\\vgg16bn.pt");
    8. vgg16bn->to(torch::Device(torch::kCUDA));
    9. }

    最后一句报错:PyTorch is not linked with support for cuda devices

    解决办法:属性->链接器->命令行:

    1. libtorch 1.5: -INCLUDE:THCudaCharTensor_zero
    2. libtorch 1.6/1.7/1.9/1.9.1/1.10/1.1:-INCLUDE:?warp_size@cuda@at@@YAHXZ 
    3. libtorch 1.8.1: -INCLUDE:?wait@Future@ivalue@c10@@QEAAXXZ
    4. 或者:-INCLUDE:?mutate@OptOutMutator@cuda@fuser@jit@torch@@UEAAPEAVStatement@2345@PEAVForLoop@kir@2345@@Z

    3.2.4 训练函数

    这里有个骚操作,前8个epoch保留前面的特征提取层参数不变(即使用预训练权重值),只更新分类层参数。

    1. void Classifier::Train(int num_epochs, int batch_size, float learning_rate, std::string train_val_dir, std::string image_type, std::string save_path){
    2. std::string path_train = train_val_dir+ "\\train"; // 数据路径
    3. std::string path_val = train_val_dir + "\\val";
    4. auto custom_dataset_train = dataSetClc(path_train, image_type).map(torch::data::transforms::Stack<>()); // dataset
    5. auto custom_dataset_val = dataSetClc(path_val, image_type).map(torch::data::transforms::Stack<>());
    6. auto data_loader_train = torch::data::make_data_loader(std::move(custom_dataset_train), batch_size); // dataloader
    7. auto data_loader_val = torch::data::make_data_loader(std::move(custom_dataset_val), batch_size);
    8. float loss_train = 0; float loss_val = 0; // 累加当前epoch内所有的loss,求平均loss
    9. float acc_train = 0.0; float acc_val = 0.0; float best_acc = 0.0; // 累加准去率,和最佳准确率
    10. for (size_t epoch = 1; epoch <= num_epochs; ++epoch) { // epoch
    11. size_t batch_index_train = 0; // 记录当前是第几个batch
    12. size_t batch_index_val = 0;
    13. if (epoch == int(num_epochs / 2)) { learning_rate /= 10; } // 每两个epoch下降一次学习率
    14. torch::optim::Adam optimizer(vgg->parameters(), learning_rate); // Learning Rate
    15. if (epoch < int(num_epochs / 8)) // 前8个epoch只更新分类层参数
    16. {
    17. for (auto mm : vgg->named_parameters())
    18. {
    19. if (strstr(mm.key().data(), "classifier")) // 只更新分类层的参数
    20. {
    21. mm.value().set_requires_grad(true);
    22. }
    23. else
    24. {
    25. mm.value().set_requires_grad(false);
    26. }
    27. }
    28. }
    29. else { // 后面epoch次更新全部参数
    30. for (auto mm : vgg->named_parameters())
    31. {
    32. mm.value().set_requires_grad(true);
    33. }
    34. }
    35. // Iterate data loader to yield batches from the dataset
    36. for (auto& batch : *data_loader_train) {
    37. auto data = batch.data; // b,c,h,w
    38. auto target = batch.target.squeeze(); // b,1
    39. data = data.to(torch::kF32).to(device).div(255.0); // data: kf32,cuda,/255
    40. target = target.to(torch::kInt64).to(device); // target: kint64, cuda
    41. optimizer.zero_grad();
    42. // Execute the model
    43. torch::Tensor prediction = vgg->forward(data);
    44. //cout << prediction << endl;
    45. auto acc = prediction.argmax(1).eq(target).sum();
    46. acc_train += acc.template item<float>() / batch_size;
    47. // Compute loss value
    48. torch::Tensor loss = torch::nll_loss(prediction, target);
    49. // Compute gradients
    50. loss.backward();
    51. // Update the parameters
    52. optimizer.step();
    53. loss_train += loss.item<float>(); // 累加loss
    54. batch_index_train++;
    55. std::cout << "Epoch: " << epoch << " |Train Loss: " << loss.item<float>() << " |Train Acc:" << acc_train / batch_index_train << "\r";
    56. }
    57. std::cout << std::endl;
    58. //validation part
    59. vgg->eval();
    60. for (auto& batch : *data_loader_val) {
    61. auto data = batch.data;
    62. auto target = batch.target.squeeze();
    63. data = data.to(torch::kF32).to(device).div(255.0);
    64. target = target.to(torch::kInt64).to(device);
    65. torch::Tensor prediction = vgg->forward(data);
    66. // Compute loss value
    67. torch::Tensor loss = torch::nll_loss(prediction, target);
    68. auto acc = prediction.argmax(1).eq(target).sum(); // val准确率计算方法
    69. acc_val += acc.template item<float>() / batch_size;
    70. loss_val += loss.item<float>(); // 累加loss
    71. batch_index_val++;
    72. std::cout << "Epoch: " << epoch << " |Val Loss: " << loss_val / batch_index_val << " |Valid Acc:" << acc_val / batch_index_val << "\r";
    73. }
    74. std::cout << std::endl;
    75. if (acc_val > best_acc) {
    76. torch::save(vgg, save_path);
    77. best_acc = acc_val;
    78. }
    79. loss_train = 0; loss_val = 0; acc_train = 0; acc_val = 0; batch_index_train = 0; batch_index_val = 0;
    80. }
    81. }

    其中 dataset-dataSetClc的定义与上一篇博客一样都是图像分类数据集

    3.2.5 主函数

    1. #include
    2. int main(int argc, char *argv[])
    3. {
    4. //2,使用预训练权重,进行推理
    5. std::string vgg_weight_path = "E:\\code\\c++\\LibtorchTutorials-main\\vgg16bn.pt";
    6. std::string train_val_dir = "F:\\zxq\\data\\hymenoptera_data";
    7. Classifier classifier(0); // gpu id: 0
    8. classifier.Initialize(2, vgg_weight_path); // 使用预训练权重初始化权重,
    9. classifier.Train(300, 4, 0.0003, train_val_dir, ".jpg", "classifer.pt");
    10. }

    验证集的准确率是0.91。

    换成猫狗大战数据集(分出4000张训练),训练和测试集loss收敛的都很快。

    4. 测试

    测试的时候发现一个问题,在2080ti训练的模型,在3080显卡下预测会报错:

    Microsoft C++ 异常: c10::Error,位于内存位置 0x000000301E6F3BF0 处。

    可以在高端显卡训练,低端显卡部署。

    4.1 主函数 

    1. // 3, infer
    2. Classifier classifier(1); // gpu id: 0
    3. classifier.Initialize(2, "E:\\code\\c++\\LibtorchTutorials-main\\vgg16bn.pt");
    4. classifier.LoadWeight("E:\\code\\c++\\LibtorchLearning\\classifer.pt");
    5. std::string train_val_dir = "F:\\zxq\\data\\custom\\dog";
    6. cv::Mat image = cv::imread(train_val_dir+"\\dog.2344.jpg");
    7. int cls_id = classifier.Predict(image); // preprocess + forward
    8. std::cout << "cls_id: " << cls_id << std::endl;

     

    换成cat图片。

     4.2 predic函数

    1. int Classifier::Predict(cv::Mat& image){
    2. // preprocess: resize, to_tensor, cuda, kf32, /255
    3. cv::resize(image, image, cv::Size(448, 448));
    4. torch::Tensor img_tensor = torch::from_blob(image.data, { image.rows, image.cols, 3 }, torch::kByte).permute({ 2, 0, 1 }); // c,h,w
    5. img_tensor = img_tensor.to(device).unsqueeze(0).to(torch::kF32).div(255.0);
    6. auto prediction = vgg->forward(img_tensor); // raw output
    7. prediction = torch::softmax(prediction,1);
    8. auto class_id = prediction.argmax(1);
    9. std::cout<
    10. int ans = int(class_id.item().toInt());
    11. float prob = prediction[0][ans].item().toFloat();
    12. return ans;
    13. }
    14. void Classifier::LoadWeight(std::string weight){
    15. torch::load(vgg,weight); // 载入训练好的权重
    16. vgg->eval();
    17. return;
    18. }

    参考:LibtorchTutorials/lesson5-TrainingVGG at main · AllentDan/LibtorchTutorials · GitHub

  • 相关阅读:
    南大通用GBase8s 常用SQL语句(238)
    Mysql主从复制、读写分离
    ElasticSearch集群的搭建
    数据结构02 队列及其应用【C++实现】
    HarmonyOS应用开发-视频播放器与弹窗
    基于高股息高分红优化的量化选股模型
    Gmail 将停止支持基本 HTML 视图
    OS10 - 计数器和警报(1)
    Pytorch 多卡并行(3)—— 使用 DDP 加速 minGPT 训练
    React Native优质开源项目精选
  • 原文地址:https://blog.csdn.net/jizhidexiaoming/article/details/127183182