Как генерировать объекты в std :: vector и без копирования? - PullRequest
8 голосов
/ 05 июля 2019

Есть ученический класс

class Student
{
public:
    inline static int current_id_max = 0;
    int id = 0;
    string name;
public:
    Student()
    {
        id = (++current_id_max);
        cout << "Student constructor\n";
    }
    Student(const string& _name)
    {
        name = _name;
        id = (++current_id_max);
        cout << "Student constructor: " << _name << endl;
    }
    Student(const Student& o)
    {
        name = o.name;
        id = (++current_id_max);
        cout << "Student constructor copy: " << name << endl;
    }
    ~Student() { cout << "Student destructor: " << name << endl; }
};

Я хочу создать 5 учеников с параметрами в векторе,

std::vector<Student> school = 
    { Student("Tom"), Student("Mike"),Student("Zhang"), Student("Wang"), Student("Li")};

Будет 5 Student constructor: name и 5 Student constructor copy: name.

Что я могу сделать, чтобы избежать бесполезного копирования?

1 Ответ

10 голосов
/ 05 июля 2019

Я предлагаю следующее:

  1. Создайте пустой вектор.
  2. Зарезервируйте количество элементов, которое вы хотели бы иметь в векторе.
  3. Использованиеstd::vector::emplace_back чтобы добавить к нему элементы.

Полный пример:

#include <iostream>
#include <vector>
#include <string>

class Student
{
   public:
      inline static int current_id_max = 0;
      int id = 0;
      std::string name;
   public:
      Student()
      {
         id = (++current_id_max);
         std::cout << "Student constructor\n";
      }
      Student(const std::string& _name)
      {
         name = _name;
         id = (++current_id_max);
         std::cout << "Student constructor: " << _name << std::endl;
      }
      Student(const Student& o)
      {
         name = o.name;
         id = (++current_id_max);
         std::cout << "Student constructor copy: " << name << std::endl;
      }
      ~Student() { std::cout << "Student destructor: " << name << std::endl; }
};

int main()
{
   std::vector<Student> school;
   school.reserve(5);
   school.emplace_back("Tom");
   school.emplace_back("Mike");
   school.emplace_back("Zhang");
   school.emplace_back("Wang");
   school.emplace_back("Li");
}

Вывод в моем тесте:

Student constructor: Tom
Student constructor: Mike
Student constructor: Zhang
Student constructor: Wang
Student constructor: Li
Student destructor: Tom
Student destructor: Mike
Student destructor: Zhang
Student destructor: Wang
Student destructor: Li
...