Вектор инициализации структур - PullRequest
63 голосов
/ 09 ноября 2011

Я хочу знать, как я могу добавить значения к своему вектору структур, используя метод push_back

struct subject
{
  string name;
  int marks;
  int credits;
};


vector<subject> sub;

Итак, как теперь я могу добавить к нему элементы?

У меня есть функциякоторый инициализирует имя строки (имя субъекта)

void setName(string s1, string s2, ...... string s6)
{
   // how can i set name too sub[0].name= "english", sub[1].name = "math" etc

  sub[0].name = s1 // gives segmentation fault; so how do I use push_back method?

  sub.name.push_back(s1);
  sub.name.push_back(s2);
  sub.name.push_back(s3);
  sub.name.push_back(s4);

  sub.name.push_back(s6);

}

вызов функции

setName("english", "math", "physics" ... "economics");

Ответы [ 5 ]

80 голосов
/ 09 ноября 2011

Создайте вектор, элемент push_back, затем измените его следующим образом:

struct subject {
    string name;
    int marks;
    int credits;
};


int main() {
    vector<subject> sub;

    //Push back new subject created with default constructor.
    sub.push_back(subject());

    //Vector now has 1 element @ index 0, so modify it.
    sub[0].name = "english";

    //Add a new element if you want another:
    sub.push_back(subject());

    //Modify its name and marks.
    sub[1].name = "math";
    sub[1].marks = 90;
}

Вы не можете получить доступ к вектору с помощью [#], пока в этом индексе не будет элемента. Этот пример заполняет [#], а затем изменяет его.

44 голосов
/ 09 ноября 2011

Если вы хотите использовать новый текущий стандарт, вы можете сделать это:

sub.emplace_back ("Math", 70, 0);

или

sub.push_back ({"Math", 70, 0});

Они не требуют построения по умолчанию subject.

8 голосов
/ 09 ноября 2011

Вы не можете получить доступ к элементам пустого вектора по нижнему индексу.
Всегда проверяйте, что вектор не пуст и индекс действителен при использовании оператора [] для std::vector.
[] нетдобавить элементы, если их нет, но это вызывает неопределенное поведение , если индекс недействителен.

Вы должны создать временный объект вашей структуры, заполнить его, а затем добавить его в вектор, используя vector::push_back()

subject subObj;
subObj.name = s1;
sub.push_back(subObj);
5 голосов
/ 29 апреля 2018

Вы также можете использовать агрегатную инициализацию из списка фигурной инициализации для подобных ситуаций.

#include <vector>
using namespace std;

struct subject {
    string name;
    int    marks;
    int    credits;
};

int main() {
    vector<subject> sub {
      {"english", 10, 0},
      {"math"   , 20, 5}
    };
}

Иногда, однако, члены структуры могут быть не такими простыми, поэтому вы должны предоставить компиляторурука в выводе его типов.

Таким образом, распространяясь на выше.

#include <vector>
using namespace std;

struct assessment {
    int   points;
    int   total;
    float percentage;
};

struct subject {
    string name;
    int    marks;
    int    credits;
    vector<assessment> assessments;
};

int main() {
    vector<subject> sub {
      {"english", 10, 0, {
                             assessment{1,3,0.33f},
                             assessment{2,3,0.66f},
                             assessment{3,3,1.00f}
                         }},
      {"math"   , 20, 5, {
                             assessment{2,4,0.50f}
                         }}
    };
}

Без assessment в инициализаторе с фигурными скобками компилятор потерпит неудачу при попытке определить тип.

Выше было скомпилировано и протестировано с gcc в c ++ 17.Однако он должен работать с C ++ 11 и далее.В c ++ 20 мы можем увидеть синтаксис указателя, я надеюсь, что он позволит использовать следующий источник

  {"english", 10, 0, .assessments{
                         {1,3,0.33f},
                         {2,3,0.66f},
                         {3,3,1.00f}
                     }},

: http://en.cppreference.com/w/cpp/language/aggregate_initialization

2 голосов
/ 14 июля 2018

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

Но я нашел новый способ сделать это, используя default_structure_element, как показано ниже ...

#include <bits/stdc++.h>
typedef long long ll;
using namespace std;

typedef struct subject {
  string name;
  int marks;
  int credits;
}subject;

int main(){
  subject default_subject;
  default_subject.name="NONE";
  default_subject.marks = 0;
  default_subject.credits = 0;

  vector <subject> sub(10,default_subject);         // default_subject to initialize

  //to check is it initialised
  for(ll i=0;i<sub.size();i++) {
    cout << sub[i].name << " " << sub[i].marks << " " << sub[i].credits << endl;
  } 
}

Тогда я думаю, что хорошо инициализировать вектор структуры, не так ли?

...