信息学奥赛复赛复习07-CSP-J2020-03表达式前置知识点-结构体、链表、链式前向星

PDF文档回复:20240930

1 2020 CSP-J 题目1 优秀的拆分

[题目描述]

链式前向星模板题,读入n个点,m条边,以及flag,若flag==1则图有向,否则无向。对每个点输出它的每一条边

[输入格式]

第一行三个数n,m,flag,题意如上所示 第2~1+m行,每行三个数,x,y,z,代表从x到y有一条长为z的边

[输出格式]

若flag=1则m行,flag=0则m2行,无向图中边a-b,存储 a -> b和 b -> a两条,所以为m * 2

每行三个数,即该点的编号、所指向点的编号,边的长度,先按第一个数升序排列,再以链式前向星中的顺序输出即可

其实就是i从1到n,再按顺序查找边输出即可, 特殊的,若该点无出边,单独一个空行

[输入输出样例]

输入 #1

5 5 0
1 2 5
1 4 6
2 3 7
3 5 3
3 4 1

输出 #1

1 4 6
1 2 5
2 3 7
2 1 5
3 4 1
3 5 3
3 2 7
4 3 1
4 1 6
5 3 3

输入 #2

4 3 1
1 3 6
3 4 1
2 1 3

输出 #2

1 3 6
2 1 3
3 4 1

说明/提示

对于100%的数据,m<=4000000;l<=1e9;保证图连通

2 相关知识点

1) 结构体

结构体属于用户自定义的数据类型,允许用户存储不同的数据类型

例如:

学生有姓名/年龄/分数,其中,姓名和年龄/分数是不同类型,因此不能使用数组准确存储

创建结构体

创建一个新的学生数据类型:学生包括(姓名,年龄,分数)

struct Student{
    //成员列表
 
    //姓名
    string name;
    //年龄
    int age;
    //分数
    int score;
 
};

声明结构体类型

Student s1;

结构体赋值

s1.name = "张三";
s1.age = 18;
s1.score = 100;

整体示例参考

#include <bits/stdc++.h>
using namespace std;
/*创建学生数据类型
  struct 类型名称 { 成员列表 }
*/
struct Student{
    //成员列表
    //姓名
    string name;
    //年龄
    int age;
    //分数
    int score;
}S[10];//定义以Student为类型的数组S 
 
int main(){
    Student s1; //声明结构体变量 
    //给s1属性赋值,通过.访问结构体变量中的属性
    s1.name = "张三";
    s1.age = 18;
    s1.score = 100;
    cout << "姓名:" << s1.name << "年龄:" << s1.age << "分数:" << s1.score << endl;
    Student s2 = { "李四" , 19 , 80 };//声明结构体变量并赋值 
    cout << "姓名:" << s1.name << "年龄:" << s1.age << "分数:" << s1.score << endl;
    
    //结构体数组 
    S[0].name = "王五";
    S[0].age = 30;
    S[0].score = 98;
    cout << "姓名:" << S[0].name << "年龄:" << S[0].age << "分数:" << S[0].score << endl;
    return 0;
}

2) 链表

是一种常见的数据结构,它是由一系列节点(Node)组成,每个节点包含两部分:数据域和指针域。数据域用于存储数据,指针域用于存储下一个节点的地址。链表的第一个节点称为头节点(Head),最后一个节点称为尾节点(Tail),尾节点的指针域指向空(NULL)

在结点中数据域用来存储数据元素,指针域用于指向下一个具有相同结构的结点。

因为只有一个指针结点,称为单链表

3) 链式前向星

链式前向星就是把数据以链表形式连接在一起,通过起点,把相同起点的边连接在一起,每个点为起点的所有边

3 思路分析
head数组,保存起点元素的最后一个位置

边数组,保存每个起点对应的链表,对应每一种颜色为一个链表

#include<bits/stdc++.h>
using namespace std;
const int N=4000010;
long long n,m,f,cnt,h[N];
/*
  h[] 存储对应元素起点的最大下标
  例如 边1->3  1->4 1->5  此时h[1]的值为3
  e数组 存储所有边,其中这些边相同起点通过链表串连到一起
  例如  边1->3  1->4 1->5
  上面3条边通过起点1连在一起,5->next为4,5->next为3 
*/ 
struct node{//相同起点为1个链表 
	int next,w,to;//to边连向的点 w 边的长度 next下一个结点 
}e[N];
/*
  链式前向星加边 
  加a到b的边 边权为c 
*/ 
void add(int a,int b,int c){
	cnt++;//增加一条边对应数组下标 
	e[cnt].to=b;//指向b结点 
	e[cnt].w=c;//边权为c 
	e[cnt].next=h[a];//下一个结点执向h[a] 指向前一个结点 
	h[a]=cnt;//h[a]指向当前结点  增加了1条边 ,h[a]加1 
}
int main(){
	cin>>n>>m>>f;//n个结点 m条边 f有向或无向图 
	for(int i=1;i<=m;i++){//一次输入m条边 
		int a,b,c;//从a到b的边 长度为c
		scanf("%d%d%d",&a,&b,&c);
		add(a,b,c);//加一条从a到b的边,长度为c
		if(f==0){//如果f为0说明为无向图,同时增加一条从b到a,长度为c的边 
			add(b,a,c);
		}
	}
	//输出
	for(int i=1;i<=n;i++){//i如果为0说明没有以i为起点的边,如果不为0,h[i]的值则是以i为起点的边数 
		for(int k=h[i];k;k=e[k].next) {//遍历i为起点的边 
			printf("%d %d %d\n",i,e[k].to,e[k].w);
		}
	}
	return 0;
} 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值