Форум программистов, компьютерный форум, киберфорум
С++ для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
 
Рейтинг 4.84/25: Рейтинг темы: голосов - 25, средняя оценка - 4.84
5 / 5 / 0
Регистрация: 16.05.2012
Сообщений: 93

Класс со значением полей "по умолчанию"

11.11.2013, 10:35. Показов 5291. Ответов 26
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Здравствуйте. Написал программу, но не могу сделать так, чтобы класс принимал значение полей по умолчанию - закомментированный вариант вызывает ошибку. Пожалуйста помогите исправить. И, если не сложно, объясните разницу между test,test2 и test3, а то в первом 2 вызова деструктора, во втором - один, а в третьем случае деструктор не вызывается


C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
#include <iostream>
using namespace std;
//Класс Студент
class TStudent
{
private:
    char *LastName; 
    char *FirstName; 
    char *Patronimyc;
    int Born;
    char *Group; 
 
public:
    TStudent();//char *LastName, char *FirstName, char *Patronimyc, char *Group, int Born); //Конструктор класса по умолчанию *просто выделим память под все поля
    
    ~TStudent(); //Деструтор класса
    TStudent (TStudent& Student); //Конструктор копирования по умолчанию
    TStudent operator= (TStudent& Student); //Оператор присвоения
    TStudent test(TStudent s);
    void SetLastName (char* sLastName);  //Присваивание Фамилии
    void SetFirstName (char* sFirstName);//Присваивание Имени
    void SetPatronimyc (char* sPatronimyc);//Приваивание отчества
    void SetGroup(char* sGroup);
 
    char* GetLastName(void); //Получение указателя на Фамилию
    char* GetFirstName(void);//Получение указателя на имя
    char* GetPatronimyc(void); //Получение указателя на Отчество
    int& BornDate(void); //Изменение даты рождения
    char* GetGroup(void);
    //Дружественная функция перезагрузки вывода в стандартный поток вывода (STDOUT)
    friend ostream& operator<< (ostream& out, TStudent Student)
    {
        out<<"Фамилия : "<< Student.LastName<< endl;
        out<<"Имя : "<< Student.FirstName<< endl;
        out<<"Отчество : "<< Student.Patronimyc<< endl;
         out<<"Группа : "<< Student.Group<< endl;
        out<<"Год рождения : "<< Student.Born<< endl<< endl;
        
        return out;
    }
 
};
//Конструктор по умолчанию *просто выделим память под все поля
TStudent::TStudent()//char *LastName, char *FirstName, char *Patronimyc, char *Group, int Born)
{     cout << "Вызов конструктора" << endl;
    //Выделим памяти для Фамилии, Имени и Отчества на которые указывают их указатели char*
    LastName = new char[];
       FirstName = new char[];
        Patronimyc = new char[];
     Group = new char[];
   };
//Конструктор копирования по-умолчанию
TStudent::TStudent (TStudent& Student)
{
    //В конструкторе копирования, так же выделим память на которые указывают указатели
    LastName = new char[]; 
    FirstName = new char[];
    Patronimyc = new char[];
     Group = new char[];
   
    //И скопируем в них значения из объекта класса на базе которого мы создаем объект путем копирования
   LastName=strdup(Student.LastName);
   FirstName =strdup(Student.FirstName);
    Patronimyc =strdup(Student.Patronimyc);
     Group = strdup(Student.Group);
     Born = Student.Born;
}
//Перезагрузка оператора присвоения 
TStudent TStudent::operator= (TStudent& Student) //Оператор присвоения
{
     LastName=strdup(Student.LastName);
   FirstName =strdup(Student.FirstName);
    Patronimyc =strdup(Student.Patronimyc);
     Group = strdup(Student.Group);
     Born = Student.Born;
    return *this;
}
 
//Присваивание Фамилии
void TStudent::SetLastName (char* sLastName)
{
    strcpy(LastName,sLastName); 
}
 
//Присваивание Имени
void TStudent::SetFirstName (char* sFirstName)
{
    strcpy(FirstName,sFirstName); 
}
 
//Присваивание отчества
void TStudent::SetPatronimyc (char* sPatronimyc)
{
    strcpy(Patronimyc, sPatronimyc); 
}
 
//Присваивание Группы
void TStudent::SetGroup (char* sGroup)
{
    strcpy(Group, sGroup); 
}
 
 
 
 
//Получение указателя на Фамилию
char* TStudent::GetLastName(void) 
{                                 
    return LastName;        }      
 
//Получение указателя на имя
char* TStudent::GetFirstName(void)  //Аналогично прошлой
{
    return FirstName;
}
 
//Получение указателя на Отчество
char* TStudent::GetPatronimyc(void)  //Так же аналогично
{
    return Patronimyc;
}
 
//Получение указателя на group
char* TStudent::GetGroup(void)  //Так же аналогично
{
    return Group;
}
 
 
 
 
//Возврат ссылки на дату рождения
int& TStudent::BornDate(void) //Возвращаем ссылку на ДатуРождения 
{
    return Born;
}
 
//Деструктор класса
 
TStudent::~TStudent()
{   cout << "Вызов деструктора" << endl;
    delete[] LastName; //Отчистим память которую мы выделили
    delete[] FirstName;//Отчистим память которую мы выделили
    delete[] Patronimyc;//Отчистим память которую мы выделили
    delete[] Group;//Отчистим память которую мы выделили
}
TStudent test(TStudent s){return s;}
TStudent test2 (TStudent &s){return s;}
TStudent& test3 (TStudent &s){return s;}
//Точка входа (главная функция)
void main(void)
{ char f[40], n[40], p[40], g[40];
int b;
    setlocale(LC_ALL,"Russian");
    cout<<"Вход ф-ции main :"<< endl;
    TStudent D1; //Создади объект класса конструктором по умолчанию
    //Заполним его
    cin >> f >> n >> p >> g >> b;
    D1.SetFirstName(n);
    D1.SetLastName(f);
    D1.SetPatronimyc(p);
    D1.SetGroup(g);
    D1.BornDate() = b;
    cout<< D1; //Выведим на экран 
 
    //Создадим объект констуктором копирования
    TStudent D2(D1);
    //Если все было реализованно верно, то указатели D2 не должны быть равны указателям D1
    //Проверим это следующим образом
    cout<<"Проверка конструктора копирования по умолчанию, путем замены первой буквы фамилии на 'F':"<< endl;
        
    D2.GetLastName()[0] ='F';//Изменим первый символ фамилии D1 на 'F'
     cout << D2; //Выведим на экран сначало D1 и D2 (результаты должны быть разными
    test(D1);  // 2 раза вызывается деструктор
    test2(D1);  // 1 раз
    test3(D1);  // деструктор не вызывается
    cout<<"Выход из ф-ции main :"<< endl;
}
0
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
11.11.2013, 10:35
Ответы с готовыми решениями:

Создайте производный класс с дополнительными полями, конструктором по умолчанию и методами чтения/изменения значений этих полей
Помогите! Завтра экзамен,времени нет! Заранее спасибо. 11. На базе заданного класса создайте производный класс с дополнительными полями,...

Как сделать значение по умолчанию для параметра, если оно должно быть значением по умолчанию для типа T?
Как сделать значение по умолчанию для параметра, если оно должно быть значением по умолчанию для типа T? Module Program Sub Method(Of...

Подсветка строк содержащих одинаковые значения полей или подсветка полей с одинаковым значением в таблице
Подскажите, возможна ли реализация в форме такой функции: При наведении курсора на поле высвечиваются по всей таблице поля имеющие такое же...

26
 Аватар для Extrim88
51 / 51 / 21
Регистрация: 08.02.2013
Сообщений: 189
11.11.2013, 13:39
Студворк — интернет-сервис помощи студентам
Цитата Сообщение от Tulosba Посмотреть сообщение
Совершенно не обязательно. В теле конструктора для членов можно добавлять this->, а в инициализаторе конструктора можно и без этого обойтись.
Хорошо, такой пример, у меня есть класс с элементами данных
C++
1
double length; double width;
Определяем конструктор:
C++
1
2
3
4
5
Class_Name::Class_Name(double length, double length)
{
length = length;
width = width;
}
Не возникнет ли в таком случае неопределенности при инициализации элементов данных? Параметры конструктора не являются переменными-членами класса, они представляют значения, которые присваиваются переменным-членам.

Добавлено через 36 секунд
Class_Name::Class_Name(double length, double width)
{
length = length;
width = width;
}

Добавлено через 1 минуту
такой вариант, в первом варианте второй параметр неправильно назвал

Добавлено через 6 минут
это я не от "балды" придумал и совершенно ни к кому не придираюсь, просто сейчас специально открыл хорошую книжку С.Прата "Язык С++. Лекции и упражнения", читал когда-то, и там вспомнилась такая выноска касательно именования параметров в конструкторе
Цитата Сообщение от Extrim88 Посмотреть сообщение
Параметры конструктора не являются переменными-членами класса, они представляют значения, которые присваиваются переменным-членам.
Собственно цитата оттуда
0
:)
Эксперт С++
4773 / 3267 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
11.11.2013, 13:51
Цитата Сообщение от Extrim88 Посмотреть сообщение
Не возникнет ли в таком случае неопределенности при инициализации элементов данных?
Неопределенности нет. В данном случае поля не будут проинициализированы вовсе. Нужно писать:
C++
1
this->length = length;
Или использовать инициализатор конструктора:
C++
1
Class_Name::Class_Name(double length, double width) : length(length), width(width) {}
1
 Аватар для Extrim88
51 / 51 / 21
Регистрация: 08.02.2013
Сообщений: 189
11.11.2013, 13:52
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
#include <iostream>
using namespace std;
//Класс Студент
class TStudent
{
private:
    char *LastName; 
    char *FirstName; 
    char *Patronimyc;
    int Born;
    char *Group; 
 
public:
    TStudent(char* last = "Petr", char* first = "Petrov", char* patr = "Petrovich", char* gr = "abc123", int bo = 25); *Group, int Born); //Конструктор класса по умолчанию *просто выделим память под все поля
    
    ~TStudent(); //Деструктор класса
    TStudent (const TStudent& Student); //Конструктор копирования по умолчанию
    TStudent operator= (const TStudent& Student); //Оператор присвоения
    TStudent test(const TStudent& s);
    void SetLastName (char* sLastName);  //Присваивание Фамилии
    void SetFirstName (char* sFirstName);//Присваивание Имени
    void SetPatronimyc (char* sPatronimyc);//Приваивание отчества
    void SetGroup(char* sGroup);
 
    char* GetLastName(); //Получение указателя на Фамилию
    char* GetFirstName();//Получение указателя на имя
    char* GetPatronimyc(); //Получение указателя на Отчество
    int& BornDate(); //Изменение даты рождения
    char* GetGroup();
    //Дружественная функция перезагрузки вывода в стандартный поток вывода (STDOUT)
    friend ostream& operator<< (ostream& out, const TStudent& Student)
    {
        out<<"Фамилия : "<< Student.LastName<< endl;
        out<<"Имя : "<< Student.FirstName<< endl;
        out<<"Отчество : "<< Student.Patronimyc<< endl;
         out<<"Группа : "<< Student.Group<< endl;
        out<<"Год рождения : "<< Student.Born<< endl<< endl;
        
        return out;
    }
 
};
//Конструктор по умолчанию *просто выделим память под все поля
TStudent::TStudent(char* last, char* first, char* patr, char* gr, int bo)
{     cout << "Вызов конструктора" << endl;
    //Выделим памяти для Фамилии, Имени и Отчества на которые указывают их указатели char*
    LastName = new char[strlen(last) + 1];
       FirstName = new char[strlen(first) + 1];
        Patronimyc = new char[strlen(patr) + 1];
     Group = new char[strlen(gr) + 1];
 
 
   };
//Конструктор копирования по-умолчанию
TStudent::TStudent (TStudent& Student)
{
    //В конструкторе копирования, так же выделим память на которые указывают указатели
    LastName = new char[]; 
    FirstName = new char[];
    Patronimyc = new char[];
     Group = new char[];
   
    //И скопируем в них значения из объекта класса на базе которого мы создаем объект путем копирования
   LastName=strdup(Student.LastName);
   FirstName =strdup(Student.FirstName);
    Patronimyc =strdup(Student.Patronimyc);
     Group = strdup(Student.Group);
     Born = Student.Born;
}
//Перезагрузка оператора присвоения 
TStudent TStudent::operator= (TStudent& Student) //Оператор присвоения
{
     LastName=strdup(Student.LastName);
   FirstName =strdup(Student.FirstName);
    Patronimyc =strdup(Student.Patronimyc);
     Group = strdup(Student.Group);
     Born = Student.Born;
    return *this;
}
 
//Присваивание Фамилии
void TStudent::SetLastName (char* sLastName)
{
    strcpy(LastName,sLastName); 
}
 
//Присваивание Имени
void TStudent::SetFirstName (char* sFirstName)
{
    strcpy(FirstName,sFirstName); 
}
 
//Присваивание отчества
void TStudent::SetPatronimyc (char* sPatronimyc)
{
    strcpy(Patronimyc, sPatronimyc); 
}
 
//Присваивание Группы
void TStudent::SetGroup (char* sGroup)
{
    strcpy(Group, sGroup); 
}
 
 
 
 
//Получение указателя на Фамилию
char* TStudent::GetLastName(void) 
{                                 
    return LastName;        }      
 
//Получение указателя на имя
char* TStudent::GetFirstName(void)  //Аналогично прошлой
{
    return FirstName;
}
 
//Получение указателя на Отчество
char* TStudent::GetPatronimyc(void)  //Так же аналогично
{
    return Patronimyc;
}
 
//Получение указателя на group
char* TStudent::GetGroup(void)  //Так же аналогично
{
    return Group;
}
 
 
 
 
//Возврат ссылки на дату рождения
int& TStudent::BornDate(void) //Возвращаем ссылку на ДатуРождения 
{
    return Born;
}
 
//Деструктор класса
 
TStudent::~TStudent()
{   cout << "Вызов деструктора" << endl;
    delete[] LastName; //Отчистим память которую мы выделили
    delete[] FirstName;//Отчистим память которую мы выделили
    delete[] Patronimyc;//Отчистим память которую мы выделили
    delete[] Group;//Отчистим память которую мы выделили
}
TStudent test(TStudent s){return s;}
TStudent test2 (TStudent &s){return s;}
TStudent& test3 (TStudent &s){return s;}
//Точка входа (главная функция)
void main(void)
{ char f[40], n[40], p[40], g[40];
int b;
    setlocale(LC_ALL,"Russian");
    cout<<"Вход ф-ции main :"<< endl;
    TStudent D1; //Создади объект класса конструктором по умолчанию
    //Заполним его
    cin >> f >> n >> p >> g >> b;
    D1.SetFirstName(n);
    D1.SetLastName(f);
    D1.SetPatronimyc(p);
    D1.SetGroup(g);
    D1.BornDate() = b;
    cout<< D1; //Выведим на экран 
 
    //Создадим объект констуктором копирования
    TStudent D2(D1);
    //Если все было реализованно верно, то указатели D2 не должны быть равны указателям D1
    //Проверим это следующим образом
    cout<<"Проверка конструктора копирования по умолчанию, путем замены первой буквы фамилии на 'F':"<< endl;
        
    D2.GetLastName()[0] ='F';//Изменим первый символ фамилии D1 на 'F'
     cout << D2; //Выведим на экран сначало D1 и D2 (результаты должны быть разными
    test(D1);  // 2 раза вызывается деструктор
    test2(D1);  // 1 раз
    test3(D1);  // деструктор не вызывается
    cout<<"Выход из ф-ции main :"<< endl;
}
1
5 / 5 / 0
Регистрация: 16.05.2012
Сообщений: 93
11.11.2013, 14:03  [ТС]
Возникают ошибки в 56 и 78 строках
0
 Аватар для Extrim88
51 / 51 / 21
Регистрация: 08.02.2013
Сообщений: 189
11.11.2013, 14:53
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
//Конструктор копирования по-умолчанию
TStudent::TStudent (const TStudent& Student)
{
    //В конструкторе копирования, так же выделим память на которые указывают указатели
    LastName = new char[strlen(Student.LastName) + 1]; 
    FirstName = new char[strlen(Student.FirstName) + 1];
    Patronimyc = new char[strlen(Student.Patronimyc) + 1];
     Group = new char[strlen(Student.Group) + 1];
   
    //И скопируем в них значения из объекта класса на базе которого мы создаем объект путем копирования
   strcpy(LastName, Student.LastName);
   strcpy(FirstName, Student.FirstName);
   strcpy(Patronimyc, Student.Patronimyc);
   strcpy(Group, Student.Group);
   Born = Student.Born;
}
C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
//Перегрузка оператора присвоения 
TStudent TStudent::operator= (TStudent& Student) //Оператор присвоения
{
if (this == &Student) //проверка на самоприсваивание
return *this;
 
delete [] LastName;
delete [] FirstName;
delete [] Patronimyc;
delete [] Group;
 
     LastName = new char[strlen(Student.LastName) + 1]; 
    FirstName = new char[strlen(Student.FirstName) + 1];
    Patronimyc = new char[strlen(Student.Patronimyc) + 1];
     Group = new char[strlen(Student.Group) + 1];
strcpy(LastName, Student.LastName);
   strcpy(FirstName, Student.FirstName);
   strcpy(Patronimyc, Student.Patronimyc);
   strcpy(Group, Student.Group);
     Born = Student.Born;
    return *this;
}
Добавлено через 2 минуты
даже еще лучше возвращать объект в операции присваивания по ссылке, а не по значению, т.е.

C++
1
TStudent& TStudent::operator= (TStudent& Student) //Оператор присвоения
соответственно в классе тоже нужно изменить прототип это функции

Добавлено через 3 минуты
Цитата Сообщение от Tulosba Посмотреть сообщение
Неопределенности нет. В данном случае поля не будут проинициализированы вовсе.
Ну а в чем тогда польза такого конструктора, если поля не будут проинициализированы вовсе?

Добавлено через 1 минуту
Адренс Аксидор, еще лучше так будет написать
C++
1
TStudent& TStudent::operator= (const TStudent& Student) //Оператор присвоения
Добавлено через 2 минуты
Адренс Аксидор, будут ошибки еще выкладывайте)у меня IDE под рукой сейчас нет, поэтому пишу прямо здесь
1
:)
Эксперт С++
4773 / 3267 / 497
Регистрация: 19.02.2013
Сообщений: 9,046
11.11.2013, 14:57
Цитата Сообщение от Extrim88 Посмотреть сообщение
Ну а в чем тогда польза такого конструктора, если поля не будут проинициализированы вовсе?
Пользы нет потому что писать так не следует. Это логическая ошибка. Для того чтобы избежать ошибки, можно параметры объявить константными. Тогда присвоить значение самому себе не получится.
1
5 / 5 / 0
Регистрация: 16.05.2012
Сообщений: 93
11.11.2013, 15:05  [ТС]
Спасибо, все заработало!
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
11.11.2013, 15:05
Помогаю со студенческими работами здесь

Select со значением по умолчанию из модели
Гуглил, но гуглится не то, что надо. Хотя проблема типовая и думаю, что сталкивались с ней. Итак, есть машина. У машины есть водитель....

Параметр со значением по умолчанию в функции
Здравствуйте, не знаете как можно сделать в JS параметр с заранее заданным значением по умолчанию Например в С++ это так: void...

Инициализация массива значением по умолчанию
Доброго времени суток уважаемые! Подскажите пожалуйста, как задать значение массива по умолчанию? Желательно, при его создании. ...

Может ли функция быть значением по умолчанию?
День добрый! Правда ли, что нельзя функции назначать как DEFAULT? Например, alter table tv modify colname time default value...

Добавление столбца в таблицу со значением по умолчанию
пишем ALTER TABLE T_001 ADD name_01 bit - всё ок а как сразу написать, чтобы name_01 bit был DEFAULT ((0)), то есть False? запрос...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
27
Ответ Создать тему
Новые блоги и статьи
Дальние перспективы сервера - слоя сети с космологическим дизайном интефейса карты и логики.
Hrethgir 07.04.2026
Дальнейшее ближайшее планирование вывело к размышлениям над дальними перспективами. И вот тут может быть даже будут нужны оценки специалистов, так как в дальних перспективах всё может очень сильно. . .
Горе от ума
kumehtar 07.04.2026
Эта мне ментальная установка, что вот прямо сейчас, мол, мне для полного счастья не хватает (нужное вписать), и когда я этого достигну - тогда и полный кайф. Одна из самых сильных ловушек на пути. . . .
Использование значений реквизитов справочника в документе, с определенными условиями и правами
Maks 07.04.2026
1. Контроль срока действия договора Алгоритм из решения ниже реализован на примере нетипового документа "ЗаявкаНаРаботу", разработанного в конфигурации КА2. Задача: уведомлять пользователя, если. . .
Доступность команды формы по условию
Maks 07.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа "СписаниеМатериалов", разработанного в конфигурации КА2. Задача: сделать доступной кнопку (команда формы "ЗавершитьСписание") при. . .
Уведомление о неверно выбранном значении справочника
Maks 06.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа "НарядПутевка", разработанного в конфигурации КА2. Задача: уведомлять пользователя, если в документе выбран неверный склад. . .
Установка Qt Creator для C и C++: ставим среду, CMake и MinGW без фреймворка Qt
8Observer8 05.04.2026
Среду разработки Qt Creator можно установить без фреймворка Qt. Есть отдельный репозиторий для этой среды: https:/ / github. com/ qt-creator/ qt-creator, где можно скачать установщик, на вкладке Releases:. . .
AkelPad-скрипты, структуры, и немного лирики..
testuser2 05.04.2026
Такая программа, как AkelPad существует уже давно, и также давно существуют скрипты под нее. Тем не менее, прога живет, периодически что-то не спеша дополняется, улучшается. Что меня в первую очередь. . .
Отображение реквизитов в документе по условию и контроль их заполнения
Maks 04.04.2026
Алгоритм из решения ниже реализован на примере нетипового документа "ПланированиеСпецтехники", разработанного в конфигурации КА2. Данный документ берёт данные из другого нетипового документа. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru