Laboratory Exercise Working with Structs
Overview
This laboratory exercise provides practice working with the concept of a struct in C. Two typical applications include storing scores for several tests for a student and storing note information (pitch and duration) for a robot beep.
Work Started in Class
Example 1: Test Scores and Averages
For a course that involves three tests and a final exam, the record for a single student might have the form:
struct student { char name [20]; double test1; double test2; double test3; double exam; }
Consider the program test-scores-1.c which declares four student variables, initializes the variables, and prints their records.
-
Copy test-scores-1.c to your account, compile and run it, and review how it works.
- Where is struct student declared? Why do you think the declaration comes before procedure printStudent and before main?
- How are the fields of variable stu1 initialized?
- Why do you think strcpy is used to initialize the name field for stu4?
- In printing the output, how are the titles and printf statements coordinated, so that the test scores appear in aligned columns?
- The format %-20s is used to print the name of a student. What does the minus sign accomplish? (Hint: What happens if the minus sign is removed?)
-
Add function computeSemAvg to test-scores-1.c. This function should take a struct student as a parameter and return (not print) the weighted average that counts each test with a weight of 1 and the final exam with a weight of 2. That is, the semester average should be computed as;
(test1 + test2 + test3 + 2*exam) / 5.0; Use computeSemAvg to add a column to the output of the program. This should include two parts (plus the definition of the function):
- expand the printf statement in printStudent to include another value — the average for the student, and
- expand the printing of the title in main to label the new column.
-
As an experiment, change the value of the test1 field within computeSemAvg to 120.0. Is this new value printed in printStudent? Do you think the parameter struct student stu references the original data or makes a copy? Explain.
-
To change a value in main, step 3 illustrates that one must pass the address of struct, so the parameter will refer back to the original value. Write a procedure add10Percent that adds 10% to test2 of a student. The relevant procedure signature is:
void add10Percent (struct student * stu)
This function would be called within main with the address operator (&), with a call such as
add10Percent (&stu1);
Within add10Percent, remember to use the asterisk * to refer back to the original struct in main:
(*stu).test2 = ...
After printing the original records, call add10Percent for each of the four students, and then print their records again to check that the test2 scores have been changed.
An Array of Student Scores
Although test-scores-1.c was satisfactory for four students, the declaration of a different variable (e.g., stu1, stu2, stu3, stu4) for each student is awkward. As an alternative, program test-scores-2.c defines an array of struct students.
In this program:
- stu represents the entire array,
- stu[0], stu[1], stu[2], stu[3] specify the individual records for each of the four students,
- stu[0].test1, stu[1].test1, stu[2].test1, stu[3].test1 refer to the test1 scores for each of the students.
-
Working with test-scores-2.c, copy functions computeSemAvg and add10Percent from test-scores-1.c. Also, add the revised printStudent procedure and the revised printf for the title.
- Compile and run the updated test-scores-2.c, and check that the output is the same as you obtained from test-scores-1.c.
- Add at least six more student records, so that the stu array contains information for at least ten students.
-
Write a procedure printMinMax that computes and prints the
maximum and minimum semester averages for the entire class. Do NOT
assume that all averages will be between 0.0 and 100.0, but rather
initialize your search for a maximum and minimum with the averages of
the first student. The signature of this procedure should be
void printMinMax (struct student stu [], int numStudents)
where numStudents indicates the number of students in the stu array. - Modify printMinMax so that it prints the maximum and minimum semester averages, but also the names of the students with those averages.
- Modify test-scores-2.c so that 10% is added to each student's score for test 2, using the add10Percent procedure. This adjustment of student scores should occur after initialization, but before scores are printed or averages computed.
typedef statements
When working with test-scores-1.c and test-scores-2.c, you may have found it somewhat tedious to write struct student in the declaration of every variable and parameter. To simplify this syntax, C allows programmers to define new types. In this case, we might write
typedef struct { char name [20]; double test1; double test2; double test3; double exam; } studentType;
This defines a new data type studentType that you can use freely within your program with no further explicit mention of the keyword struct.
-
Copy test-scores-3.c to your account, compile and run it, and review how the typedef statement works.
- What happens if you move the typedef declaration after the definition of printStudent?
- Add an additional field, semAvg, to the studentType definition, but leave the initialization as it is. Does the program compile and run?
- Add printing of the average to printStudent, and observe what value is printed for stu.semAvg. How is this field initialized, if other fields of the struct already have values?
-
After initializing the studentType array, use a loop to compute and store each student's semester average:
int i; for (i = 0; i < 4; i++) { stu[i].semAvg = computeSemAvg (stu[i]); }
Check that these computed averages are now printed by the program.
Homework
Consider a song as a sequence of notes, each of which has a pitch and a duration. Playing a song might including announcing the song's title and then playing the notes. In this context, a note might have the following specification:
typedef struct { int pitch; double duration; } noteType;
-
Write a program to announce two songs and play them, organizing your program in two parts:
-
Write a procedure to play a song, using the following signature:
void playSong (char * title, noteType song[ ], int numNotes)
In this signature, the title is a string giving the title of the song, song is the sequence of notes as an array, and numNotes specifies how many notes are in the song array.
Procedure playSong should use eSpeak to announce a song, using the given title. Then, playSong should go through the note sequence, telling the Scribbler 2 robot to play each note with the designated pitch and frequency.
-
Write a main program containing at least two songs, defined as noteTupe arrays of at least 10 notes each. After declaring and initializing the song arrays, use playSong to announce and play each song.
-
-
Write a procedure to adjust the length of a note, using the following signature
void adjustNoteLength (noteType * note, double adjustment)
note refers back to the variable in main, so that a call to adjustNoteLength will change the original stored note. In processing, adjustment indicates the amount of change for a note's duration.
- When adjustment is 0.5, the duration of the note will be changed to half as long as the original duration.
- When adjustment is 1.0, the duration of the note will be unchanged.
- When adjustment is 2.0, the duration of the original note will be doubled.
Within the program,
- play each song once, as initialized.
- call adjustNoteLength for the first song with an adjustment of 0.5, so each note will be shorter. Then play the song using playSong, so the entire song will be completed in half the time of the original.
- call adjustNoteLength for the second song with an adjustment of 2.0, so each note will be shorter. Then play the song using playSong, so the entire song will require twice the time of the original.
created 1 August 2011 by Erik Opavsky revised 8 August 2011 by Erik Opavsky revised 14 November 2011 by Erik Opavsky minor editing 14 November 2011 by Henry M. Walker reading reference added 17 August 2012 by Henry M. Walker modest reformatting 1-2 February 2014 by Henry M. Walker readings added 19 September 2014 by Henry M. Walker lab reworked 28-30 December 2014 by Henry M. Walker lab reformated 19 October 2016 by Henry M. Walker |
![]() ![]() |
For more information, please contact Henry M. Walker at walker@cs.grinnell.edu. |