C++第2周项目4——动态链表初试

简介: 课程首页地址:http://blog.csdn.net/sxhelijian/article/details/7910565,本周题目链接:http://blog.csdn.net/sxhelijian/article/details/8635385【项目4-动态链表初试】数据依然来自score.txt,在程序中建立一个动态链表:每读入一个同学的数据,计算总分,分配结点的存储空间并赋值,并

课程首页地址:http://blog.csdn.net/sxhelijian/article/details/7910565,本周题目链接:http://blog.csdn.net/sxhelijian/article/details/8635385


【项目4-动态链表初试】数据依然来自score.txt,在程序中建立一个动态链表:每读入一个同学的数据,计算总分,分配结点的存储空间并赋值,并建立起前后相链的关系。在建立链表的同时,要进行统计,以便于求出所有同学总分的平均成绩。动态链表建立后,从头结点开始,依次输出所有总分高于平均总分且没有挂科的同学的学号、姓名、总分。

参考解答:

#include  <iostream>
#include  <fstream>
#include <cstdlib>  //在codeblocks下,exit(1)需要这个头文件
using namespace std;
struct Student
{
    char num[13]; //尽管都是由数字构成,但看作为字符更合适。
    char name[10]; //每个汉字占两个字节,中国人的名字,5个汉字够用
    int cpp;
    int math;
    int english;
    int grade;
    struct Student *next;   //指向下一节点的指针
};

int main( )
{
    Student *head=NULL,*p,*q;
    int stuNum=0,i;
    int sum=0,ave;   //用于求平均
    ifstream infile("score.txt",ios::in); 	//以输入的方式打开文件
    if(!infile)   				//测试是否成功打开
    {
        cerr<<"open error!"<<endl;
        exit(1);
    }
    //下面从文件中读取数据,同时建立动态链表,并为求总分之和
    stuNum=0;
    while(!infile.eof())
    {
        p = new Student;
        infile>>p->num>>p->name>>p->cpp>>p->math>>p->english;
        p->grade = p->cpp + p->math + p->english;
        sum+=p->grade;
        p->next=NULL;
        if (stuNum==0)
            head=p;   //是第一个节点
        else
            q->next=p;   //用q记录刚刚产生的结点,如果有下一个,需要q将之连起来
        ++stuNum;
        q=p;
    }
    infile.close();

    //求平均成绩
    ave=sum/stuNum;
    cout<<"总分平均为:"<<ave<<endl;

    //依次输出所有总分高于平均总分且没有挂科的同学的学号、姓名、总分。
    cout<<"总分高于平均总分且没有挂科的同学有:"<<endl;
    p=head;
    i=1;
    while(p!=NULL)
    {
        if(p->grade>=ave&&p->cpp>=60&&p->math>=60&&p->english>=60)
        {
            cout<<i<<" "<<p->num<<" "<<p->name<<" "<<p->grade<<endl;
            i++;
        }
        p=p->next;
    }
    return 0;
}

【项目4扩展(选做)】(1)链表建立起来后,将总分低于总均总分的节点删除(注意删除后前后链仍然能够连起来,不用的空间也能正常释放);(2)问题:在完成项目2、3和项目4基础上,请说出利用结构体数组和动态链表实现此类应用各自的优劣,可以将你的体会写到各任务的体会中。

//这个程序在一位热心读者提供的代码基础上做了修改。
#include<iostream>
#include<cstdlib>
#include<fstream>
#include<iomanip>
using namespace std;
struct node
{
    char num[15];
    char name[10];
    int cpp;
    int math;
    int english;
    int score;
    node* pnext;
};
int main()
{
    node *head=NULL, *p,*q, *k;
    int num=0;
    float sum=0,ave;
    ifstream readfile("score.txt",ios::in);
    if(!readfile)
    {
        cerr<<"the score.txt can't be read!"<<endl;
        exit(1);
    }
    while(!readfile.eof())
    {
        p=new node;
        readfile>>p->num>>p->name>>p->cpp>>p->math>>p->english;
        p->score=p->cpp+p->english+p->math;
        sum+=p->score;
        p->pnext=NULL;
        if(num==0)
            head=p;
        else
            q->pnext=p;
        num++;
        q=p;
    }
    readfile.close();
    ave=sum/num;
    cout<<"总体平均分是:   "<<ave<<endl<<endl;
    //将总分低于总均总分的节点删除
    while(head!=NULL&&head->score<ave)//首先,将头部该删除的删除掉
    {
        q=head;
        head=head->pnext;
        delete(q);
    }
    //通过下面q、p一后一前两个指针,删除掉需要删除的节点
    q=head;
    p=head->pnext;
    while(p!=NULL)
    {
        if(p->score<ave)//删除p
        {
            k=p;   //p该删除,由k标记住
            p=p->pnext;  //下一步要考察的,是pnext指向的
            q->pnext=p;  //这个很关键,要被删除的k,就这样脱离了链接
            delete(k);   //释放空间
        }
        else  //保留,p和q都往后移
        {
            q=p;
            p=p->pnext;
        }
    }
    //输出剩余的结点,便于观察结果
    p=head;
    cout<<"总分高于平均总分的同学有: "<<endl<<endl;
    while(p!=NULL)
    {
        cout<<setw(10)<<p->num<<setw(10)<<p->name<<setw(10)<<p->score<<endl;
        p=p->pnext;
    }
    return 0;
}

  有学生对这段代码提出疑问:

while(head!=NULL&&head->score<ave)//首先,将头部该删除的删除掉  
    {  
        q=head;  
        head=head->pnext;  
        delete(q);  
    }  
  学生说:对这段while循环代码不理解,我的理解是:如果这样写的话,把总分低于平均总分的删除掉的话,整个链表都被七零八落的,就像锁链一样,如果隔三差五的打断以后,就变成一段一段的,无法再连接起来了。

  还有下面的一句:q=head; 按照上面的while循环,head已经到了链表的结尾了,q=head,无非就是在链尾了。我感觉程序是行不通的,但是,学生运行你的程序还是可以正常运行并输出的。
  把上面的while循环屏蔽以后,感觉整个程序非常清晰,而且还能正常的运行并输出想要的正确结构,和你原来的程序运行的结构是一样的!
  对上面的那段while循环加q=head;这部分不理解!还要麻烦老师有时间解释一下!

  我的答复是:

  这个题目中的核心就是链表中某些节点的删除。一些不再要的节点删除后,该有的链要链起来,删除的节点的空间释放掉,并不会“七零八落”。

while(head!=NULL&&head->score<ave)//首先,将头部该删除的删除掉  
    {  
        q=head;  
        head=head->pnext;  
        delete(q);  
    }  

  这段代码的效果是,如果head指向的节点该删除,就一直删除,直到head指向的节点的成绩一定大于平均。这是需要的。之所以运行结果不受影响,只是意外,本题给的测试数据第一节点就不必删除。
  这段代码并没有让head指向了链表尾,因为还有head->score<ave条件限制。
  有了这个做保证,在考察p是否该删除时,q指向的是它的前一个节点。若p不该删除,p和q一起往后走;而若p该删除,k=p;记住了该删除的节点,p=p->pnext;表示下一循环要考察的结点, q->pnext=p;在链表中就断掉了p指向的节点(k),最后释放k指向的空间。
  这是很正统的链表删除操作,建议找数据结构的书看一看,不少程序设计的书上也有。







目录
相关文章
|
7月前
|
编译器 C++ 开发者
【Conan 入门教程 】使用Conan 2.X和Autotools高效构建C/C++项目
【Conan 入门教程 】使用Conan 2.X和Autotools高效构建C/C++项目
374 1
|
7月前
|
C++
【链表】还不会用C++实现链表?一文教会你各种链表的实现
【链表】还不会用C++实现链表?一文教会你各种链表的实现
288 0
|
7月前
|
存储 算法 Linux
【实战项目】网络编程:在Linux环境下基于opencv和socket的人脸识别系统--C++实现
【实战项目】网络编程:在Linux环境下基于opencv和socket的人脸识别系统--C++实现
246 7
WK
|
1月前
|
机器学习/深度学习 人工智能 算法
那C++适合开发哪些项目
C++ 是一种功能强大、应用广泛的编程语言,适合开发多种类型的项目。它在游戏开发、操作系统、嵌入式系统、科学计算、金融、图形图像处理、数据库管理、网络通信、人工智能、虚拟现实、航空航天等领域都有广泛应用。C++ 以其高性能、内存管理和跨平台兼容性等优势,成为众多开发者的选择。
WK
75 1
|
2月前
|
Ubuntu Linux 编译器
Linux/Ubuntu下使用VS Code配置C/C++项目环境调用OpenCV
通过以上步骤,您已经成功在Ubuntu系统下的VS Code中配置了C/C++项目环境,并能够调用OpenCV库进行开发。请确保每一步都按照您的系统实际情况进行适当调整。
498 3
|
3月前
|
C++
【C++案例】一个项目掌握C++基础-通讯录管理系统
这篇文章通过一个通讯录管理系统的C++项目案例,详细介绍了如何使用C++实现添加、显示、删除、查找、修改和清空联系人等功能。
55 3
|
5月前
|
Rust 测试技术 编译器
Rust与C++的区别及使用问题之Rust项目中组织目录结构的问题如何解决
Rust与C++的区别及使用问题之Rust项目中组织目录结构的问题如何解决
|
4月前
|
编译器 C++ 开发者
Visual Studio属性表:在新项目中加入已配置好的C++库
通过以上步骤可以确保Visual Studio中新项目成功地加入了之前已配置好的C++库。这个过程帮助开发者有效地管理多个项目中共享的库文件,提升开发效率。
125 0
|
5月前
|
存储 C++
C++的list-map链表与映射表
```markdown C++ 中的`list`和`map`提供链表和映射表功能。`list`是双向链表,支持头尾插入删除(`push_front/push_back/pop_front/pop_back`),迭代器遍历及任意位置插入删除。`map`是键值对集合,自动按键排序,支持直接通过键来添加、修改和删除元素。两者均能使用范围for循环遍历,`map`的`count`函数用于统计键值出现次数。 ```
|
6月前
|
存储 C++
C++的list-map链表与映射表
这篇教程介绍了C++中`list`链表和`map`映射表的基本使用。`list`链表可通过`push_front()`、`push_back()`、`pop_front()`和`pop_back()`进行元素的添加和删除,使用迭代器遍历并支持在任意位置插入或删除元素。`map`是一个键值对的集合,元素自动按键值排序,可使用下标操作符或`insert()`函数插入元素,通过迭代器遍历并修改键值对,同时提供`count()`方法统计键值出现次数。教程中包含多个示例代码以帮助理解和学习。