c - 从数组和单独结构计算平均值
问题描述
试图制作一个程序来打印输入到文件中的数据。
除了输入分数的计算平均值之外,一切都运行良好。
我似乎无法弄清楚该怎么做,即使它应该很简单,但我就是无法理解它。
我目前得到的错误是:
"temp->mark = temp->mark + studentArray[j];" (Invlalid operands to
binary + (have 'float' and 'char *').
如果有人可以帮助我,将不胜感激。我试过以下
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <math.h>
struct student{
char name[30];
int id;
float mark;
};
int count = 0;
void student_update(char *stuadd);
void display(char *stuadd);
void main(int argc, char *studentArray[100])
{
int choice;
while(1)
{
printf("Welcome to Student Archives\n\n");
printf("1. Display Students' Details\n");
printf("2. Calculate average of all students’ marks \n");
printf("3. Add new student to the record \n");
printf("4. Quit Program\n");
scanf("%d",&choice);
switch(choice)
{
case 1:display(studentArray[100]);
break;
case 2:
break;
case 3:
student_update(studentArray[100]);
break;
case 4: printf("Program Terminated.\n");
exit(0);
default: printf("Wrong Choice. Enter again\n");
break;
}
}
}
void display(char *stuadd)
{
FILE *fptr;
char ch;
int rec = count;
fptr = fopen("stuadd.txt", "r");
struct student *temp = (struct student *)malloc(sizeof(struct student));
if (fptr == NULL)
printf("File does not exist.");
else
{
while (rec)
{
fread(temp->name, 50, 1, fptr);
printf(" %s\n", temp->name);
fread(&temp->id, sizeof(int), 1, fptr);
printf("%d", temp->id);
fread(&temp->mark, sizeof(int), 1, fptr);
printf("%.2f", temp->mark);
rec--;
}
}
fclose(fptr);
free(temp);
free(temp->name);
}
void calculateAverage(char *studentArray[100])
{
struct student *temp = (struct student *)malloc(sizeof(struct student));
int j;
float avg;
temp->mark = avg = 0;
for(j = 0; j < 100; j++)
{
temp->mark = temp->mark + studentArray[j];
}
avg = (float)temp->mark / j;
printf("Average of students' total marks are: %.2f",avg);
}
void student_update(char *stuadd)
{
FILE *fptr;
fptr = fopen("stuadd.txt", "a+");
struct student *temp = (struct student *)malloc(sizeof(struct student));
if (fptr == NULL)
printf("\nError.");
else
{
printf("\nEnter the students' name\n");
scanf(" %[^\n]s", temp->name);
printf("Enter the students' ID\n");
scanf("%d", &temp->id);
printf("Enter the students' mark\n");
scanf("%f", &temp->mark);
fprintf(fptr, "%s %d %.2f", temp->name, temp->id, temp->mark);
count++;
}
fclose(fptr);
free(temp);
free(temp->name);
}
解决方案
发布的代码无法编译!
在 ubuntu linux 下,使用:
gcc -ggdb -Wall -Wextra -Wconversion -pedantic -std=gnu11 -c "untitled.c" (in directory: /home/richard/Documents/forum)
编译器输出以下内容:
untitled.c:16:6: warning: return type of ‘main’ is not ‘int’ [-Wmain]
void main(int argc, char *studentArray[100])
^~~~
untitled.c: In function ‘main’:
untitled.c:16:15: warning: unused parameter ‘argc’ [-Wunused-parameter]
void main(int argc, char *studentArray[100])
^~~~
untitled.c: In function ‘display’:
untitled.c:48:10: warning: unused variable ‘ch’ [-Wunused-variable]
char ch;
^~
untitled.c:45:20: warning: unused parameter ‘stuadd’ [-Wunused-parameter]
void display(char *stuadd)
^~~~~~
untitled.c: In function ‘calculateAverage’:
untitled.c:83:33: error: invalid operands to binary + (have ‘float’ and ‘char *’)
temp->mark = temp->mark + studentArray[j];
~~~~~~~~~~ ^ ~~~~~~~~~~~~~~~
untitled.c:86:29: warning: conversion to ‘float’ from ‘int’ may alter its value [-Wconversion]
avg = (float)temp->mark / j;
^
untitled.c: In function ‘student_update’:
untitled.c:91:27: warning: unused parameter ‘stuadd’ [-Wunused-parameter]
void student_update(char *stuadd)
^~~~~~
Compilation failed.
还有一些其他问题,例如:
free(temp);
free(temp->name);
那是在分配的内存被传递给之后访问指向分配的内存的指针free()
。结果是未定义的行为。建议:
free(temp->name);
free(temp);
关于以下陈述
FILE *fptr;
fptr = fopen("stuadd.txt", "a+");
struct student *temp = (struct student *)malloc(sizeof(struct student));
if (fptr == NULL)
printf("\nError.");
始终在调用 C 库函数后立即检查错误指示。
将错误消息输出到stderr
, notstdout
当错误指示来自 C 库函数时,立即调用perror();
以输出您的错误消息和系统认为发生错误的文本原因,全部stderr
调用任何堆分配函数时:malloc
calloc
realloc
1) 返回的类型void*
可以分配给任何指针。强制转换只会使代码混乱,使其更难以理解、调试等。 2) 始终检查 (!=NULL) 返回值以确保操作成功。建议:
FILE *fptr;
fptr = fopen("stuadd.txt", "a+");
if ( !fptr )
{
perror("fopen failed");
exit( EXIT_FAILURE );
}
// implied else, fopen successful
struct student *temp = malloc(sizeof(struct student));
if( !temp )
{
perror( "malloc failed" );
exit( EXIT_FAILURE );
}
// implied else, malloc successful
关于:
scanf(" %[^\n]s", temp->name);
那个呼唤scanf()
是无稽之谈。由于输入格式说明符将在遇到换行符序列时%[^\n]
停止输入,因此下一个字符不可能成为 a 调用任何函数族时,请始终检查返回值以确保操作成功。当使用输入格式说明符和/或始终包含比输入缓冲区长度小 1 的 MAX CHARACTERS 修饰符时,以避免任何缓冲区溢出的可能性(以及由此产生的未定义行为)。建议去掉格式字符串的尾部,检查返回值,限制可以输入的字符总数,如:stdin
stdin
s
scanf()
%s
%[...]
s
if( scanf(" %29[^\n]", temp->name) != 1 )
{
fprintf( stderr, "scanf failed to input the student name\n" );
exit( EXIT_FAILURE );
}
关于:
for(j = 0; j < 100; j++)
{
temp->mark = temp->mark + studentArray[j];
}
没有数组studentArray[]
,所以这永远不会产生预期的结果。
关于错误消息:
avg = (float)temp->mark / j;
untitled.c:83:33: error: invalid operands to binary + (have ‘float’ and ‘char *’)
temp->mark = temp->mark + studentArray[j];
当然,不能将“float”值添加到指向 char 数组的指针中。你实际上想完成什么?
以上只是发布代码中问题的“冰山一角”。建议使用调试器并单步执行您的代码以确定许多问题
推荐阅读
- java - Java 14 javax.net.ssl.SSLException 仅在某些具有相同 cacerts 文件的计算机上发生
- llvm - 如何修改 LLVM IR 中的常量全局数组?
- nlp - 是否有任何 Python 库可以将平假名中的句子转换为汉字?
- javascript - 是否可以使用 ajax 调用整个 php 文件?
- c# - 播放器跳跃动画未在 unity2d 中播放
- javascript - 如何在 react-quill 工具栏中实现自定义代码块按钮
- unity3d - 顶点位移后动态重新计算法线
- c - 标签上的 Gtk 按钮按下事件
- php - 如何根据输入重定向到不同的页面
- c# - 为什么我在客户端上看不到响应类的属性?