Вы близки по всем пунктам @ tadman поймал вашу самую большую ошибку при создании массива из 1 указателя с emp_t *everyone[1]
(теперь удалено в вашем вопросе).
Остальная частьпроблемы с вашим кодом - это, скорее, сравнительно небольшие исправления или улучшения.
Для начала, избегайте использования "магических чисел" в вашем коде (например, 20, 8, 2
).Если вам нужны константы, #define
их или используйте глобальный enum
для определения нескольких констант одновременно, например,
/* an enum can be used to #define multiple constants */
enum { MAXID = 8, MAXTITLE = 20, MAXEMP = 128 };
, а затем
typedef struct {
char *name;
char title[MAXTITLE]; /* avoid "magic-numbers", use constants */
char id[MAXID];
int yearsExp;
} emp_t;
Ваша create_emp()
функциябудет в основном работать как есть, но нет необходимости приводить возвращение malloc, calloc
или realloc
в C, см .: ли я приводить результат malloc? .Также я бы не использовал new
в качестве временного имени структуры.Хотя в C нет никакого реального конфликта, new
является ключевым словом в C ++, и если вы будете кодировать оба, лучше помнить об этом.С парой настроек вы могли бы написать create_emp()
следующим образом:
emp_t *create_emp (const char *name, const char *title,
const char *id, int yrxp) {
emp_t *newemp; /* new is a keyword in C++ best to avoid */
newemp = malloc (sizeof *newemp); /* don't cast return of malloc */
/* if you allocate, you must validate */
if (newemp == NULL) {
perror ("malloc-newemp");
return NULL;
}
newemp->name = malloc (strlen(name) + 1);
if (newemp->name == NULL) {
perror ("malloc-newemp->name");
free (newemp);
return NULL;
}
strcpy (newemp->name, name);
strcpy (newemp->title, title);
strcpy (newemp->id, id);
newemp->yearsExp = yrxp;
return newemp;
}
( примечание: всегда проверять каждое выделение. malloc, calloc & realloc
может и потерпеть неудачу)
Наконец, в main()
вы можете использовать индекс (ndx
ниже) вместо magic-number 2
и увеличивать индекс с каждым добавлением.Хотя у вас хорошо получается использовать модификаторы field-width для управления размером выходного поля, для строк (и всех спецификаторов преобразования) вы можете включить флаг '-'
как часть спецификатора преобразования, чтобы создать поле Выравнивание влево , чтобы выровнять вывод правильно.Остальная часть вашего кода в порядке.
В целом, вы можете сделать что-то вроде следующего:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
/* an enum can be used to #define multiple constants */
enum { MAXID = 8, MAXTITLE = 20, MAXEMP = 128 };
#define ADMIN "Administrator" /* simple #defines are fine */
#define MNG "Manager" /* string literals are fine as well */
#define PROG "Programmer" /* (up to you) */
typedef struct {
char *name;
char title[MAXTITLE]; /* avoid "magic-numbers", use constants */
char id[MAXID];
int yearsExp;
} emp_t;
emp_t *create_emp (const char *name, const char *title,
const char *id, int yrxp) {
emp_t *newemp; /* new is a keyword in C++ best to avoid */
newemp = malloc (sizeof *newemp); /* don't cast return of malloc */
/* if you allocate, you must validate */
if (newemp == NULL) {
perror ("malloc-newemp");
return NULL;
}
newemp->name = malloc (strlen(name) + 1);
if (newemp->name == NULL) {
perror ("malloc-newemp->name");
free (newemp);
return NULL;
}
strcpy (newemp->name, name);
strcpy (newemp->title, title);
strcpy (newemp->id, id);
newemp->yearsExp = yrxp;
return newemp;
}
void free_emp (emp_t *employee) {
free (employee->name);
free (employee);
}
int main (void) {
int i, ndx = 0; /* use an index instead of magic-numbers */
emp_t *everyone[MAXEMP] = {NULL};
everyone[ndx++] = create_emp ("Mike Papamichail", PROG, "A197482", 3);
everyone[ndx++] = create_emp ("Maria Mamalaki", MNG, "Z104781", 6);
everyone[ndx++] = create_emp ("Sam Sunami", ADMIN, "B426310", 10);
for (i = 0; i < ndx; i++) { /* %- to left justify fields */
if (everyone[i]) { /* validate not NULL */
printf ("%-15s \t %-15s \t %-10s \t %4d\n", everyone[i]->name,
everyone[i]->title, everyone[i]->id,
everyone[i]->yearsExp);
free_emp (everyone[i]);
everyone[i] = NULL;
}
}
return 0;
}
( примечание: изменение от everyone[0]->name
на everyone[i]->name
и т. д., иначе выход никогда не изменится)
Пример использования / Вывод
$ ./bin/emp_struct
Mike Papamichail Programmer A197482 3
Maria Mamalaki Manager Z104781 6
Sam Sunami Administrator B426310 10
Просмотрите все и дайте мне знать, если у вас естьдополнительные вопросы.