【图论】(二)图论基础与路径问题

news2024/12/29 8:03:10

图论基础与路径问题

  • 图的构造
    • 邻接矩阵
    • 邻接表
  • 所有可达路径
    • 邻接矩阵存储
    • 邻接表存储
  • 字符串接龙
  • 有向图的完全可达性

图的构造

这里仅对图论路径问题中图的构造做整理总结归纳,具体详细相关概念请参考代码随想录上的整理总结:

  • 图论理论基础
  • 深度优先搜索理论基础
  • 所有可达路径-dfs实战
  • 广度优先搜索理论基础

我们如何用代码来表示一个图呢?

一般使用邻接表、邻接矩阵或者用类来表示。

邻接矩阵

  • 邻接矩阵使用二维数组来表示图结构。 邻接矩阵是从节点的角度来表示图,有多少节点就申请多大的二维数组。
  • 例如: grid[2][5] = 6,表示 节点 2 连接 节点5 为有向图,节点2 指向 节点5,边的权值为6。
  • 如果想表示无向图,即:grid[2][5] = 6,grid[5][2] = 6,表示节点2 与 节点5 相互连通,权值为6
  • 在一个 n (节点数)为8 的图中,就需要申请 8 * 8 这么大的空间。

  • 这种表达方式(邻接矩阵) 在 边少,节点多的情况下,会导致申请过大的二维数组,造成空间浪费。

  • 而且在寻找节点连接情况的时候,需要遍历整个矩阵,即 n * n 的时间复杂度,同样造成时间浪费。

邻接矩阵的优点:

  • 表达方式简单,易于理解
  • 检查任意两个顶点间是否存在边的操作非常快
  • 适合稠密图,在边数接近顶点数平方的图中,邻接矩阵是一种空间效率较高的表示方法。

缺点:

  • 遇到稀疏图,会导致申请过大的二维数组造成空间浪费
  • 遍历 边 的时候需要遍历整个n * n矩阵,造成时间浪费

邻接表

  邻接表 使用 数组 + 链表的方式来表示。 邻接表是从边的数量来表示图,有多少边 才会申请对应大小的链表。

邻接表的构造如图:


这里表达的图是:

  • 节点1 指向 节点3 和 节点5
  • 节点2 指向 节点4、节点3、节点5
  • 节点3 指向 节点4
  • 节点4指向节点1

有多少边 邻接表才会申请多少个对应的链表节点。

从图中可以直观看出 使用 数组 + 链表 来表达 边的连接情况

邻接表的优点:

  • 对于稀疏图的存储,只需要存储边,空间利用率高
  • 遍历节点连接情况相对容易

缺点:

  • 检查任意两个节点间是否存在边,效率相对低,需要 O(V)时间,V表示某节点连接其他节点的数量。
  • 实现相对复杂,不易理解

所有可达路径

卡码网题目链接(ACM模式)

力扣题目链接 - 797. 所有可能的路径

题目描述:
  给定一个有 n 个节点的有向无环图,节点编号从 1 到 n。请编写一个函数,找出并返回所有从节点 1 到节点 n 的路径。每条路径应以节点编号的列表形式表示。

输入描述:

  • 第一行包含两个整数 N,M,表示图中拥有 N 个节点,M 条边
  • 后续 M 行,每行包含两个整数 s 和 t,表示图中的 s 节点与 t 节点中有一条路径

输出描述: 输出所有的可达路径,路径中所有节点的后面跟一个空格,每条路径独占一行,存在多条路径,路径输出的顺序可任意。如果不存在任何一条路径,则输出 -1。

注意输出的序列中,最后一个节点后面没有空格! 例如正确的答案是 1 3 5,而不是 1 3 5 5后面没有空格!

输入示例:

5 5
1 3
3 5
1 2
2 4
4 5

输出示例:

1 3 5
1 2 4 5  

提示信息:


写在前面:

  • 在上述谈到图的两张存储方式中,这里通过两种方式逐一巩固。
  • 此外,本题是目是深度优先搜索比较好的入门题,若对深搜不清楚的需先提前了解深度搜索理论与过程。

邻接矩阵存储

  • 邻接矩阵 使用 二维数组来表示图结构, 邻接矩阵是从节点的角度来表示图,有多少节点就申请多大的二维数组。
  • 本题有n 个节点,因为节点标号是从1开始的,为了节点标号和下标对齐,我们申请 n + 1 * n + 1 这么大的二维数组。
    vector<vector<int>> graph(n + 1, vector<int>(n + 1, 0));
    
  • 输入每行两个整数 s 和 t,表示图中的 s 节点与 t 节点中有一条路径,则表示 graph[s][t] = 1;,输入m条边程序则有:
    while (m--) {
        cin >> s >> t;
        // 使用邻接矩阵 ,1 表示 节点s 指向 节点t
        graph[s][t] = 1;
    }
    

深搜三部曲:

1. 确认递归函数,参数

  • 首先我们dfs函数一定要存一个图,用来遍历,需要存一个目前我们遍历的节点,定义为x
  • 还需要存一个n,表示终点,我们遍历的时候,用来判断当 x==n 时候 标明找到了终点
  • 至于 单一路径 和 路径集合 可以放在全局变量,那么代码是这样的:
    vector<vector<int>> result; 	// 收集符合条件的路径
    vector<int> path; 				// 0节点到终点的路径
    // x:目前遍历的节点
    // graph:存当前的图
    // n:终点
    void dfs (const vector<vector<int>>& graph, int x, int n) {
    

2. 确认终止条件

当目前遍历的节点 为 最后一个节点 n 的时候 就找到了一条 从出发点到终止点的路径。

// 当前遍历的节点x 到达节点n 
// 找到符合条件的一条路径
if (x == n) 
{ 
    result.push_back(path);
    return;
}

3. 处理目前搜索节点出发的路径

接下来是走 当前遍历节点x的下一个节点。

首先是要找到 x节点指向了哪些节点呢? 遍历方式是这样的:

for (int i = 1; i <= n; i++) 	// 遍历节点x链接的所有节点
{ 
    if (graph[x][i] == 1) 		// 找到 x指向的节点,就是节点i
    { 	
    	
    }
}

接下来就是将 选中的x所指向的节点,加入到 单一路径来

path.push_back(i); // 遍历到的节点加入到路径中来

进入下一层递归

dfs(graph, i, n); // 进入下一层递归

最后就是回溯的过程,撤销本次添加节点的操作

path.pop_back(); // 回溯,撤销本节点

该过程整体代码:

for (int i = 1; i <= n; i++) 	// 遍历节点x链接的所有节点
{ 
    if (graph[x][i] == 1)		// 找到 x链接的节点
     {
        path.push_back(i); 		// 遍历到的节点加入到路径中来
        dfs(graph, i, n); 		// 进入下一层递归
        path.pop_back(); 		// 回溯,撤销本节点
    }
}

程序实现:

#include <iostream>
#include <vector>
#include <algorithm>
#include <queue>

using namespace std;


vector<vector<int>> result;		// 手机符合条件的路径
vector<int> path;				//0节点到终点的路径
// graph:存当前的图
// x:目前遍历的节点
// n:终点
void dfs(const vector<vector<int>> graph, int x, int n)
{
	// 当前遍历的节点x 到达节点n 
	if(x == n)
	{
		result.push_back(path);
		return;
	}
	// 遍历节点x链接的所有节点
	for(int i = 1; i <= n; i++)
	{
		// 找到 x指向的节点,就是节点i
		if(graph[x][i] == 1)
		{
			// 将选中的x所指向的节点,加入到 单一路径来。
			path.push_back(i);
			// 进入下一层递归
			dfs(graph,i,n);
			// 回溯的过程,撤销本次添加节点的操作
			path.pop_back();
		}
	}
}

int main()
{
	// m条边 n个节点 
	// graph[s][t] = 1表示节点s可以指向t
	int m,n,s,t;
	cin >> n >> m;
	vector<vector<int>> graph(n+1, vector<int>(n+1,0));
	while(m--)
	{
		cin >> s >> t;
		// 使用邻接矩阵 表示无线图,1 表示 s 与 t 是相连的
		graph[s][t] = 1;
	}
	
	// 任何路径都是从节点1开始
	path.push_back(1);
	// 开始遍历
	dfs(graph,1,n);
	
	// 打印结果
	if(result.size() == 0)
		cout << -1 << endl;
	for(int i = 0; i < result.size(); i++)
	{
		for(int j = 0; j < result[i].size() - 1; j++)
		{
			cout << result[i][j] << " ";
		}
		cout << result[i][result[i].size() - 1] << endl;
	}
	
	return 0;
}

邻接表存储

  用邻接表存储和邻接矩阵存储的区别只需要修改图的定义存储、dfs函数内的参数以及对链表的遍历,核心框架、思路均不变。

程序实现:

//邻接表法
vector<vector<int>> result;		// 手机符合条件的路径
vector<int> path;				//0节点到终点的路径
// graph:存当前的图
// x:目前遍历的节点
// n:终点
void dfs(const vector<list<int>> graph, int x, int n)
{
	// 当前遍历的节点x 到达节点n 
	if(x == n)
	{
		result.push_back(path);
		return;
	}
	// 遍历节点x链接的所有节点
	for(int node: graph[x])
	{
		// 将选中的x所指向的节点,加入到 单一路径来。
		path.push_back(node);
		// 进入下一层递归
		dfs(graph,node,n);
		// 回溯的过程,撤销本次添加节点的操作
		path.pop_back();
	}
}

int main()
{
	// m条边 n个节点 
	// graph[s][t] = 1表示节点s可以指向t
	int m,n,s,t;
	cin >> n >> m;
	vector<list<int>> graph(n+1);
	while(m--)
	{
		cin >> s >> t;
		// 使用邻接矩阵 表示无线图,1 表示 s 与 t 是相连的
		graph[s].push_back(t);
	}
	
	// 任何路径都是从节点1开始
	path.push_back(1);
	// 开始遍历
	dfs(graph,1,n);
	
	// 打印结果
	if(result.size() == 0)
		cout << -1 << endl;
	for(int i = 0; i < result.size();i++)
	{
		for(int j = 0; j < result[i].size() - 1; j++)
		{
			cout << result[i][j] << " ";
		}
		cout << result[i][result[i].size() - 1] << endl;
	}
	
	return 0;
}

字符串接龙

卡码网题目链接(ACM模式)

题目描述:

字典 strList 中从字符串 beginStr 和 endStr 的转换序列是一个按下述规格形成的序列:

  • 序列中第一个字符串是 beginStr
  • 序列中最后一个字符串是 endStr
  • 每次转换只能改变一个字符。
  • 转换过程中的中间字符串必须是字典 strList 中的字符串。

  给你两个字符串 beginStrendStr 和一个字典 strList,找到从 beginStrendStr 的最短转换序列中的字符串数目。如果不存在这样的转换序列,返回 0

输入描述:

  第一行包含一个整数 N,表示字典 strList 中的字符串数量。 第二行包含两个字符串,用空格隔开,分别代表 beginStr 和 endStr。 后续 N 行,每行一个字符串,代表 strList 中的字符串。

输出描述:

  输出一个整数,代表从 beginStr 转换到 endStr 需要的最短转换序列中的字符串数量。如果不存在这样的转换序列,则输出 0。

输入示例:

6
abc def
efc
dbc
ebc
dec
dfc
yhn

输出示例:

4

提示信息: 从 startStr 到 endStr,在 strList 中最短的路径为 abc -> dbc -> dec -> def,所以输出结果为 4


本题只需要求出最短路径的长度就可以了,不用找出具体路径。

所以这道题要解决两个问题:

  • 图中的线是如何连在一起的
  • 起点和终点的最短路径长度

首先题目中并没有给出点与点之间的连线,而是要我们自己去连,条件是字符只能差一个。

所以判断点与点之间的关系,需要判断是不是差一个字符,如果差一个字符,那就是有链接。

然后就是求起点和终点的最短路径长度,这里无向图求最短路,广搜最为合适,广搜只要搜到了终点,那么一定是最短的路径。 因为广搜就是以起点中心向四周扩散的搜索。

本题如果用深搜,会比较麻烦,要在到达终点的不同路径中选则一条最短路。 而广搜只要达到终点,一定是最短路。

另外需要有一个注意点:

  • 本题是一个无向图,需要用标记位,标记着节点是否走过,否则就会死循环!
  • 使用set来检查字符串是否出现在字符串集合里更快一些

程序实现:

#include <iostream>
#include <string>
#include <queue>
#include <unordered_set>
#include <unordered_map>

using namespace std;

int main()
{
	int n;
	string beginStr;
	string endStr;
	string str;
	unordered_set<string> strSet;	// 字典 存放每一个字符串
	
	cin >> n;
	cin >> beginStr >> endStr;
	
	for(int i = 0; i < n;i++){
		cin >> str;
		strSet.insert(str);
	}
	
	// 记录strSet里的字符串是否被访问过,同时记录路径长度
	unordered_map<string, int> visitMap; // <记录的字符串,路径长度>
	
	// 初始化队列
	queue<string> que;
	que.push(beginStr);
	
	// 加入队列即标记访问
	visitMap.insert(pair<string,int>(beginStr, 1));
	
	while(!que.empty())
	{
		// 获取队头字符串,拿出来
		string word = que.front();
		que.pop();
		// 这个字符串在路径中的长度为 visitMap[word]
		int path = visitMap[word];
		
		// 开始在这个str中,挨个字符去替换
		for(int i = 0; i < word.size(); i++)
		{
			// 用一个新字符串替换str,因为每次要置换一个字符
			string newWord = word;
			// 遍历26的字母
			for(int j = 0; j < 26; j++)
			{
				newWord[i] = j + 'a';
				// 发现替换字母后,字符串与终点字符串相同
				if(newWord == endStr)
				{
					cout <<  path + 1 << endl; // 找到了路径 
					return 0;
				}
				// 字符串集合里出现了newWord,并且newWord没有被访问过
				if (strSet.find(newWord) != strSet.end() 
					&& visitMap.find(newWord) == visitMap.end())
				{
					// 添加访问信息,并将新字符串放到队列中
					visitMap.insert(pair<string, int>(newWord, path + 1));
					 que.push(newWord);
				}
			}
		}
	}
	// 没找到输出0
	return 0;
}

有向图的完全可达性

卡码网题目链接(ACM模式)

题目描述:

  给定一个有向图,包含 N 个节点,节点编号分别为 1,2,…,N。现从 1 号节点开始,如果可以从 1 号节点的边可以到达任何节点,则输出 1,否则输出 -1。

输入描述:

  第一行包含两个正整数,表示节点数量 N 和边的数量 K。 后续 K 行,每行两个正整数 s 和 t,表示从 s 节点有一条边单向连接到 t 节点。

输出描述: 如果可以从 1 号节点的边可以到达任何节点,则输出 1,否则输出 -1。

输入示例:

4 4
1 2
2 1
1 3
2 4

输出示例:

1

提示信息:


思路

本题给我们是一个有向图, 因此路径是存在方向的

递归三部曲:

1. 确认递归函数,参数

需要传入地图,当前的节点数(到当前节点是通路),同时还需要一个数组,用来记录我们都走过了哪些节点,这样好知道最后有没有把所有房间都遍历的。
所以 递归函数参数如下:

// key 当前的节点数(到当前节点是通路)
// visited 记录访问过的房间 
void dfs(const vector<list<int>>& graph, int key, vector<bool>& visited) {

2. 确认终止条件

遍历的时候,什么时候终止呢?这里有一个很重要的逻辑,就是在递归中,是处理当前访问的节点,还是处理下一个要访问的节点

如果我们是处理当前访问的节点,当前访问的节点如果是 true ,说明是访问过的节点,那就终止本层递归,如果不是true,我们就把它赋值为true,因为这是我们处理本层递归的节点。

代码就是这样的:

// 写法一:处理当前访问的节点
void dfs(const vector<list<int>>& graph, int key, vector<bool>& visited) {
    if (visited[key]) {
        return;
    }
    visited[key] = true;
    list<int> keys = graph[key];	// 获取当前节点连接的节点链表
    for (int key : keys) {
        // 深度优先搜索遍历
        dfs(graph, key, visited);
    }
}

如果我们是处理下一层访问的节点,而不是当前层。那么就要在 深搜三部曲中第三步:处理目前搜索节点出发的路径的时候对 节点进行处理。

这样的话,就不需要终止条件,而是在 搜索下一个节点的时候,直接判断 下一个节点是否是我们要搜的节点。

代码就是这样的:

// 写法二:处理下一个要访问的节点
void dfs(const vector<list<int>>& graph, int key, vector<bool>& visited) {
    list<int> keys = graph[key];	// 获取当前节点连接的节点链表
    for (int key : keys)
     {
      	// 确认下一个是没访问过的节点
        if(visited[key] == false) {
            visited[key] = true;
            dfs(graph, key, visited);
        }
    }
}

可以看出,如何看待 我们要访问的节点,直接决定了两种不一样的写法

上述程序中,好像没有发现回溯的逻辑。本题是需要判断 1节点 是否能到所有节点,那么我们就没有必要回溯去撤销操作了,只要遍历过的节点一律都标记上。

那什么时候需要回溯操作呢?
当我们需要搜索一条可行路径的时候,就需要回溯操作了,因为没有回溯,就没法“调头”,

程序实现:

#include <iostream>
#include <vector>
#include <list>

using namespace std;

// key 当前得到的可以 
// visited 记录访问过的房间 
void dfs(vector<list<int>>& graph, int key, vector<bool>& visited)
{
	list<int> keys = graph[key];	// 获取当前节点下挂的节点链表
	for(int key: keys)
	{
		if(visited[key] == false)	// 遍历每一个节点 并做dfs标记
		{
			visited[key] = true;
			dfs(graph,key,visited);
		}
	}
}

int main()
{
	int n, m, s, t;
	cin >> n >> m;
	
	vector<list<int>> graph(n+1);

	while(m--){
		cin >> s >> t;
		graph[s].push_back(t);
	}
	
	vector<bool> visited(n + 1, false);
	visited[1] = true; 			// 节点1 预先处理
	dfs(graph, 1, visited);		// dfs计算能到达的最远节点,能到达并做标记
	
	for(int i = 1; i <= n; i++)
	{
		if(visited[i] == false)
		{
			cout << -1 << endl;	
			return 0;
		}
	}
	
	cout << 1 << endl;	
	return 0;
}

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

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

相关文章

C语言函数栈帧的创建与销毁(32)

文章目录 前言一、什么是函数栈帧&#xff1f;二、理解函数栈帧能解决什么问题&#xff1f;三、函数栈帧的创建和销毁解析什么是栈&#xff1f;认识相关寄存器和汇编指令 四、解析函数栈帧的创建和销毁预备知识函数的调用堆栈准备环境转到反汇编函数栈帧的创建函数栈帧的销毁 五…

采用反相正基准电压电路的反相运算放大器

1 简介 本设计使用采用反相正基准电压的反相放大器将 –5V 至 –1V 的输入信号转换为 3.3V 至 0.05V 的输出电压。该电路可用于将传感器负输出电压转换为可用的 ADC 输入电压范围。 2 设计目标 2.1 输入 2.2 输出 2.3 电源 3 电路设计 根据设计目标&#xff0c;最终设计的电…

2.1类和对象(上)

本篇博客来梳理类和对象的基础知识 一、类的定义 1&#xff0e;类定义格式 &#xff08;1&#xff09;关键字&#xff1a;class。类中的变量称为类的属性/成员变量&#xff0c;类中的函数称为类的方法/成员函数 &#xff08;2&#xff09;为区分成员变量&#xff0c;一般会加…

MES管理系统对中小企业有哪些帮助

MES管理系统解决方案对中小企业具有显著的帮助&#xff0c;主要体现在以下几个方面&#xff1a; 一、提升生产效率 MES管理系统能够实时监控生产过程&#xff0c;提供准确的生产数据和及时的反馈。这种实时监控与数据分析能力&#xff0c;使中小企业能够精准把握生产脉搏&…

如何应对动态图片大小变化?Python解决网页图片截图难题

背景介绍 随着互联网的发展&#xff0c;许多网站&#xff0c;尤其是电商平台&#xff0c;如京东&#xff08;JD.com&#xff09;&#xff0c;为了提升用户体验&#xff0c;采用了许多动态内容加载技术。当我们使用爬虫获取商品图片时&#xff0c;往往会遇到一些棘手问题&#…

中科星图GVE(案例)——AI提取指定采样区域的建筑物范围

目录 简介 函数 gve.Image.fromGeometry(geometry,source,options) gve.Services.AI.buildingExtraction(fromGridRes) 代码 结果 ​编辑 知识星球 机器学习 简介 要提取指定采样区域的建筑物范围&#xff0c;可以使用遥感图像处理和计算机视觉技术。以下是一种可能的…

软考攻略/超详细/系统集成项目管理工程师/基础知识分享14

5.4 软件实现 5.4.1 软件配置管理&#xff08;掌握&#xff09; 软件配置管理&#xff08;SCM&#xff09;是一种标识、组织和控制修改的技术。软件配置管理应用于整个软件工程过程。 SCM活动的目标就是标识变更、控制变更、确保变更正确 SCM的目的是使错误降为最小&#xff0…

申报审批|基于springBoot的入校申报审批系统设计与实现(附项目源码+论文+数据库)

私信或留言即免费送开题报告和任务书&#xff08;可指定任意题目&#xff09; 目录 一、摘要 二、相关技术 三、系统设计 四、数据库设计 五、核心代码 六、论文参考 七、源码获取 一、摘要 传统办法管理信息首先需要花费的时间比较多&#xff0c;其次数据出…

数据库中间件 -- MyCat

1、什么是数据库中间件 数据库中间件(Database Middleware)是一种位于应用程序与数据库管理系统(DBMS)之间的软件层。它的主要目的是为应用程序提供更加高效、可靠和透明的数据库访问,同时解决多种数据库管理问题。 The domain name Mycat.io is for sale 1.1、常见的数…

新质生产力在制造业中的“新”主要体现在哪

新质生产力&#xff0c;以其独特的技术创新、模式变革和思维升级&#xff0c;正逐步重塑制造业的面貌&#xff0c;引领其走向更加智能化、绿色化和高效化的未来。 一、技术创新&#xff1a;驱动产业升级的核心引擎 新质生产力在制造业中的首要“新”&#xff0c;体现在技术创新…

Chromium 书签加载过程分析c++

一、书签存储路径: %localappdata%\Chromium\User Data\Default\Bookmarks %localappdata%\Chromium\User Data\Default\Bookmarks.bak 【备份文件】 本机测试存储的Bookmarks文件如下&#xff08;未加密的可以直接打开&#xff09;&#xff1a; {"checksum": &q…

Allegro平台正式进军匈牙利市场,Allegro怎么快速上架产品?

近日&#xff0c;波兰电商平台Allegro正式宣布进军匈牙利市场&#xff0c;此举标志着Allegro在中东欧地区的扩张步伐再次加速。作为一家在波兰本土享有盛誉的电商平台&#xff0c;Allegro此举无疑为匈牙利乃至整个中欧地区的电商市场注入了新的活力。 Allegro此次进军匈牙利市…

比较三组迭代次数的变化

(A,B)---6*30*2---(0,1)(1,0) 让A是结构5&#xff0c;让B全是0。收敛误差为7e-4&#xff0c;收敛199次取迭代次数平均值&#xff0c;得到迭代次数为129535.3 (A,B)---6*30*2&#xff08;5&#xff09;---(0,1)(1,0) 然后让A分别是0&#xff0c;1&#xff0c;2&#xff0c;3&a…

服装生产管理:SpringBoot技术实现

1 绪论 1.1 研究背景 当今时代是飞速发展的信息时代。在各行各业中离不开信息处理&#xff0c;这正是计算机被广泛应用于信息管理系统的环境。计算机的最大好处在于利用它能够进行信息管理。使用计算机进行信息控制&#xff0c;不仅提高了工作效率&#xff0c;而且大大的提高…

通过AI技术克服自动化测试难点(上)

本文我们一起分析一下AI技术如何解决现有的自动化测试工具的不足和我们衍生出来的新的测试需求。 首先我们一起看一下计算机视觉的发展历史&#xff0c;在上世纪70年代&#xff0c;处于技术萌芽期&#xff0c;由字符的识别技术慢慢进行演化&#xff0c;发展到现在&#xff0c;人…

C/S模型的简单实现(UDP服务器)、本地套接字(sockaddr_un )的讲解

目录 1.UDP 1.1 UDP服务器 1.2 TPC和UDP的比较 1.3 C/S模型 -- UDP recvfrom、sendto server client 2.本地套接字 2.1 套接字比较 2.2 函数参数选用 2.3 server 2.4 client 2.5 实现对比 1.UDP 1.1 UDP服务器 UDP 是一种无连接的传输协议&#xff0c;类似于发送…

SpringBoot MyBatis连接数据库设置了encoding=utf-8还是不能用中文来查询

properties的MySQL连接时已经指定了字符编码格式&#xff1a; url: jdbc:mysql://localhost:3306/sky_take_out?useUnicodetrue&characterEncodingutf-8使用MyBatis查询&#xff0c;带有中文参数&#xff0c;查询出的内容为空。 执行的语句为&#xff1a; <select id&…

已经被这几种广告彻底逼疯……还好有救了

这个假期回家团聚&#xff0c;爸妈小心翼翼问我手机越来越难用了&#xff0c;让我帮忙看看是不是中病毒了&#xff0c;了解后才知道原来事情是这样的&#xff1a; 以前开屏广告不小心误触已经让人恼火&#xff0c;现在是手机轻微动一动就会进入广告&#xff0c;打开app最后都不…

quantlab_ai版本v0.1代码发布: 从研报中提取因子并建模(附代码与研报集下载)

原创内容第676篇&#xff0c;专注量化投资、个人成长与财富自由。 今天我们继续开发AI大模型自动读研报。 从研报到模型&#xff0c;大致分成几步&#xff1a; ["propose_hypo_exp", "propose", "exp_gen", "coding", "runnin…

【玩转 JS 函数式编程_010】3.2 JS 函数式编程筑基之:以函数式编程的方式活用函数(上)

写在前面 按照惯例&#xff0c;过长的篇幅分开介绍&#xff0c;本篇为 JavaScript 函数式编程核心基础的第二部分——以函数式编程的方式活用函数的上篇&#xff0c;分别介绍了 JS 函数在排序、回调、Promise 期约、以及连续传递等应用场景下的用法演示。和之前章节相比难度又有…