图论基础(python蓝桥杯)

news2024/11/27 14:54:36

图的基本概念

图的种类

怎么存放图呢?

优化

DFS

不是最快/最好的路,但是能找到一条连通的道路。(判断两点之间是不是连通的)

蓝桥3891

import os
import sys
sys.setrecursionlimit(100000)
# 请在此输入您的代码
n, m = map(int, input().split()) # n个点, 小明序号m
G = [[] for _ in range(n + 1)] # 邻接表,存放图。
rudu = [0] * (n + 1) # 记录每个点的入度
vis = [0] * (n + 1) # dfs的标记数组,看是否遍历过
# 二元组,分别表示每个子树数量和编号
dis = [[0, i] for i in range(n + 1)] # 排序用的二元组
for _ in range(n - 1):
  l, r = map(int, input().split())
  G[r].append(l) # r是l的父亲
  rudu[l] += 1
for i in range(1, n + 1):
  if rudu[i] == 0:
    root = i # 入度为0的是根节点,找到根节点,从根节点开始遍历。

def dfs(u):
  # 同时记录每个点的子树节点数
  dis[u][0] = -1 # 1改成-1,以便都从小到大排序
  vis[u] = 1
  for v in G[u]:
    if vis[v] == 0:
      dfs(v)
      dis[u][0] += dis[v][0]

dfs(root)
dis.sort()
# print(dis)
for i, (x, y) in enumerate(dis, 1): # 取出dis的排名,1的意思是索引从1开始
  if y == m:
    print(i)
    break

BFS

按层次分节点(几步能走的点)

不断这样取,直到终点。

蓝桥1509

import os
import sys

# 请在此输入您的代码
from collections import deque
def bfs(s, t):
  # s起点, t终点。
  dis = [-1] * 100001
  queue = deque()
  # 将起点塞入队列中,打上标记。
  queue.append(s)
  dis[s] = 0
  # 当队列非空
  while len(queue) != 0:
    # 取出队首元素u
    u = queue.popleft()
    # 判断u是否为终点
    if u == t:
      return dis[u]
    # 将u相连的所有点v,只要v未标记,则打标记,入队列
    for v in [u - 1, u + 1, u * 2]:
      # 特判:越界、已标记、障碍物
      if 0 <= v <= 100000 and dis[v] == -1:
        queue.append(v)
        dis[v] = dis[u] + 1
  return -1
n, k = map(int, input().split())
print(bfs(n, k))

蓝桥3819

import os
import sys

# 请在此输入您的代码
from collections import deque

def bfs(x, y, dis):
  queue = deque()
  vis = [[0] * m for i in range(n)]
  # 将起点入队列
  queue.append([x, y])
  dis[x][y] = 0
  vis[x][y] = 1
  while len(queue) != 0:
    x, y = queue.popleft()
    # 要求所有点,这步省略
    for deltax, deltay in [(1, 0), (0, 1), (-1, 0), (0, -1)]:
      xx, yy = x + deltax, y + deltay
      # 未越界,未标记,未障碍物
      if 0 <= xx < n and 0 < yy < m and vis[xx][yy] == 0 and g[xx][yy] != '#':
          queue.append([xx, yy])
          dis[xx][yy] = dis[x][y] + 1
          vis[xx][yy] = 1

n, m = map(int, input().split())
INF = 1e9 # 把路堵死了,永远走不到终点。
A, B, C, D = map(int, input().split())
A, B, C, D = A - 1, B - 1, C - 1, D - 1
g = [input() for i in range(n)]
E = int(input())
dis1 = [[INF] * m for i in range(n)]
dis2 = [[INF] * m for i in range(n)]
bfs(A, B, dis1)
bfs(C, D, dis2)
res = dis1[C][D]
if res <= E:
  print(res)
else:
  # 枚举所有圣泉
  res = INF
  for i in range(n):
    for j in range(m):
      if g[i][j] == 'V':
        res = min(res, dis1[i][j] + dis2[i][j])
  if res == INF:
    print("No")
  else:
    # 初始能量为E,总距离res, 后面的res-E需要花费两倍时间,因为需要等待能量恢复
    print((res - E) * 2 + E)

拓扑排序

拓扑排序是一种针对“有向无环图”的算法,用于解决一些有依赖关系的问题。

拓扑排序保证了处理到某个点时,其所有的入点已经处理过了。

例如下面这个图,拓扑排序可以保证:

处理点2之前,点4和点6都处理过。

处理点3之前,点2和点6都处理过。

比如学大学物理必须先学高数和线性代数。

拓扑排序的顺序并不是唯一的,就刚刚的例子,你可以先学高数再学线代,也可以先学线代再学高数。

下图是拓扑排序的几种可能性

如果有环的话找不到起点,自己想想应该就能想出来。

BFS实现拓扑排序

  1. 先预处理每个点的入度,这个在读入边的时候处理。
  2. 每次将入度为0的点入队列。
  3. 每次取点u的时候,对于从u出发的所有点v的入度-1

1此时的入度为0,相当于做了一个操作,把1->4和1->6的边给删掉了,然后发现4和6的入度又为0了。

  • 在枚举边u->v的时候,可以进行状态转移,于是可以和动态规划结合起来。
  • 这样的dp也叫DAG-DP(有向无环图上的动态规划)
  • 状态转移一般只发生在枚举所有边的时候。
模板
from collections import deque

def topo():
    q = deque()
    for i in range(1, n + 1):
        if ru[i] == 0:
            q.append(i)
    ans = []
    while len(q) != 0:
        u = q.popleft()
        ans.append(u)
        for v in G[u]:
            ru[v] -= 1
            if ru[v] == 0:
                q.append(v)
    if len(ans) != n:
        print("no topo")
    else:
        print(*ans, sep=' ')
n, m = map(int, input().split())  
G = [[] for i in range(n + 1)]
ru = [0] * (n + 1)
for _ in range(m):
  u, v = map(int, input().split())
  G[u].append(v)
topo()

蓝桥1337

import os
import sys

# 请在此输入您的代码
from collections import deque

def topo():
  q = deque()
  for i in range(1, n + 1):
    if ru[i] == 0:
      q.append(i)
  while len(q) != 0:
    # 取出队首元素
    u = q.popleft()
    # 对于和u相邻的每个点v
    for v in G[u]:
      # 从u走到v,说明dp[v]可以从dp[u] + 1转移过来
      dp[v] = max(dp[v], dp[u] + 1)
      ru[v] -= 1
      if ru[v] == 0:
        q.append(v)

# dp[i] 表示走到i的最长路,也就是最大值。      
n, m = map(int, input().split())
dp = [0] * (n + 1)  
G = [[] for i in range(n + 1)]
ru = [0] * (n + 1)
for _ in range(m):
  u, v = map(int, input().split())
  G[u].append(v)
topo()
print(max(dp))

蓝桥3351

import os
import sys
from queue import PriorityQueue
# 请在此输入您的代码

def topo():
    q = PriorityQueue()
    for i in range(1, n + 1):
        if ru[i] == 0:
            q.put(i)
    ans = []
    while not q.empty():
        u = q.get()
        ans.append(u)
        for v in G[u]:
            ru[v] -= 1
            if ru[v] == 0:
                q.put(v)
    if len(ans) != n:
        print(-1)
    else:
        print(*ans, sep=' ')
n, m = map(int, input().split())
G = [[] for i in range(n + 1)]
ru = [0] * (n + 1)
for _ in range(m):
  u, v = map(int, input().split())
  G[u].append(v)
  ru[v] += 1
topo()

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1556773.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

C语言程序编译和链接

在ANSI C的任何⼀种实现中&#xff0c;存在两个不同的环境。 第1种是翻译环境&#xff0c;在这个环境中源代码被转换为可执⾏的机器指令&#xff08;⼆进制指令&#xff09;。 第2种是执⾏环境&#xff0c;它⽤于实际执⾏代码。 如果再把编译器展开成3个过程&#xff0c;那就变…

基于SpringBoot + Vue实现的中国陕西民俗网设计与实现+毕业论文

介绍 本系统包含管理员、用户两个角色。 管理员角色&#xff1a;登录、用户管理功能、民俗介绍管理功能(发布和管理民俗文化的介绍文章)、公告信息管理功能(发布网站的重要通知和活动信息)、商品管理功能(对商家发布的商品进行监管)、商品评价管理功能(监管商品评价内容&#…

ES6 学习(二)-- 字符串/数组/对象/函数扩展

文章目录 1. 模板字符串1.1 ${} 使用1.2 字符串扩展(1) ! includes() / startsWith() / endsWith()(2) repeat() 2. 数值扩展2.1 二进制 八进制写法2.2 ! Number.isFinite() / Number.isNaN()2.3 inInteger()2.4 ! 极小常量值Number.EPSILON2.5 Math.trunc()2.6 Math.sign() 3.…

C++11 shared_from_this学习

最近学习网络变成发现一些C源码库中封装对象时会公有继承enable_shared_from_this&#xff1b; 用一个案例进行说明&#xff0c;案例代码如下&#xff1a; #include <iostream> #include <memory> #include <stdio.h>using namespace std;class C : public…

【NFS】NFS使用汇总

1. NFS介绍 NFS(Network File System)&#xff0c;网络文件系统&#xff0c;它可以让不同主机能够通过 TCP/IP 网络共享资源。它从宏观主体上简化来看&#xff0c;就是两部分&#xff1a;服务端和客户端。 服务端&#xff0c;可以认为它就是来存东西的&#xff0c;这个东西对…

jupyter 设置工作目录

本博客主要介绍&#xff1a; 如何为jupyter设置工作目录 1.打开 anaconda prompt , 执行 jupyter notebook --generate-config 执行这个命令后会生成一个配置文件 2. 打开jupyter_notebook_config.py文件编辑 搜索notebook_dir&#xff0c;把这行代码的注释取消&#xff0c;…

Pillow教程03:图像处理的基本步骤+分离split+合并merge+混合blend+composite遮罩

--------------Pillow教程集合--------------- Python项目18&#xff1a;使用Pillow模块&#xff0c;随机生成4位数的图片验证码 Python教程93&#xff1a;初识Pillow模块&#xff08;创建Image对象查看属性图片的保存与缩放&#xff09; Pillow教程02&#xff1a;图片的裁剪…

vue3使用富文本编辑器 Editor.js

一、安装 Editor.js npm i editorjs/editorjs --save 二、在页面中引入并使用 样式就不发了&#xff0c;自己按自己的来 三、转换语言&#xff0c;默认是英文 editor new EditorJS({holder: this.$refs.editor,// 自动聚焦autofocus: true,// 其他配置... tools: {},i18…

第十四届蓝桥杯第十题:蜗牛分享

问题描述 输入格式 输出格式 输出共一行&#xff0c;一个浮点数表示答案&#xff08;四舍五入保留两位小数&#xff09;。 样例输入 3 1 10 11 1 1 2 1样例输出 4.20样例说明 蜗牛路线&#xff1a;(0,0)→(1,0)→(1,1)→(10,1)→(10,0)→(11,0)(0,0)→(1,0)→(1,1)→(10,1…

浏览器工作原理与实践--栈空间和堆空间:数据是如何存储的

对于前端开发者来说&#xff0c;JavaScript的内存机制是一个不被经常提及的概念 &#xff0c;因此很容易被忽视。特别是一些非计算机专业的同学&#xff0c;对内存机制可能没有非常清晰的认识&#xff0c;甚至有些同学根本就不知道JavaScript的内存机制是什么。 但是如果你想成…

【aws】架构图工具推荐

碎碎念 以前以为日本冰箱论是个梗&#xff0c;结果居然是真的。用光盘传真其实还能理解&#xff08;毕竟我也喜欢电子古董2333&#xff09;&#xff0c;但是画架构图居然用的是excel&#xff0b;截图&#xff01;啊苍天呐&#xff0c;然后看到隔壁工位用excel画web原型又感觉释…

【力扣】300. 最长递增子序列(DFS+DP两种方法实现)

目录 题目传送最长递增子序列[DFS 方法]DFS方法思路图思路简述代码大家可以自行考虑有没有优化的方法 最长递增子序列[DP]方法DP方法思路图思路简述代码方案 题目传送 原题目链接 最长递增子序列[DFS 方法] DFS方法思路图 思路简述 对于序列中的每一个数字只有选择和不选择两…

Echarts地图之——如何给地图添加外边框轮廓

有时候我们希望给地图外围加一圈边框来增加美感 但实际情况中&#xff0c;我们需要把国界的边框和各个省份属于国界的边框相吻合&#xff0c;否则就会造成两者看起来是错位的感觉 这就需要我们把echarts registerMap的全国省份json和国界边框json的坐标相一致。 这个json我们可…

Java项目实战笔记--基于SpringBoot3.0开发仿12306高并发售票系统--(二)项目实现-第五篇-核心功能车票预定开发及nacos集成

本文参考自 Springboot3微服务实战12306高性能售票系统 - 慕课网 (imooc.com) 本文是仿12306项目实战第&#xff08;二&#xff09;章——项目实现 的第五篇&#xff0c;本篇讲解该项目的核心功能——余票查询、车票预定功能的基础版开发&#xff0c;以及讲解项目与Nacos的集成…

图的基础和图的遍历(--蓝桥云)

图的基础概念 度数&#xff1a;出边入边的条数 有向边&#xff1a;有箭头 图的存储方式 //邻接表 List<int []> list[N] list<x>//存放x的所有出点的信息 list[i][j]{first,second}//其中first表示从i出发的某个出点的编号&#xff08;这个出点是i的第j个出点&…

【Entity Framework】EF中DbSet类详解

【Entity Framework】EF中DbSet类详解 文章目录 【Entity Framework】EF中DbSet类详解一、概述二、定义DbSet2.1 具有DbSet属性的DbContext2.2 具有 IDbSet 属性的 DbContext 2.3 具有 IDbSet 属性的 DbContext三、DbSet属性四、DbSet方法五、DbContext动态生成DbSet 一、概述 …

【JavaSE】java刷题--数组练习

前言 本篇讲解了一些数组相关题目&#xff08;主要以代码的形式呈现&#xff09;&#xff0c;主要目的在于巩固数组相关知识。 上一篇 数组 讲解了一维数组和二维数组的基础知识~ 欢迎关注个人主页&#xff1a;逸狼 创造不易&#xff0c;可以点点赞吗~ 如有错误&#xff0c;欢迎…

JavaEE 初阶篇-深入了解多线程安全问题(出现线程不安全的原因与解决线程不安全的方法)

&#x1f525;博客主页&#xff1a; 【小扳_-CSDN博客】 ❤感谢大家点赞&#x1f44d;收藏⭐评论✍ 文章目录 1.0 多线程安全问题概述 1.1 线程不安全的实际例子 2.0 出现线程不安全的原因 2.1 线程在系统中是随机调度且抢占式执行的模式 2.2 多个线程同时修改同一个变量 2.3 线…

C++基础之虚函数(十七)

一.什么是多态 多态是在有继承关系的类中&#xff0c;调用同一个指令&#xff08;函数&#xff09;&#xff0c;不同对象会有不同行为。 二.什么是虚函数 概念&#xff1a;首先虚函数是存在于类的成员函数中&#xff0c;通过virtual关键字修饰的成员函数叫虚函数。 性质&am…

c语言:用do-while输出前40项的斐波那契数值

求Fibonacci数列的前40个元素。该数列的特点是第1、2两个数为1、1。从第3个数开始&#xff0c;每数是其前两个数之和。 分析&#xff1a;从题意可以用如下等式来表示斐波那契数列&#xff1a; 1&#xff0c; 1&#xff0c; 2&#xff0c; 3&#xff0c; 5&#xff0c; 8&#x…