NOIP2018考前抱佛脚——图论基础复习
存图方式
邻接矩阵存图
int map[MAXN][MAXM];
for(int i = 1;i <= n;++i)
{
for(int j = 1;j <= m;++j)
{
int x = read(),y = read(),w = read();
map[x][y] = w;
}
}
邻接表存图
//最大顶点数
const int MAXN = 100010;
// vector实现邻接表定义
vector<int> e[MAXN];
// 邻接表初始化操作
// 将起点为`i`的边链全部清空
e[i].clear();
// 增加边 i 到 j 的边
e[i].push_back(j);
// 查询边
for(int i = 0;i < n;++i)
{
for(int j = 0;j < (int)e[i].size();++j)
{
//do something
}
}
链式前向星
const int MAXN = 1000010;
const int MAXM = 1000010;
struct edge{
int to; // 这条边的另外一个顶点
int next; // 下一条边的数组下标
}e[MAXM];
//顶点 i 的第一条边的数组下标
int head[MAXN];
memset(head,-1,sizeof(head));
//存储
inline void add(int x,int y,int id)
{
edge[id].to = y;
edge[id].next = head[x];
head[x] = id;
}
//遍历
for(int i = head[x];i != -1;i = e[i].next)
{
//do something
}
最小生成树
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
const int MAX_N = 50010;
const int MAX_M = 200000;
struct edge {
int u,v,w;
bool operator < (const edge &a)const{
return w < a.w;
}
}e[MAX_M];
int fa[MAX_N],n,m;
int get(int x){
if(fa[x] == x){
return fa[x];
}
return fa[x]=get(fa[x]);
}
int main() {
cin>>n>>m;
for(int i=0;i<m;i++){
scanf("%d%d%d",&e[i].u,&e[i].v,&e[i].w);
}
sort(e,e+m);
for(int i=1;i<=n;++i){
fa[i]=i;
}
int sum=0;
for(int i=0;i<m;++i){
int x = get(e[i].u),y = get(e[i].v);
if (x!=y){
fa[x]=y;
sum += e[i].w;
}
}
printf("%d",sum);
return 0;
}
例1 P1536 村村通
题目描述
某市调查城镇交通状况,得到现有城镇道路统计表。表中列出了每条道路直接连通的城镇。市政府“村村通工程”的目标是使全市任何两个城镇间都可以实现交通(但不一定有直接的道路相连,只要相互之间可达即可)。请你计算出最少还需要建设多少条道路?
输入输出格式
输入格式:
每个输入文件包含若干组测试测试数据,每组测试数据的第一行给出两个用空格隔开的正整数,分别是城镇数目N(N<1000)和道路数目M;随后的M行对应M条道路,每行给出一对用空格隔开的正整数,分别是该条道路直接相连的两个城镇的编号。简单起见,城镇从1到N编号。
注意:两个城市间可以有多条道路相通。例如:
3 3 1 2 1 2 2 1 这组数据也是合法的。当N为0时,输入结束。
输出格式:
对于每组数据,对应一行一个整数。表示最少还需要建设的道路数目。
输入输出样例
输入样例#1: 复制
4 2
1 3
4 3
3 3
1 2
1 3
2 3
5 2
1 2
3 5
999 0
0
输出样例#1: 复制
1
0
2
998
标程
#include<bits/stdc++.h>
int fa[20010];
int fin(int a){
if(fa[a] == a) return a;
else return fin(fa[a]);
}
int main(int argc, char const *argv[])
{
int n; scanf("%d",&n);
int m;
while(n){
scanf("%d",&m);
int t[10010];
memset(t,0,sizeof(t));
for(int i = 1;i <= n;i++ ) fa[i]=i;
for(int i = 1;i <= m;i++ ){
int a,b;
scanf("%d%d",&a,&b);
int k1=fin(b);
fa[k1]=fin(a);
}
for(int i = 1; i <= n; i++) fa[i] = fin(fa[i]);
for(int i = 1; i <= n; i++) t[fa[i]] ++;
int ans=0;
for(int i = 1; i <= 1000; i++) if(t[i]) ans++;
printf("%d\n",ans-1);
scanf("%d",&n);
}
return 0;
}
例2 P1546 最短网络 Agri-Net
题目背景
农民约翰被选为他们镇的镇长!他其中一个竞选承诺就是在镇上建立起互联网,并连接到所有的农场。当然,他需要你的帮助。
题目描述
约翰已经给他的农场安排了一条高速的网络线路,他想把这条线路共享给其他农场。为了用最小的消费,他想铺设最短的光纤去连接所有的农场。
你将得到一份各农场之间连接费用的列表,你必须找出能连接所有农场并所用光纤最短的方案。每两个农场间的距离不会超过100000
输入输出格式
输入格式:
第一行: 农场的个数,N(3<=N<=100)。
第二行..结尾: 后来的行包含了一个N*N的矩阵,表示每个农场之间的距离。理论上,他们是N行,每行由N个用空格分隔的数组成,实际上,他们限制在80个字符,因此,某些行会紧接着另一些行。当然,对角线将会是0,因为不会有线路从第i个农场到它本身。
输出格式:
只有一个输出,其中包含连接到每个农场的光纤的最小长度。
输入输出样例
输入样例#1: 复制
4
0 4 9 21
4 0 8 17
9 8 0 16
21 17 16 0
输出样例#1: 复制
28
标程
#include<bits/stdc++.h>
using namespace std;
const int MAX_N=101,MAX_M=10010;
int G[110][110];
const int INF=0x3f3f3f3f;
int fa[MAX_N],n,m;
struct kin
{
int u,v,w;
}rxz[MAX_M];
bool cmp(kin x,kin y){
return x.w<y.w;
}
int fin(int x){
if(fa[x]==x) return fa[x];
return fa[x]=fin(fa[x]);
}
int main(){
scanf("%d",&n);
memset(G,INF,sizeof(G));
for(int i=0;i<n;i++){
for(int j = 0; j < n; j++)
{
int w;
scanf("%d",&w);
if(w!=0) G[i][j]=w;
}
}
int eid=0;
for(int i = 0; i < n; i++)
{
for(int j = 0; j < n; j++)
{
if(G[i][j]<INF){
rxz[eid].u=i;
rxz[eid].v=j;
rxz[eid].w=G[i][j];
eid++;
}
}
}
sort(rxz,rxz+eid,cmp);/*
for(int i = 0; i < eid; i++)
{
cout<<rxz[i].u<<ends<<rxz[i].v<<ends<<rxz[i].w<<endl;
}*/
for(int i = 1; i <= MAX_N; ++i)
{
fa[i]=i;
}
int rest = n,ans=0;
for(int i = 0; i < eid && rest>1; ++i)
{
int x = fin(rxz[i].u),y=fin(rxz[i].v);
if (x!=y) {
fa[x]=y;
//cout<<rxz[i].w<<ends;
ans = ans + rxz[i].w;
rest--;
//cout<<ans<<endl;
}
}
printf("%d",ans);
return 0;
}
例3 P1991 无线通讯网
题目描述
国防部计划用无线网络连接若干个边防哨所。2 种不同的通讯技术用来搭建无线网络;
每个边防哨所都要配备无线电收发器;有一些哨所还可以增配卫星电话。
任意两个配备了一条卫星电话线路的哨所(两边都ᤕ有卫星电话)均可以通话,无论
他们相距多远。而只通过无线电收发器通话的哨所之间的距离不能超过 D,这是受收发器的功率限制。收发器的功率越高,通话距离 D 会更远,但同时价格也会更贵。
收发器需要统一购买和安装,所以全部哨所只能选择安装一种型号的收发器。换句话说,每一对哨所之间的通话距离都是同一个 D。你的任务是确定收发器必须的最小通话距离 D,使得每一对哨所之间至少有一条通话路径(直接的或者间接的)。
输入输出格式
输入格式:
从 wireless.in 中输入数据第 1 行,2 个整数 S 和 P,S 表示可安装的卫星电话的哨所数,P 表示边防哨所的数量。接下里 P 行,每行两个整数 x,y 描述一个哨所的平面坐标(x, y),以 km 为单位。
输出格式:
输出 wireless.out 中
第 1 行,1 个实数 D,表示无线电收发器的最小传输距离,精确到小数点后两位。
输入输出样例
输入样例#1: 复制
2 4
0 100
0 300
0 600
150 750
输出样例#1: 复制
212.13
标程
#include<bits/stdc++.h>
using namespace std;
int fa[520];
struct edge
{
int u,v;
double w;
bool operator < (const edge &a)
{
return w < a.w;
}
}G[300010];
inline int fin(int x)
{
return fa[x] == x ? x : fa[x] = fin(fa[x]);
}
int main(int argc, char const *argv[])
{
int x[520],y[520];
int s,p,num;
cin >> s >> p;
num = p;
for(int i = 1;i <= p;++i)
{
fa[i] = i;
cin >> x[i] >> y[i];
}
int k = 0;
for(int i = 1;i <= p;++i)
{
for(int j = 1;j <= p;++j)
{
G[k].u = i;
G[k].v = j;
G[k].w = sqrt((x[i] - x[j])*(x[i] - x[j]) + (y[i] - y[j])*(y[i] - y[j]));
k++;
}
}
sort(G,G+k);
int i;
for(i = 0;i < k;++i)
{
int nx = fin(G[i].u) , ny = fin(G[i].v);
if(nx != ny)
{
fa[nx] = fin(fa[ny]);
num--;
}
if(num <= s) break;
}
printf("%.2lf",G[i].w);
return 0;
}
最短路
模板
#include<bits/stdc++.h>
#define P std::pair<long long,int >
const int MAXN = 100086;
const int MAXM = 521314;
const int inf = 0x7fffffff;
inline int read()
{
int x = 0,t = 1; char ch = getchar();
while(ch<'0'||ch>'9'){if(ch=='-')t=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
return x*t;
}
struct edge{
int u,v,w,next;
}e[MAXM];
int head[MAXN],dis[MAXN],cnt;
bool vis[MAXN];
int n,m,s;
std::priority_queue<P,std::vector<P>,std::greater<P> > q;
inline void add(int u,int v,int w)
{
cnt++;
e[cnt].w = w;
e[cnt].u = u;
e[cnt].v = v;
e[cnt].next = head[u];
head[u] = cnt;
}
inline void init()
{
n = read(); m = read(); s = read();
for(int i = 1;i <= m;++i)
{
int x = read(),y = read(),w = read();
add(x,y,w);
}
for(int i = 0;i <= n;++i)
dis[i] = inf;
dis[s] = 0;
q.push(std::make_pair(0,s));
while(!q.empty())
{
int x = q.top().second;
q.pop();
if(!vis[x])
{
vis[x]=1;
for(int i = head[x];i;i = e[i].next)
{
int v = e[i].v;
dis[v] = std::min(dis[v],dis[x]+e[i].w);
q.push(std::make_pair(dis[v],v));
}
}
}
for(int i=1;i<=n;i++)
{
printf("%d ",dis[i]);
}
}
int main(void)
{
init();
return 0;
}
NOIP2018考前抱佛脚——图论基础复习的更多相关文章
- NOIP2018考前抱佛脚——数据结构基础及STL实现
目录 动态数组 栈 队列 优先队列 动态数组 srand(time(0)); std::vector<int> qwq; for(int i = 1;i <= 10;++i) qwq ...
- NOIP2018考前抱佛脚——搜索复习
目录 搜索 DFS 例1 P1101 单词方阵 题目描述 输入输出格式 输入输出样例 标程 例2 P1605 迷宫 题目背景 输入输出格式 输入输出样例 标程 例3 P1019 单词接龙 题目描述 输 ...
- 《CSS权威指南》基础复习+查漏补缺
前几天被朋友问到几个CSS问题,讲道理么,接触CSS是从大一开始的,也算有3年半了,总是觉得自己对css算是熟悉的了.然而还是被几个问题弄的"一脸懵逼"... 然后又是刚入职新公司 ...
- Java基础复习笔记系列 九 网络编程
Java基础复习笔记系列之 网络编程 学习资料参考: 1.http://www.icoolxue.com/ 2. 1.网络编程的基础概念. TCP/IP协议:Socket编程:IP地址. 中国和美国之 ...
- Java基础复习笔记系列 八 多线程编程
Java基础复习笔记系列之 多线程编程 参考地址: http://blog.csdn.net/xuweilinjijis/article/details/8878649 今天的故事,让我们从上面这个图 ...
- Java基础复习笔记系列 七 IO操作
Java基础复习笔记系列之 IO操作 我们说的出入,都是站在程序的角度来说的.FileInputStream是读入数据.?????? 1.流是什么东西? 这章的理解的关键是:形象思维.一个管道插入了一 ...
- Java基础复习笔记系列 五 常用类
Java基础复习笔记系列之 常用类 1.String类介绍. 首先看类所属的包:java.lang.String类. 再看它的构造方法: 2. String s1 = “hello”: String ...
- Java基础复习笔记系列 四 数组
Java基础复习笔记系列之 数组 1.数组初步介绍? Java中的数组是引用类型,不可以直接分配在栈上.不同于C(在Java中,除了基础数据类型外,所有的类型都是引用类型.) Java中的数组在申明时 ...
- C语言基础复习总结
C语言基础复习总结 大一学的C++,不过后来一直没用,大多还给老师了,最近看传智李明杰老师的ios课程的C语言入门部分,用了一周,每晚上看大概两小时左右,效果真是顶一学期的课,也许是因为有开发经验吧, ...
随机推荐
- Netbeans 8.0配置Python开发环境
1. 菜单栏:工具->插件->设置->添加 配置如下信息: http://deadlock.netbeans.org/hudson/job/nbms-and-javadoc/last ...
- django2.1---终端打印orm转义的sql语句
print(connection.queries)可以打印转义后的sql语句 例子: from django.http import JsonResponse,HttpResponse from .m ...
- Netty 解码器抽象父类 ByteToMessageDecoder 源码解析
前言 Netty 的解码器有很多种,比如基于长度的,基于分割符的,私有协议的.但是,总体的思路都是一致的. 拆包思路:当数据满足了 解码条件时,将其拆开.放到数组.然后发送到业务 handler 处理 ...
- 并发编程 —— ConcurrentHashMap size 方法原理分析
前言 ConcurrentHashMap 博大精深,从他的 50 多个内部类就能看出来,似乎 JDK 的并发精髓都在里面了.但他依然拥有体验良好的 API 给我们使用,程序员根本感觉不到他内部的复杂. ...
- TRIZ解决问题方法
个人觉的成功是有规律的,那些成功的人士,都有一套处理事情的秘籍.只要我们的思维方式把那些秘籍融会贯通,并快速执行,我们有一天也会成功的. TRIZ解决问题的5点方法. 1.确定最终目标. 2.列出阻碍 ...
- Hibernate映射 --- 自身跟自身的一对多关联
自身跟自身的一对多关联 什么时候会出现自身一对多关系呢?下面举个例子. 淘宝店里商品分类,一级分类:家用电器,个人化妆,运动户外等 家用电器下面二级分类:大家电,生活电器,厨房电器等 二级分类大家 ...
- Oracle- 复杂查询及总结
一.复杂查询 1. 列出至少有一个员工的所有部门编号.名称,并统计出这些部门的平均工资.最低工资.最高工资. 1.确定所需要的数据表: emp表:可以查询出员工的数量: dept表:部门名称: emp ...
- thinkphp 百度编辑器和layer简单用法
百度编辑器1.4.3.3和layer插件简单案例 :后台单页面管理 增删改查操作 此处为默认图片保存路径,如果要修改保存路径,需要修改config文件. 添加页. <extend name=&q ...
- vb.net連接ACCESS数据库
'導入命名空間Imports System.Data.OleDb '定義變量 Dim Sql As String 'OleDb連線 Dim SqlAC As OleDbConnection Dim C ...
- ORM--Entity Framework 学习(01)
关键词:Entity Framework:微软官方提供的ORM工具,ORM让开发人员节省数据库访问的代码时间,将更多的时间放到业务逻辑层代码上.EF提供变更跟踪.唯一性约束.惰性加载.查询事物等.开发 ...