算法基础课第一部分

news2024/12/26 0:21:05

算法基础课

  • 第一讲 基础算法
    • 快速排序
    • 归并排序
    • 二分
      • 整数二分模板
      • AcWing 789. 数的范围(整数二分法)
      • AcWing 1236.递增三元组
      • AcWing 730. 机器人跳跃问题
      • AcWing 1227. 分巧克力
      • AcWing 1221. 四平方和(二分法/哈希)
      • 蓝桥杯-扫地机器人 (二分+贪心)
      • AcWing 790. 数的三次方根(浮点二分法)
      • AcWing 680. 剪绳子(浮点二分法)
    • 高精度
    • 前缀
      • 一维前缀和
      • AcWing 795. 前缀和
      • AcWing 3956. 截断数组
      • AcWing 1230. K倍区间(一维前缀和+同余定理)
      • 202012-2csp-期末预测之最佳阈值(排序+一维前缀和)
      • 二维前缀和
      • AcWing 796. 子矩阵的和
      • AcWing 126. 最大的和
      • AcWing 99. 激光炸弹(二维前缀和+边界处理)
      • 202104-2-csp-邻域均值(二维前缀和+边界处理)
    • 差分
      • 一维差分
            • 同时改变一个区间中数的大小
      • AcWing 797. 差分
      • AcWing 3729. 改变数组元素
      • 202203-2csp-出行计划
      • AcWing 100. 增减序列
      • AcWing 101. 最高的牛(差分+区间处理)
            • 水位每上涨一个高度差后对数组中数的关系有怎样的影响
      • 202109-2-csp-非零段划分
      • AcWing 2014. 岛(贪心+模拟)
      • 二维差分
      • AcWing 798. 差分矩阵(二维差分)
      • 201409-2-csp-画图---二维差分
    • 快速排序
    • 双指针算法
      • AcWing 799. 最长连续不重复子序列---滑动窗口
      • AcWing 3768. 字符串删减---滑动窗口
      • AcWing 1238. 日志统计---滑动窗口
      • AcWing 800. 数组元素的目标和---对撞指针---操作两个数组
      • AcWing 2816. 判断子序列---操作两个数组
      • AcWing1532. 找硬币---对撞指针
      • AcWing1574.接雨水---对撞指针
    • 位运算
    • 离散化
  • 第二讲 数据结构
    • 集合
      • AcWing 3542. 查找
    • 单链表
    • 双链表
    • 栈(stack)
      • AcWing 3302. 中缀表达式求值---向零取整
      • 201903-2-csp-二十四点---向下取整
      • 中缀表达式转后缀表达式---AcWing 3302. 中缀表达式求值---改编题目
    • 队列(queue/deque/priority_queue)
      • 201712-2csp游戏(queue)---约瑟夫问题
    • 单调栈
    • 单调队列
    • KMP
    • Trie
    • 并查集
      • AcWing 837. 连通块中点的数量
    • 哈希表(map/unordered_map)
      • 201412-1csp门禁系统(map)
      • 201409-1csp相邻数对(map)
      • 201312-1csp出现次数最多的数(map)
      • 202006-2csp稀疏向量(map)
      • AcWing.3447. 子串计算
      • AcWing 3581. 单词识别
  • 第三讲 搜索与图论
    • leetcode
        • leetcode.200岛屿数量(bfs/dfs)
        • leetcode.130被围绕的区域(dfs,bfs)
        • leetcode.547省份数量(dfs,bfs)
    • Flood fill算法---洪水覆盖算法
      • AcWing1113. 红与黑
      • AcWing2060. 奶牛选美---尽可能少的区域内涂色---Flood fill算法---枚举
      • 201512-3-csp-画图---洪水覆盖算法
    • DFS---深度优先搜索
      • AcWing 842. 排列数字
      • AcWing 843. n-皇后问题
      • AcWing 1432. 棋盘挑战---对角线---八皇后问题
      • 201709-4-csp-通信网络---正反向两次dfs---图论
      • 201312-5-csp-I’m stuck!---正反两个方向dfs
      • 201503-4-csp-网络延时---图论
    • BFS---广度优先搜索
      • AcWing 844. 走迷宫---最短路(权重相同时)
      • 201604-4-csp-游戏---最短路---拆点(通过升维来对状态点进行细分)
      • 201409-4-csp-最优配餐---最短路
      • 201403-4-csp-无线网络--拆点(通过升维来对状态点进行细分)
      • 201509-4-csp-高速公路
      • AcWing1101. 献给阿尔吉侬的花束---最短路
      • AcWing 845. 八数码---至少需要进行多少次交换
      • AcWing 3385. 玛雅人的密码
    • 拓扑排序
      • AcWing 848. 有向图的拓扑序列
    • 最短路径之Dijkstra算法
      • AcWing 849. Dijkstra求最短路 I---(重边√有环√---非负---有向/无向图√)---应用于稠密图
      • AcWing 850. Dijkstra求最短路 II------(重边√有环√---非负---有向/无向图√)---应用于稀疏图
    • bellman-ford---有边数限制的最短路问题
      • AcWing 853. 有边数限制的最短路
    • spfa
      • AcWing 851. spfa求最短路-带负权---有点小小万能的感觉?
      • 201609-4-csp-交通规划
      • 201903-5-csp-317号子任务
      • AcWing 3255. 行车路线
      • AcWing 852. spfa判断负环
    • 最小生成树
    • 欧拉路径
    • 染色法判定二分图
    • 匈牙利算法

第一讲 基础算法

快速排序

归并排序

二分

整数二分模板

关键------画一个仅有整数的一维横轴

bool check(int x) {/* ... */} // 检查x是否满足某种性质   // check()判断mid是否满足性质




// 区间[l, r]被划分成[l, mid]和[mid + 1, r]时使用:
int bsearch_1(int l, int r)//右半区间的左端点==mid
{
    while (l < r)
    {
        int mid = l + r >> 1;
        if (check(mid)) r = mid; 
        else l = mid + 1;
    }
    return l;
}




// 区间[l, r]被划分成[l, mid - 1]和[mid, r]时使用:
int bsearch_2(int l, int r)//左半区间的右端点==mid 
{
    while (l < r)
    {
        int mid = l + r + 1 >> 1;//当l=mid时答案在右区间,为了避免进入死循环————[m,r]更新过后m会一直等于m(m+1==r的情况) 所以要上取整 
        if (check(mid)) l = mid;
        else r = mid - 1;
    }
    return l;
}


AcWing 789. 数的范围(整数二分法)

AcWing 789. 数的范围


#include <iostream>
#include <cstring>
#include <algorithm>
using namespace std;

const int N = 100010;

int n,q;
int a[N];
int main()
{
    cin>>n>>q;
    for (int i = 0; i < n; i ++ ) cin >> a[i];

    while(q--)
    {
        int k;
        cin >> k;
        int l=0,r=n-1;
        while(l<r)//右半区间的左端点
        {
            int mid=(l+r)/2;//l+r>>1;
            if(a[mid]<k)//此处的符号<不能变
                l=mid+1;
            else
                r=mid;//此时a[mid]>=k
                
        }
        if(a[l]!=k)//此时说明该序列中不含有k 
           cout<<-1<<" "<<-1<<endl; 
        else
            {
                cout <<l<<" ";
                l=0,r=n-1;
                while (l<r)//左半区间的右端点
                {
                    int mid=(l+r+1)/2;
                    if(a[mid]<=k)//此处的符号<=不能变
                        l=mid;
                    else
                        r=mid-1;//此时a[mid]>k
                        
                }
                cout <<l<<endl;
            }
            
        
    }
    
    
}
 /*
为什么需要+1?
原因是如果不加上1,那么mid得到的是下取整的数,
那么有可能[m,r]更新过后m会一直等于m(m+1==r的情况)会陷入死循环。
*/

AcWing 1236.递增三元组

递增三元组

#include<iostream>
#include<algorithm>
#include<cstring>
using namespace std;
typedef long long LL;
const int N=1e5+10;
const int INF=0x3f3f3f3f;
int n;
int a[N],b[N],c[N];
LL res;
int main()
{
	cin>>n;
	for(int i=1;i<=n;i++) cin>>a[i]; 
	for(int i=1;i<=n;i++) cin>>b[i]; 
	for(int i=1;i<=n;i++) cin>>c[i]; 
	 
	//由于二分的前提是单调序列 所以预先对a b c排序 直接sort
	sort(a+1,a+n+1);
	sort(b+1,b+n+1);
	sort(c+1,c+n+1);
	
		
	for(int i=1;i<=n;i++)
	{
	//直接用STL中的两个二分函数解决
	LL x = (lower_bound(a+1,a+1+n,b[i])-(a+1));     //在数组a中找比b[i]小的数  的个数x 
    LL y = n-(upper_bound(c+1,c+1+n,b[i])-(c+1)); //在数组c中找比b[i]大的数  的个数y 
    res+=x*y;
	}
	cout<<res<<endl;
	return 0;
 } 

AcWing 730. 机器人跳跃问题

AcWing 730. 机器人跳跃问题
在这里插入图片描述视频讲解

且答案具有单调性,证明可以使用二分法解决
易知,h[0]越大,整个过程中所有的e越大,因此,具有单调性
剪枝 :因为h[i]的范围是 [ 1 , 1 e 5 ] [1,1e5] [1,1e5],因此根据公式,只要在某一状态下e达到1e5,之后一定递增或不变,直接返回true

在这里插入图片描述


#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;

const int N = 1e5+10;
int n;
int h[N];
bool check(int mid)
{
    for (int i = 1; i <= n; i ++ )
    {
        mid=mid*2-h[i];//每次登录到新的柱子后的能量
        if(mid<0)  return false;//说明下一个柱子一定跳不上去
        if(mid>=N)  return true;//此时后面一定全部满足
    }
    return true;
}
int main()
{
    cin>>n;
    for (int i = 1; i <= n; i ++ )cin>>h[i];
    int l=0,r=N;
    while(l<r)
    {
        int mid=(l+r)/2;
        if(check(mid)) r=mid;
        else
            l=mid+1;
    }
    cout <<l<<endl;
    return 0;
    
}

AcWing 1227. 分巧克力

AcWing 1227. 分巧克力
视频讲解
在这里插入图片描述


#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int N = 1e5+10;
int n;
int k;
int h[N],w[N];
bool check(int mid)
{
    LL res=0;//表示可以切割出来的矩形个数
    for(int i=0;i<n;i++)//枚举一下所有的巧克力块
    {
        res+=(LL)(h[i]/mid)*(w[i]/mid);
        if(res>=k)
            return true;
    }
    return false;
    
}
int main()
{
    cin>>n>>k;
    for(int i=0;i<n;i++)
    {
        cin>>h[i]>>w[i];
    }
    int l=1,r=1e5;
    while(l<r)
    {
        int mid=(l+r+1)/2;//此时 mid属于有区间,为了避免死循环
        if(check(mid))
            l=mid;
        else
            r=mid-1;
    }
    cout<<l<<endl;
    return 0;
}

AcWing 1221. 四平方和(二分法/哈希)

AcWing 1221. 四平方和
在这里插入图片描述

视频讲解

在这里插入图片描述
在这里插入图片描述

完全暴力写法 一定会超时

#include <cstring>
#include <iostream>
#include <algorithm>
#include <cmath>

using namespace std;

const int N = 2500010;

int n;

int main()
{
    cin >> n;
    for (int a = 0; a * a <= n; a ++ )
        for (int b = a; a * a + b * b <= n; b ++ )
            for (int c = b; a * a + b * b + c * c <= n; c ++ )
            {
                int t = n - a * a - b * b - c * c;
                int d = sqrt(t);//开平方
                if (d * d == t)
                {
                    printf("%d %d %d %d\n", a, b, c, d);
                    return 0;
                }
            }
}

//超时 

哈希表的写法,但超时了

#include<bits/stdc++.h>
using namespace std;
typedef pair<int, int> PII;
#define x first
#define y second

const int N = 5e6+10;
unordered_map<int, PII> S;
int n;
int main()
{
    cin >> n;
   for(int c=0;c*c<=n;c++)
        for(int d=c;d*d<=n;d++)
        {
            int t=c*c+d*d;
            if(S.count(t)==0)//如果之前没有出现过此结果,则添加入
                S[t]={c,d};
        }
    for(int a=0;a*a<=n;a++)
        for(int b=a;b*b<=n;b++)
        {
            int t=n-a*a-b*b;
            if(S.count(t))
            {
                 cout<<a<<" "<<b<<" "<<S[t].x<<" "<<S[t].y<<endl;
                 return 0;
            }
               
            
        }
    return 0;
}

蓝桥杯-扫地机器人 (二分+贪心)

扫地机器人(二分+贪心)
AcWing3176.扫地机器人

#include<bits/stdc++.h>
using namespace std;
const int N=1e5+10;
int n,k;
int w[N];

bool check(int x)
{
	int total=0;
	for(int i=1;i<=k;i++)
	{
		if(total>=w[i]-x)
		{
			if(total>=w[i])
				total=w[i]+x;
			else
				total+=x;
		}
		else
			return false;
	}
	if(total>=n)
		return true;
	else
		return false;
}
int main()
{
	cin>>n>>k;
	for(int i=1;i<=k;i++)	cin>>w[i];
	sort(w+1,w+k+1);
	int l=0,r=n;
	while(l<r)
	{
		int mid=l+r>>1;
		if(check(mid))
			r=mid;
		else
			l=mid+1;
	}
	cout<<(l-1)*2<<endl;
	return 0;
}

AcWing 790. 数的三次方根(浮点二分法)

AcWing 790. 数的三次方根



#include <iostream>
#include <cstring>
#include <algorithm>
#include<bits/stdc++.h>
using namespace std;

double n;

int main()
{
    cin>>n;
    double l=-10000,r=10000;//答案ans范围的初始化
    while(r-l>=1e-8)
    {
        double mid = (l+r)/2;//中间值
        
        if(mid*mid*mid<n)
            l=mid;
        else
            r=mid;
    }//二分过程参考
    printf("%.6lf",l);//记得保留小数
    return 0;
}

AcWing 680. 剪绳子(浮点二分法)

AcWing 680. 剪绳子

视频讲解

思路分析+代码注释详解

高精度

前缀

一维前缀和

AcWing 795. 前缀和

前缀和算法
预处理时间复杂度:O(n)
求前缀和时间复杂度:O(1)
用空间来换取时间
前缀和算法(前缀和的下标都从1开始,避免出现越界)
算法理论
前缀和其实就是 动态规划 的思想

acwing 795. 一维前缀和

/*
一维前缀和 ( O(1) 的时间求出一段区间的和 )
b[i] = a[1] + a[2] + … a[i]=b[i-1]+a[i] 
a[l] + … + a[r] = b[r] - b[l - 1]
*/
#include<bits/stdc++.h>
using namespace std;

const int N=1e5+10;

int a[N];//输入矩阵 
int b[N];//前缀和矩阵 
int n,m;
int l,r;

int main()
{
	cin>>n>>m;
	for(int i=1;i<=n;i++)
	{
		cin>>a[i];
		b[i]=b[i-1]+a[i];//前缀和的初始化 
	 } 
	 while(m--)
	 {
	 	cin>>l>>r;
	 	cout<<b[r]-b[l-1]<<endl;//输出区间和 
	 }
	return 0;
 } 

AcWing 3956. 截断数组

AcWing 3956. 截断数组

#include<bits/stdc++.h>
using namespace std;
const int N=1e5+10;
typedef long long LL;
LL ans;
LL cnt;
int n;
int a[N];
int main()
{
	cin>>n;
	for(int i=1;i<=n;i++)
	{
		int x;
		cin>>x;
		a[i]=a[i-1]+x;//前缀和数组 
	}
	
	for(int i=1;i<=n-2;i++)//枚举的第一个位置 
	{
		if(a[i]==a[n]/3)//第一个刀位置
			cnt++;
		if(a[n]-a[i+1]==a[n]/3)//第二个位置 ---通过枚举第一个位置间接枚举第二个位置
			ans+=cnt;
	}
	if(a[n]%3!=0 || n<3)
		cout<<0<<endl;
	else
		cout<<ans<<endl;
	return 0;
}

AcWing 1230. K倍区间(一维前缀和+同余定理)

AcWing 1230. K倍区间

解析

直接一维前缀和+数学推理
求区间[l,r]的和是k的倍数的个数。
求区间和,我们可以通过前缀和来求出。我们规定sum[i]表示第1个元素到第i个元素的和。

那么sum[r] - sum[l-1]就是区间[l,r]的和。
区间[l,r]的和是k的倍数即(sum[r] - sum[l-1])%k = = 0  即sum[r]%k = = sum[l-1]%k

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;//数据开大点 
const int N = 1e5+10;
int n,k;
LL a[N],b[N];

LL cnt[N];//哈希存储 模相同的区间个数   cnt[x]:余数为x的个数
int main()
{
    cin>>n>>k;
    cnt[0]=1;
    //就是之前找到的是两个 区间的 mod之后相同的值
	//但是对于 mod == 0区间 我可以不用找另外区间,我本身就是符合 mod k == 0 这个条件
	//循环里统计的时候没有考虑这个情况,所以之后要再加上
	for (int i=1;i<=n;i++)
    {
    	cin>>a[i];
        b[i]=b[i-1]+a[i]; //求前缀和
		cnt[b[i]%k]++;// 前缀和取余k的余数的数量  
    }
    
    LL ans=0;//ll数据太大 
    for (int i=0; i<k; i++)//余数必在 0~k-1之间 
    {
       ans+=cnt[i]*(cnt[i]-1)/2; //数学C n 取 2 的公式 
    }
    cout<<ans<<endl;
    return 0;
}

202012-2csp-期末预测之最佳阈值(排序+一维前缀和)

AcWing 3298. 期末预测之最佳阈值

这道题我主要用到前缀和的算法,首先对数据从小到大进行排序,然后利用前缀和计算出每个数左边有多少个0,进而计算出该数右边有多少个1(包括该数在内),然后让二者相加,即为最后的正确的个数,至于如果有相同的数的话,就只计算出第一个数即可,因为其他的都小于等于第一个数。

从规则来看,我们可以利用前i-1位同学的情况来推断第i位同学的情况。从样例1中可以得到启示
因此考虑使用前缀和算法来进行优化。
注意 : 遇到相同数时,应只算第一个!!!
为什么呢?
因为这个分界线只能在不同的y之间画,在相同的y之间画的话,i和i−1意义相同,在套公式的时候实际破坏了前缀和,没有发挥前缀和的作用
比如,s[0][i]表示前i个数中0的个数,那如果连续两个y相等,那么计算这个的时候,会多算。本来想算的是第i-1个数前,但是第i-1个数和第i个数是相等的,从而多算了。
在计算s[1][m]−s[1][i−1]也是同理。
#include<bits/stdc++.h>
using namespace std;
const int N=1e5+10;
typedef pair<int,int> PII;
#define x first
#define y second
PII a[N];//用pair型数组a记录每个输入的y和result 
int m;
int s[2][N];//s[0][i]---表示前i个数中0的个数               s[1][i]---表示前i个数中1的个数

int main()
{
    cin>>m;
    for(int i=1;i<=m;i++) cin>>a[i].x>>a[i].y;
    sort(a+1,a+m+1);//按第一个键值排序(从小到大) 对数组a按y(安全指数)进行升序排序




    for(int i=0;i<2;i++)
        for(int j=1;j<=m;j++)
            s[i][j]=s[i][j-1]+(i==a[j].y);//两个  一维前缀和 




    int cnt=-1;//匹配项的最大值 
	int res=0;//保存取得最大值时的安全指数
    for(int i=1;i<=m;i++)
    {
        int t=s[0][i-1]+(s[1][m]-s[1][i-1]);//表示每个数作为阈值时预测正确的数量----前i中预测为0 + 前i中预测为1 
        if(t>=cnt) 
        {
            cnt=t;
            res=a[i].x; 
        }
        while(i+1<=m&&a[i].x==a[i+1].x) i++;//遇到相同数时,应只算第一个!!! ----经典的错误-标准的零分 
    }
    cout<<res<<endl;
    return 0;
}

二维前缀和

AcWing 796. 子矩阵的和

acwing 796. 子矩阵的和

/*
二维前缀和 
b[i, j] = 第i行j列格子左上部分所有元素的和
b[i,j] = b[i-1,j] + b[i,j-1] - b[i-1,j-1] + a[i,j];
以(x1,y1)为左上角,(x2,y2)为右下角的子矩阵的和为:
b[x2,y2] - b[x1-1, y2] - b[x2,y1-1] + b[x1-1, y1-1]
*/

#include<bits/stdc++.h>
using namespace std;
const int N=1e3+10;

int a[N][N];//输入矩阵 
int b[N][N]; //前缀和矩阵


int main()
{
    int n,m,q;
	cin>>n>>m>>q;
	for(int i=1;i<=n;i++)
	{
		for(int j=1;j<=m;j++)
		{
			cin>>a[i][j];
			b[i][j]=b[i-1][j]+b[i][j-1]-b[i-1][j-1]+a[i][j];//前缀和数组的初始化 
		}
	}
	while(q--)
	{
	    int x1,x2,y1,y2;
		cin>>x1>>y1>>x2>>y2;
		cout<<b[x2][y2]-b[x1-1][y2]-b[x2][y1-1]+b[x1-1][y1-1]<<endl;//输出子矩阵的和
	}
	return 0;
 } 

AcWing 126. 最大的和

AcWing 126. 最大的和
暴力

#include<iostream>
using namespace std;
const int N=110;
int g[N][N],s[N][N];
int n;
int main(){
    cin>>n;
    for(int i=1;i<=n;i++)
        for(int j=1;j<=n;j++){
            cin>>g[i][j];
            s[i][j]=s[i-1][j]+s[i][j-1]-s[i-1][j-1]+g[i][j];
        }
    int res=-1e9;
    //枚举一下上下左右四个点---二维前缀和
    for(int x1=1;x1<n;x1++)
        for(int y1=1;y1<n;y1++)
            for(int x2=x1;x2<=n;x2++)
                for(int y2=y1;y2<=n;y2++)
                    res=max(res,s[x2][y2]-s[x1-1][y2]-s[x2][y1-1]+s[x1-1][y1-1]);
    cout<<res<<endl;
    return 0;
}

AcWing 99. 激光炸弹(二维前缀和+边界处理)

AcWing 99. 激光炸弹
一个数组就行了,在自己身上求前缀和。
xy坐标是从0开始的。


#include<bits/stdc++.h>
using namespace std;

typedef long long LL;
const int N = 5010;
int n;
int R;
int g[N][N];//原矩阵与前缀和矩阵共用

int main()
{
    cin>>n>>R;
    int x,y,w;
    int l=R,r=R;//因为没有定义地图的长和宽,自己来更新
    for (int i = 0; i < n; i ++ )
    {
        cin>>x>>y>>w;
        x++,y++;//这里是因为习惯上将二维前缀和起始点坐标从(1,1)开始
         l=max(l,x),r=max(r,y);
		 g[x][y]+=w;
    }
    
    for (int i = 1; i <= l; i ++ )
        for (int j = 1; j <= r; j ++ )
                g[i][j]+=g[i-1][j]+g[i][j-1]-g[i-1][j-1];
                
    int res=0;
    for (int i = R; i <= l; i ++ )//至少从左上角的第一个R*R的正方形开始
        for (int j = R; j <= r; j ++ )
            res=max(res,(g[i][j]-g[i-R][j]-g[i][j-R]+g[i-R][j-R]));
    
    cout<<res<<endl;
    return 0;
}

202104-2-csp-邻域均值(二维前缀和+边界处理)

AcWing 3412. 邻域均值

根据题意,需要求一个点附近r范围内矩阵的和,即范围为(2r+1)2大小的矩阵,考虑使用二维前缀和算法

由于矩阵有边界,因此需要对边界范围进行处理:
当要下溢出时,将边界设置为1;当要上溢出时,将边界设置为n

矩阵的大小为(x2-x1+1)*(y2-y1+1)

小细节:如果直接算均值,均值可能为小数,此处可以改写一些判断条件
737
#include<bits/stdc++.h>
using namespace std;

const int N=1e3+10;

int n;
int r;
int t,L;
int matrix[N][N];//输入矩阵
int b[N][N]; //前缀和矩阵 
int main()
{
	cin>>n>>L>>r>>t;
	for(int i=1;i<=n;i++)
		for(int j=1;j<=n;j++)
		{
			int x;
			cin>>x;
			b[i][j]=b[i-1][j]+b[i][j-1]-b[i-1][j-1]+x;	//初始化二维前缀和数组 
		}

	int ans=0;
	for(int i=1;i<=n;i++)
		for(int j=1;j<=n;j++)
		{
			//判断矩阵的边界
			int x1=max(i-r,1);
			int y1=max(j-r,1);
			int x2=min(i+r,n);
			int y2=min(j+r,n);
			
			int num=(x2-x1+1)*(y2-y1+1);//统计矩阵中元素的个数
			int sum=b[x2][y2]-b[x1-1][y2]-b[x2][y1-1]+b[x1-1][y1-1];
			if(sum<=num*t)//利用前缀和矩阵    求     矩阵的和
				ans++; 
		}
	cout<<ans<<endl;
	
	return 0;
 } 

差分

深入剖析差分的本质探究差分解法的由来

一维差分

同时改变一个区间中数的大小

AcWing 797. 差分

AcWing 797. 差分
问题:acwing 2041. 干草堆


/*
给区间[l, r]中的每个数加上c:B[l] + c, B[r + 1] - c
差分可以看成前缀和的逆运算
*/

/*
首先给定一个原数组a:a[1], a[2], a[3],,,,,, a[n];
然后我们构造一个数组b : b[1] ,b[2] , b[3],,,,,, b[i];
使得 a[i] = b[1] + b[2 ]+ b[3] +,,,,,, + b[i]

也就是说,a数组是b数组的前缀和数组,反过来我们把b数组叫做a数组的差分数组。
换句话说,每一个a[i]都是b数组中从头开始的一段区间和。
*/
#include<iostream>
using namespace std;
const int N=100010;

int n,m;
int a[N],b[N];//a数组是b数组的前缀和数组 

int main()
{
	cin>>n>>m;
	for(int i=1;i<=n;i++)
	{
		cin>>a[i];
		b[i]=a[i]-a[i-1];//构建差分数组 
	}
	while(m--)
	{
		int l,r,c;
		cin>>l>>r>>c;
		b[l]=b[l]+c,b[r+1]=b[r+1]-c; //将序列中[l, r]之间的每个数都加上c
	}	 
	for(int i=1;i<=n;i++)
		a[i]=a[i-1]+b[i];//前缀和运算
	for(int i=1;i<=n;i++)
		cout<<a[i]<<" ";//输出最终序列 
	return 0;
 } 

AcWing 3729. 改变数组元素

AcWing 3729. 改变数组元素

#include<bits/stdc++.h>
using namespace std;
const int N=2e5+10;

int b[N];//差分数组

int main()
{
	int T;
	cin>>T;
	while(T--)
	{
		int n;
		cin>>n;
		for(int i=0;i<=n;i++) b[i]=0;//因为差分数组要使用多次,每次使用前初始化
		
		for(int i=1;i<=n;i++)
		{
			int x,l,r;
			cin>>x;
			l=max(1,i-x+1);//差分数组的左端 
			r=i;//差分数组的右端 
			b[l]+=1;
			b[r+1]-=1;
		 }
		 for(int i=1;i<=n;i++)
		 {
		 	b[i]=b[i]+b[i-1];
		 	cout<<!!b[i]<<" ";
		 }
		cout<<endl;
	 } 
	return 0;
 } 

202203-2csp-出行计划

AcWing 4455. 出行计划

请添加图片描述

请添加图片描述
请添加图片描述符合题意的做核酸区间+1----->差分来做

满分写法

#include<bits/stdc++.h>
using namespace std;
const int N=2e5+10;

int n;
int m;
int k;
int res[N]={0};//差分数组 
int main()
{
	cin>>n>>m>>k;
	for(int i=0;i<n;i++) //判断出进入该场所需要的最早时间核酸报告(left)和最晚时间核酸报告(right) 
	{
		int t;//t时刻进入场所
		int c;//场所维持时间
		cin>>t>>c;
		int l,r;
		l=max(t-k-c+1,0);
		r=max(t-k,0);
		res[l]+=1, res[r+1]-=1;//表示处于这个时间段内,场所是可以访问的 			
	}
	for(int i=1;i<=N;i++)//一维前缀和 
	{
		res[i]=res[i-1]+res[i];
	}
	for(int i=0;i<m;i++)//输出查询 
	{
		int q;
		cin>>q;
		cout<<res[q]<<endl;	
	}	
	
	return 0;
}

AcWing 100. 增减序列

AcWing 100. 增减序列
AcWing 100. 疑难点详解,看完不会你来打我

//要使最后的数都一样,那么b数组中的b2=>bn  一定全 0

//贪心的思想,来使得b中所有数变成零
//我们知道我们在做b[L]++,b[R+1]--;操作的时候,要找两个数配对,那么 负数++,正数--,是不是就最快了。
// 但是最终结果可能依然不是全 0 的,因为 abs(sum(正数))可能!=abs(sum(负数))
//所以,我们可以 让最后不等于0 的数全和 b1||bn+1来换。


/*
那么题目就变成了对一个数组可进行三种操作
1对两个元素一个加一一个减一
2对一个元素加一
3对一个元素减一
*/        //后面两种操作实质是对一个元素和队首或队尾进行操作


/*最终的序列有 abs(pos-neg)+1种
因为还剩下的abs(pos-neg) 种操作是对队尾或队首的操作
队尾或队首就会影响这个数列的值,所以最多加那么多次,最少一种
*/
#include<bits/stdc++.h>
using namespace std;
const int N=1e5+10;
typedef long long LL;
int n;
int a[N];
int b[N];
LL pos,neg;
int main()
{
	cin>>n;
	for(int i=1;i<=n;i++)
	{
		cin>>a[i];
		b[i]=a[i]-a[i-1];//差分数组的初始化 
	}
	for(int i=2;i<=n;i++)
	{
		if(b[i]>0)
			pos+=b[i];//差分数组第二个元素起所有正数和 
		else
			neg-=b[i];//差分数组第二个元素起所有负数和的绝对值 
	}
	cout<<min(pos,neg)+abs(pos-neg)<<endl;
	cout<<abs(pos-neg)+1<<endl; 
	
	
	return 0;
 } 

AcWing 101. 最高的牛(差分+区间处理)

AcWing 101. 最高的牛

题目中说对于两头牛它们可以互相看见,说明两牛之间的牛的身高都比这两只低
因此根据最优的原则,我们可知中间的牛可以都比这两只小1即可 。

现在我们考虑关系会不会有交叉的情况。
假设i<j<k<l;存在关系ik和jl
因为存在关系ik,因此k的身高大于j,又因为存在jl,所以j的身高大于k
前后互相矛盾,因此不存在关系存在交叉的情况。

所以对于该问题,我们可以假设全部都是最高身高
然后每出现一对关系,就将他们之间的牛的身高全减1
因为涉及区间加减1,我们可以采用差分和前缀和的关系来解决该问题
具体实现看代码,注意关系判重。
#include<bits/stdc++.h>
using namespace std;
const int N=5e3+10;

int height[N];//差分数组 
set<pair<int,int>> S; 
int main()
{
	int N,P,H,M;
	cin>>N>>P>>H>>M;
	height[1]=H;//直接初始化差分数组
	for(int i=0;i<M;i++)
	{
		int A,B;
		cin>>A>>B;
		if(A>B) swap(A,B);//调整顺序 
		if(!S.count({A,B}))//去重 
		{
			S.insert({A,B});
			height[A+1]--;//A-----B之间的所有牛身高-- 
			height[B]++;
		} 
	
	}
		for(int i=1;i<=N;i++)//求差分数组的一维前缀和 
		{
			height[i]+=height[i-1];
			cout<<height[i]<<endl;
		}
	
	return 0;
}
水位每上涨一个高度差后对数组中数的关系有怎样的影响

202109-2-csp-非零段划分

AcWing 4007. 非零段划分

区间原地划分时可以观察相邻元素之间的大小关系是否与划分有关。
前缀和与差分实现单位时间内区间数值整体加1。
 当a[i]>a[i-1]时,只要p取到区间a[i-1]到a[i]-1中的值,都能构成一个新的非零段。这就是p与数组的关系
 根据这个关系利用前缀和与差分实现单位时间内区间数值整体加1,将双重循环改进至单层,降低计算时间。
再回看这里,这里干了个什么事呢?
就是 如果a[i]>a[i-1],也就是后一个数比前一个大
那么,当p取到它们中间的值时,就会出现一个非零段;
而当p比 a[i]大的时候,就会都变成0,没有非零段;
例如(a[1]=3) >(a[0]=0) 
当p取0、1、2时都会出现一个非零段,但当p取3时,就都变成0,没有非零段
b[0] = 1 b[3] = -1
b[0]从0变为1,为什么?
p=0的时候,有1个非零段,同理p=1、2 、3也只有一个非零段,只用对b[0]加1就行了
为什么只加一个1呢?
因为差分数组只需要改变一个值,就可以影响一个区间的值
为什么又要把b[3]从0变为-1呢?
因为p>3的时候,3也变成0了,没有非零段了
实际上就是从1到n持续循环这个过程,按数组a的元素的值,为差分数组b的下标
最后再对差分的原数组(进行前缀和操作)进行最大值求和;
#include<bits/stdc++.h>
using namespace std;
const int N=5e5+10;
const int M=1e4+10;

int n;
int a[N];
int b[M];//差分数组 
int main()
{
	cin>>n;
	
	for(int i=1;i<=n;i++)
	{
		cin>>a[i];
		if(a[i-1]<a[i])//a[i-1]到a[i]-1段的p都能构成新的非零段
		{
			b[a[i-1]]++;//处理差分数组以实现区间整体+1
			b[a[i]]--; 
		}		
	}
	int ans=0;//返回取值最大的p 

	for(int i=1;i<=M;i++)//差分数组  求  一维前缀和
	{
		b[i]+=b[i-1];
		if(b[i]>ans) ans=b[i];
	 } 
	cout<<ans<<endl;
	
	return 0;
}
利用---海平面下降--海岛个数的思想去理解这个题目
AcWing 2014. 岛问题的变形
AcWing 2014. 岛---海平面一直上升
AcWing 4007. 非零段划分---海平面一直下降
#include<bits/stdc++.h> 
using namespace std;
const int N = 500010, M = 10010;

int n;
int a[N];//山峰的高度 

//海水退却到i时,会比高度为i+1时,多出来的山峰个数 
int cnt[M];//所有高度为i的山峰的贡献
int main()
{
    cin>>n;
    a[0]=a[n+1]=0;//为了左右两边判断方便 
    for(int i=1;i<=n;i++) cin>>a[i];
    n=unique(a+1,a+n+1)-a-1;//判重---所有连续高度相同的山峰可以看做为一个 
 

    for(int i=1;i<=n;i++)
    {
        int x=a[i-1];
		int y=a[i];//处理y高度山峰的贡献 
		int z=a[i+1];
        if (x<y && z<y) cnt[y]++;//---海平面下降---山峰个数新增加一个 
        else if(x>y && z>y) cnt[y]--;//---海平面下降---山峰会被减少一个
		 
        //这里少了两种情况---x<y<z//x>y>z---这两种---在海平面下降的情况下山峰的个数不会发生变化 
    }

    int res=0;	//定义答案 
	int sum=0;	//定义当前山峰漏出来的段数 
    for (int i=M-1;i;i--)//随着海平面的下降---山峰的增加个数 ----起初没有山峰(全在海平面以下) 
    {
        sum+=cnt[i];
        res=max(res,sum);
    }

    cout<<res<<endl;
    return 0;
}

AcWing 2014. 岛(贪心+模拟)

AcWing 2014. 岛(离散化+差分)c++最短代码,同非零段划分(含对差分算法的深入探究)
贪心+模拟

#include<bits/stdc++.h>
#include <iostream>
#include <cstring>
#include <algorithm>

#define x first
#define y second 
using namespace std;
typedef pair<int, int> PII;
const int N = 100010;

int n;
int h[N];
PII q[N];

int main()
{
    cin>>n; 
    for (int i=1; i<=n;i++)	cin>>h[i];

    n=unique(h+1,h+n+1)-h-1;  // 判重----删掉相邻的重复元素 
    h[n+1]=0;  // 后续代码可能会用到第n + 1个位置,需要把第n + 1个位置清空

    for (int i=1; i<=n;i++) q[i] = {h[i],i};
    sort(q+1,q+n+1);//按高度排序 

    int ans = 0; 
	int cnt = 1;//初始时岛屿数量----没有水的时候 
    for (int i = 1; i <= n; i ++ )
    {
        int k=q[i].y;//当前海平面长到k高度 
        if (h[k-1]<h[k] && h[k+1]<h[k]) cnt -- ;//比两边都大 
        else if (h[k-1]>h[k] && h[k+1]>h[k]) cnt ++ ;//比两边都矮时 

        if (q[i].x != q[i + 1].x)//按高度排序(失散多年的兄弟),当高度不一样时跟新ans 
            ans = max(ans, cnt);
    }
    cout<<ans<<endl;

    return 0;
}

差分写法

#include <iostream>
#include <cstring>
#include <algorithm>
#include <map>


typedef long long LL;

using namespace std;
const int N = 100005,M = 1e9+1;
int a[N];
map<int ,int >b;
int n;
int main()
{
    cin >> n;

    for (int i = 1; i <= n; i ++ ){
        cin >> a[i];
        if(a[i]>a[i-1]){
            //数的大小在[a[i-1],a[i]-1]之间的所有数大小都+1
            b[a[i-1]]++,b[a[i]]--;
        }
    }
    LL sum = 0 ,res = 0;
    for (auto i:b ){
        //求前缀和
        sum+=i.second;
        res = max(res,sum);
    }
    cout << res;
}

二维差分

AcWing 798. 差分矩阵(二维差分)

798. 差分矩阵
【c++详细题解】

/*
给以(x1,y1)为左上角,(x2,y2)为右下角的子矩阵中的所有元素加上c:
S[x1,y1]+=c, S[x2 +1,y1]-=c, S[x1,y2+1]-=c, S[x2+1,y2+1]+=c
*/

#include <iostream>
using namespace std;
const int N = 1e3 + 10;
int a[N][N];//输入矩阵 
int b[N][N];//b是差分矩阵 
int n, m, q;

void insert(int x1, int y1, int x2, int y2, int c)
{
    b[x1][y1] += c;
    b[x1][y2 + 1] -= c;
    b[x2 + 1][y1] -= c;
    b[x2 + 1][y2 + 1] += c;
}

int main()
{
    cin>>n>>m>>q;
    for (int i = 1; i <= n; i++)
        for (int j = 1; j <= m; j++)
            cin >> a[i][j];
    for (int i = 1; i <= n; i++)
        for (int j = 1; j <= m; j++)
            insert(i, j, i, j, a[i][j]);      //始化构造差分数组
  
    // 多次进行矩阵区域插入
    while (q--)
    {
        int x1, y1, x2, y2, c;
        cin>>x1>>y1>>x2>>y2>>c;
        insert(x1, y1, x2, y2, c);
    }   
    // 对差分矩阵求二维前缀和
    for (int i = 1; i <= n; i++)
    {
        for (int j = 1; j <= m; j++)
        {
            b[i][j] += b[i - 1][j] + b[i][j - 1] - b[i - 1][j - 1];
            cout<<b[i][j]<<" ";
        }
    	cout<<endl; 
    }
    
    return 0;
}

201409-2-csp-画图—二维差分

AcWing 3203. 画图
在这里插入图片描述

每个格子的编号---是它的左下角坐标---从[i][j]开始

暴力做法

#include<bits/stdc++.h>
using namespace std;
const int N=110;

bool st[N][N];//标记[i][j]格子是否被涂色---每个格子的坐标---是它左下角的坐标 

int main()
{
	int n;
	cin>>n;
	while(n--)
	{
		int x1,x2,y1,y2;
		cin>>x1>>y1>>x2>>y2;
		for(int i=x1;i<x2;i++)//每个格子的坐标---是它左下角的坐标 
		{
			for(int j=y1;j<y2;j++)
			{
				st[i][j]=true;
			}
		}
	}
	
	int ans=0;
	for(int i=0;i<N;i++)
	{
		for(int j=0;j<N;j++)
		{
			if(st[i][j])//将涂色的格子统计个数 
				ans++;
		}
	}
	cout<<ans<<endl; 
	return 0;
}
每个格子的坐标---是它右上角坐标---二维差分方便---从[1][1]开始
#include<bits/stdc++.h>
using namespace std;
const int N=110;
int b[N][N];

int ans;
void insert(int x1, int y1, int x2, int y2, int c)//二维差分模板 
{
    b[x1][y1] += c;
    b[x1][y2 + 1] -= c;
    b[x2 + 1][y1] -= c;
    b[x2 + 1][y2 + 1] += c;
}
int main()
{
	int n;
	cin>>n;
	
	while(n--)
	{
		int x1,x2,y1,y2;
		cin>>x1>>y1>>x2>>y2;
		insert(x1+1,y1+1,x2,y2,1); //坐标从 [i][j]开始---方便---二维差分 
		
	}
	for(int i=1;i<N;i++)
	{
		for(int j=1;j<N;j++)
		{
			b[i][j]+=b[i-1][j]+b[i][j-1]-b[i-1][j-1]; //二维前缀和 
			if(b[i][j])	ans++;
		}
	}
	cout<<ans<<endl;
	return 0;
}

快速排序

双指针算法

在这里插入图片描述

算法模型---思路:先找出暴力解法,根据题目性质,优化到双指针

1.对撞指针------左右两个指针,向中间靠拢。

2.快慢指针------左右两个指针,一快一慢

3.滑动窗口

4.归并排序---操作两个数组

AcWing 799. 最长连续不重复子序列—滑动窗口

AcWing 799. 最长连续不重复子序列

#include<bits/stdc++.h>
using namespace std;
const int N=1e5+10;
int a[N];
int s[N];
int main()
{
	int n;
	cin>>n;
	int ans=0;//答案 
	for(int i=0;i<n;i++) cin>>a[i];
	//[l,r]区间端点 ----双指针维护的是以a[r]为结尾的最长不重复子序列,长度为r-l+1
	for(int r=0,l=0;r<n;r++) 
	{
		s[a[r]]++;//存储子序列中a[r]出现的次数 
		while(l<r && s[a[r]]>1)   //当a[r]重复时,先把a[l]次数减1,再右移l
		{
			s[a[l]]--;
			l++;//区间左端点----向右移动 
		}
		ans=max(ans,r-l+1); //更新最大长度 
	}
	cout<<ans<<endl;
	return 0;
 } 

AcWing 3768. 字符串删减—滑动窗口

AcWing 3768. 字符串删减

#include<bits/stdc++.h>
using namespace std;
const int N=1e2+10;
string s;
int ans;
int main()
{
	int n;
	cin>>n;
	cin>>s;
	for(int l=0;l<n;l++)//左指针右移----[l,r] 
	{
		if(s[l]=='x')//如果s[l]不是'x'时,继续往右移动----直到枚举到第一个'x' 
		{
			int r=l+1;//右指针设置 
			while(r<n && s[r]=='x')	r++;//右指针右移 
			ans+=max(0,r-l-2);//答案+这个区间需要删除的'x'的数量 
			l=r-1;//重新设置左指针 
		}
	}
	cout<<ans<<endl;
	return 0;
 } 

AcWing 1238. 日志统计—滑动窗口

蓝桥杯2018年第九届真题-日志统计

排序+双指针 
对所有的赞按照时间从小到大排序
通过双指针i,j维护长度不大于d的区间,并记录该区间的中所有帖子获得的赞数
#include<bits/stdc++.h>
using namespace std;
const int N = 1e5+10;
typedef pair<int, int> PII;
PII q[N];//用于存储日志ts, id
#define ts first
#define id second
int n,d,k;//n行,d时间间隔,k个赞
int ts,id;
int cnt[N];//用来存储每个id当前获得的点赞数
bool st[N];用于存储每个帖子是否是热帖

int main()
{
    cin>>n>>d>>k;
    for (int i=0;i<n;i++) cin>>q[i].ts>>q[i].id;
    sort(q,q+n);//按时间排序 

    for (int r=0,l=0;r<n;r++)//双指针算法,[l,r]
    {
        cnt[q[r].id]++;//当前第r个记录顺序的id的点赞数++;
        
        
        while(q[r].ts-q[l].ts>=d)//如果俩个帖子时间相差超过d------说明该赞无效
        {
            cnt[q[l].id]--;//获赞的时间太久远了,赞作废
            l++;//要把指针l右移
        }
        if(cnt[q[r].id]>=k)//如果该id贴赞超过k,说明是热帖
            st[q[r].id]=true;
            
    }
    
    for(int i=0;i<N;i++)//最多有1e5个id
        if(st[i])
            cout<<i<<endl;
    return 0;
    
}

AcWing 800. 数组元素的目标和—对撞指针—操作两个数组

AcWing 800. 数组元素的目标和

#include<bits/stdc++.h>
using namespace std;
const int N=1e5+10;
int a[N];
int b[N];
int main()
{
	int n,m,x;
	cin>>n>>m>>x;
	for(int i=0;i<n;i++) cin>>a[i];
	for(int i=0;i<m;i++) cin>>b[i];
	
	
	for(int l=0,r=m-1;l<n;l++)//对撞指针---l左指针  r右指针 
	{
		while(r>=0 && a[l]+b[r]>x) r--;//右指针左移 
		
		if(r>=0 && a[l]+b[r]==x)//符合答案,输出 
			{
				cout<<l<<" "<<r<<endl;
				break;
			}
	}
	return 0;
}

AcWing 2816. 判断子序列—操作两个数组

AcWing 2816. 判断子序列
题解

#include<bits/stdc++.h>
using namespace std;
const int N=1e5+10;

int a[N];
int b[N];
int main()
{
	int n,m;
	cin>>n>>m;
	for(int i=0;i<n;i++) cin>>a[i];
	for(int i=0;i<m;i++) cin>>b[i];
	
	int l=0;//第一个数组的指针 
	for(int r=0;r<m;r++)//第二个数组的指针移动 
	{
		if(l<n && a[l]==b[r])	l++;//第一个数组的指针移动 
	}
	if(l==n)
		cout<<"Yes"<<endl;
	else
		cout<<"No"<<endl;
	return 0;
}

AcWing1532. 找硬币—对撞指针

acwing1532. 找硬币

双指针要求具有单调性

#include<bits/stdc++.h>
using namespace std;
const int N=1e5+10;
int a[N];
int main()
{
	int n,m;
	cin>>n>>m;
	for(int i=1;i<=n;i++)
		cin>>a[i];
	sort(a+1,a+n+1);
	int l=1,r=n;
	int suit_l=0,suit_r=0;
	while(l<r)//指针未相遇就一直执行
	{
		//下面的if 不能调动顺序-------经典错误,标准零分
		//如果调动顺序-----2,3,1时,假如2/3不满足,会修改l/r的值,使得到达1的时候,l/r变动 
		if(a[l]+a[r]==m)//-----1 
			{
				suit_l=a[l];
				suit_r=a[r];
				break;
			}
		if(a[l]+a[r]>m)//------2
			r--;
		if(a[l]+a[r]<m)//------3
			l++;
		
	}
	if(suit_l+suit_r==m)
		cout<<suit_l<<" "<<suit_r<<endl;
	else
		cout<<"No Solution"<<endl;
	return 0;
}

AcWing1574.接雨水—对撞指针

acwing1574. 接雨水

双指针,从左和右出发两个指针,每次记录左右两侧最大值
l_max为[1,l]最大值,r_max为[r,n]最大值
每次从最大值小的那一方开始收水
比如l_max < r_max,可以收左边的水,因为l左侧[1,l]最大值确定,右侧[l,n]有值大于左侧最大值,反之亦然。
一列一列收水,一次收完一列水
#include<bits/stdc++.h>
using namespace std;
const int N=1e5+10;
int a[N];

int main()
{
	int n;
	cin>>n;
	for(int i=1;i<=n;i++)	
		cin>>a[i];
	int l=1,r=n;
	int maxl=0,maxr=0;
	int ans=0;//保存的返回值--接到的雨水数量 
	while(l<r)
	{
		maxl=max(maxl,a[l]);//更新左边最大值-----当前l的左边最高值 
		
		maxr=max(maxr,a[r]);//更新右边最大值-----当前r的右边最高值 
		
		if(maxl<maxr)
		{
			ans+=maxl-a[l];
			l++;//左指针---右移 
		 }
		else
		{
			ans+=maxr-a[r];
			r--;//右指针---左移 
		 } 
		
	}
	cout<<ans<<endl; 
	return 0;
 } 

位运算

离散化

第二讲 数据结构

集合

AcWing 3542. 查找

AcWing 3542. 查找

#include<bits/stdc++.h>
using namespace std;

int main()
{
	int n,m;
	unordered_set<int> S;//定义一个集合存储a数组中的元素 
	cin>>n;
	while(n--)
	{
		int x;
		cin>>x;
		S.insert(x);
	 } 
	 
	 
	cin>>m;
	while(m--)
	{
		int x;
		cin>>x;
		if(S.count(x))
			cout<<"YES"<<endl;
		else
			cout<<"NO"<<endl;
	}
	return 0; 
 } 

单链表

双链表

栈(stack)

stack是一种先进后出(First In Last Out,FILO)的数据结构
只有stack顶端的元素,才有机会被外界取用。Stack不提供遍历功能





数据存取操作
push(elem);//向栈顶添加元素
pop();//从栈顶移除第一个元素
top();//返回栈顶元素




大小操作
empty();//判断堆栈是否为空
size();//返回堆栈的大小


AcWing 3302. 中缀表达式求值—向零取整

#include<bits/stdc++.h>
using namespace std;
unordered_map<char,int> pr;//保存运算符优先级
stack<int> num;//运算数栈 
stack<char> op;//运算符栈 

void eval()//执行某种运算 
{
	int x;
	
	auto b=num.top();num.pop();
	auto a=num.top();num.pop();
	auto c=op.top();op.pop();
	
	if(c=='+') x=a+b;
	else if(c=='-') x=a-b;
	else if(c=='*') x=a*b;
	else x=a/b;
	num.push(x);
}

int main() 
{
	pr['-']=1;//初始化运算符的优先级 
	pr['+']=1;
	pr['*']=2;
	pr['/']=2;
	string s;
	cin>>s;
	for(int i=0;i<s.size();i++)
	{
		if (isdigit(s[i]))//如果是运算数时 
        {
            int j = i, x = 0;
            while (j < s.size() && isdigit(s[j]))
                x = x * 10 + s[j ++ ] - '0';
            num.push(x);
            i = j - 1;
        }
		else if(s[i]=='(')//左括号---直接入栈 
			op.push(s[i]); 
		else if(s[i]==')')//右括号---操作到左括号为止
		{
			while(op.top()!='(')
				eval();
			op.pop();//左括号出栈 
		 }
		else//当是运算符时 
		{
			while(op.size() && op.top()!='(' && pr[op.top()]>=pr[s[i]]) eval();
			op.push(s[i]);
		 } 
	}
	
	while(op.size())
		eval();
	cout<<num.top()<<endl;
	return 0;
}

201903-2-csp-二十四点—向下取整

AcWing 3273. 二十四点

#include<bits/stdc++.h>
using namespace std;
stack<int> num;
stack<char> op;
unordered_map<char,int> pr;
int n;

void eval()
{
	auto b=num.top();num.pop();
	auto a=num.top();num.pop();
	auto c=op.top();op.pop();
	int x;
	if(c=='+')
		x=a+b;
	else if(c=='-')
		x=a-b;
	else if(c=='x')
		x=a*b;
	else//向下取整---除法 
	{
		if(a*b>=0)//当ab同号时 
			x=a/b;
		else
		{
			if(a%b==0)
				x=a/b;
			else
				x=a/b-1;//c++默认除法---向零取整---题干要求---向下取整 
		} 
	}
	num.push(x); 
 } 
int main()
{
	pr['+']=pr['-']=1;
	pr['x']=pr['/']=2;//整除---向下取整---c++默认是向零取整 
	cin>>n;
	while(n--)
	{
		//清空栈的操作 
		num = stack<int>();
        op = stack<char>();
		string s;
		cin>>s;
		for(int i=0;i<s.size();i++)
		{
			if(s[i]>='0' && s[i]<='9')		//如果是运算数时 
				num.push(s[i]-'0');
			else 		//如果是运算符时 
			{
				while(op.size() && pr[op.top()]>=pr[s[i]])	eval();
				op.push(s[i]); 
			}
		 }
		while(op.size())	eval();
		if(num.top()==24)
			cout<<"Yes"<<endl;
		else
			cout<<"No"<<endl; 
		 
	}
	return 0;
}

中缀表达式转后缀表达式—AcWing 3302. 中缀表达式求值—改编题目

是由上面的中缀表达式求值---去掉运算数栈----遇到运算数直接输出
#include<bits/stdc++.h>
using namespace std;
unordered_map<char,int> pr;//保存运算符优先级

stack<char> op;//运算符栈 

void eval()//执行某种运算 
{
	int x;
	
	auto c=op.top();op.pop();
	cout<<c<<" ";
}

int main() 
{
	pr['-']=1;//初始化运算符的优先级 
	pr['+']=1;
	pr['*']=2;
	pr['/']=2;
	string s;
	cin>>s;
	for(int i=0;i<s.size();i++)
	{
		if (isdigit(s[i]))//如果是运算数时 
        {
            int j = i, x = 0;
            while (j < s.size() && isdigit(s[j]))
                x = x * 10 + s[j ++ ] - '0';
            cout<<x<<" "; 
            i = j - 1;
        }
		else if(s[i]=='(')//左括号---直接入栈 
			op.push(s[i]); 
		else if(s[i]==')')//右括号---操作到左括号为止
		{
			while(op.top()!='(')
				eval();
			op.pop();//左括号出栈 
		 }
		else//当是运算符时 
		{
			while(op.size() && op.top()!='(' && pr[op.top()]>=pr[s[i]]) eval();
			op.push(s[i]);
		 } 
	}
	
	while(op.size())
		eval();
	return 0;
}

队列(queue/deque/priority_queue)

Queue——单端队列

Queue是一种先进先出(First In First Out,FIFO)的数据结构-------常见队列
只有queue的顶端元素,才有机会被外界取用-------------Queue不提供遍历功能
 qeque<int> q;




queue存取、插入和删除操作
q.push(elem);//往队尾添加元素
auto t=q.pop();//从队头移除第一个元素
auto t=q.back();//返回最后一个元素
auto t=q.front();//返回第一个元素





queue大小操作
empty();//判断队列是否为空
size();//返回队列的大小





Deque——双端队列

dueque 可以在头尾两端分别做元素的插入和删除操作
deque<int> dq;



deque双端插入和删除操作
push_back(elem);//在容器尾部添加一个数据
push_front(elem);//在容器头部插入一个数据
pop_back();//删除容器最后一个数据
pop_front();//删除容器第一个数据

优先队列—堆

priority_queue//优先队列-----队列和排序的完美结合体
元素被赋予优先级,当访问元素时,具有最高级优先级的元素先被访问



默认操作
q.empty()          //如果队列为空,则返回true,否则返回false
q.size()              //返回队列中元素的个数
q.pop()              //删除队首元素,但不返回其值
q.top()               //返回具有最高优先级的元素值,但不删除该元素
q.push(item)      //在基于优先级的适当位置插入新元素

大顶堆构造一个空的优先队列(此优先队列默认为大顶堆)
priority_queue<int,vector<int>,less<int>> big_heap;

小根堆构造一个空的优先队列,此优先队列是一个小顶堆
priority_queue<int,vector<int>,greater<int>>small_heap;

201712-2csp游戏(queue)—约瑟夫问题

AcWing 3253. 游戏

#include<bits/stdc++.h>
using namespace std;
queue<int> q;

int n,k;
bool check(int x)
{
	if(x%k==0||x%10==k)
		return true;
	else
		return false;
}
int main()
{
	cin>>n>>k;
	for(int i=1;i<=n;i++)
		q.push(i);
	int count=1;// 从1开始报数
	while(q.size()>1)
	{
		int t= q.front();
		q.pop();
		if(!check(count))// 如果这个数不用出局,那就再加到队尾
			q.push(t);
		count++;
	}
	cout<<q.front()<<endl;
	return 0;
 } 

单调栈

单调队列

KMP

Trie

并查集

在这里插入图片描述

并查集思想(重点)
我们可以把每个连通分量看成一个集合,该集合包含了连通分量的所有点。而具体的连通方式无关紧要,好比集合中的元素没有先后顺序之分,只有“属于”与“不属于”的区别。图的所有连通分量可以用若干个不相交集合来表示。
分为三部分:
(1)初始化:使每个结点的初始根节点为自己,并且每个结点构成一颗树,树的深度是1;
(2)查找:使用递归来查找每个结点的父亲结点;
(3)合并:将不同父节点的结点合并;
注:这里的并查集是优化后的,即:进行了路径压缩。如果题目中无要求,可以只写简单的并查集算法

AcWing 837. 连通块中点的数量

AcWing 837. 连通块中点的数量

3

#include<bits/stdc++.h>
using namespace std;
const int N=1e5+10;

int n,m;
int p[N];//存储i的父亲结点的编号
int cnt[N];//记录---i所在树中结点的个数
 
int find(int x)//返回x的祖宗结点+路径压缩 
{
	if(x!=p[x])//x不是根节点时 
		p[x]=find(p[x]); //认祖为父 
	return p[x];//返回x的祖宗结点	
}
int main()
{
	cin>>n>>m;
	for(int i=1;i<=n;i++)
	{
		p[i]=i;
		cnt[i]=1; 
	}
		
	while(m--)
	{
		string op;
		int a,b;
		cin>>op>>a;
		if(op=="C")
		{
			cin>>b;
			int aa=find(a);
			int bb=find(b);
			if(aa!=bb)
			{
				p[aa]=bb;
				cnt[bb]+=cnt[aa];
			}			
		}
			
		else if(op=="Q1")
		{
			cin>>b;
			if(find(a)==find(b))
				cout<<"Yes"<<endl;
			else
				cout<<"No"<<endl;
		 }
		 else 
		 {
		 	cout<<cnt[find(a)]<<endl;
		  } 
	 } 
	return 0;
}

哈希表(map/unordered_map)

map<key,value> m--------Map所有的元素都是pair---------map内部的所有元素都是有序的
第一个可以称为关键字(key),每个关键字只能在map中出现一次;第二个可能称为该关键字的值(value);
first                                    			second






map插入数据元素操作
//1通过pair的方式插入对象
m.insert(pair<int, string>(3, "小张"));
m.inset(make_pair(-1, "校长"));
//2通过数组的方式插入值--------可以覆盖以前该关键字对应的值
mapStu[3] = "小刘";
mapStu[5] = "小王";





map查找操作
方法一:[]
map<int, int> mp;
cout << mp[1] << endl;

count(key)         返回指定key出现的次数
-------如果有,返回1;否则,返回0。注意,map中不存在相同元素,所以返回值只能是1或0






map大小操作
size();//返回容器中<key,value>的个数





遍历操作
for(auto &t : m)
    cout<<"key:"<<t.first<<" value:"<<t.second<<endl;
 
unordered_map--------其元素的排列顺序是杂乱的,无序的

201412-1csp门禁系统(map)

    采用map存储数据,key存储编号,value存储次数
    边输入边输出,先m[x]++,再输出m[x]
#include<bits/stdc++.h>
#include<unordered_map>
using namespace std;

const int N=1e3+10;

unordered_map<int,int>m;
int n;
int main()
{
	cin>>n;
	int temp;
	for(int i=0;i<n;i++)
	{
		cin>>temp;
		m[temp]++;
		cout<<m[temp]<<" ";
	}
	return 0;
 } 

201409-1csp相邻数对(map)

AcWing 3202. 相邻数对

    为了便于直接寻找相差1的数,我们采用map结构
    map用来判断是否存在相应的值,如果存在则为1,不存在为0
    用vector存储整个数值,然后对每个数值进行单独判断
#include<bits/stdc++.h>
using namespace std;

const int N=1e3+10;

int n;
vector<int> v;//数组用来存储数据遍历一遍
unordered_map<int,int> m;//用哈希表来判断该点是否存在
int ans=0;//返回答案 
int main()
{
	cin>>n;
	for(int i=0;i<n;i++)
	{
		int x;
		cin>>x;
		v.push_back(x);//用数组存储下来 
		m[x]=1; //存在该点 
	}
	for(auto i: v)
	{
		if(m[i-1]!=0)
		{
			m[i-1]=0;
			ans++;
		}
	 }
	 cout<<ans<<endl;
	 return 0; 
 } 

201312-1csp出现次数最多的数(map)

#include<bits/stdc++.h>
using namespace std;
const int N= 1e4+10;
map<int,int> m;

int n;
int main()
{
	cin>>n;
	for(int i=0;i<n;i++)
	{
		int index;
		cin>>index;
		m[index]++;
	}
	
	int max=0,index=0;
	for(auto x:m)
	{
		if(x.second>max)
		{
			index=x.first;
			max=x.second;
		}
	 } 
	 cout<<index<<endl;
	return 0;
}

202006-2csp稀疏向量(map)


map用于存储稀疏数据是最有效的,也可以用来存储稀疏向量。

2个向量不必都存储,能够边读入数据边计算可以节省存储,也有助于提高计算速度。

先读入数据存储在数据结构中,再进行处理是倒腾,既浪费存储又浪费时间,完全没有必要。

#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
map<int,int> M;
LL n,a,b;
LL ans=0;
int main()
{
	cin>>n>>a>>b;
	int index,val;
	for(int i=0;i<a;i++)
	{
		cin>>index>>val;
		M[index]=val;
	}
	for(int i=0;i<b;i++)
	{
		cin>>index>>val;
		ans+=val*M[index];//这里利用了map的性质,快速查找到了index(key)对应的value
	}
	cout<<ans<<endl;
	
	return 0;
 } 

AcWing.3447. 子串计算

3447. 子串计算

#include<bits/stdc++.h>
using namespace std;
const int N =110;

int main()
{
	string str;
	while(cin>>str)
	{
		map<string,int> hash;//map天然字典序---排序 
	
		for(int r=0;r<str.size();r++)//枚举一下右边端点 
			for(int l=0;l<=r;l++)//枚举一下左边端点 			
				hash[str.substr(l,r-l+1)]++;//substr:第一个参数:起始下标,第二个参数:截取字符串的长度(r-l+1) 
	
		for(auto &t:hash)
			if(t.second>1)
				cout<<t.first<<" "<<t.second<<endl;	
	}
	return 0;
 } 

AcWing 3581. 单词识别

AcWing 3581. 单词识别

#include<bits/stdc++.h>
using namespace std;
const int N=1e3+10;

int main()
{
	string str;
	getline(cin,str);
	map<string,int> hash;
	
	for(int i=0;i<str.size();i++)
	{
		int j=i;
		string word;
		if(isalpha(str[j]))//如果遇到的是字母时
		{
			while(j<str.size() && isalpha(str[j]))
				word+=tolower(str[j++]);//将小写字母-->大写字母的函数---toupper()
			i=j;//跳出上次循环时---j指向的是非字母
			hash[word]++; 
		}
			
	}
	
	for(auto &t:hash)
	{
		cout<<t.first<<":"<<t.second<<endl; 
	}
	return 0;
}

第三讲 搜索与图论

leetcode

leetcode.200岛屿数量(bfs/dfs)

leetcode.200岛屿数量(bfs dfs)

leetcode.130被围绕的区域(dfs,bfs)

leetcode.130被围绕的区域(dfs,bfs)

leetcode.547省份数量(dfs,bfs)

leetcode.547省份数量(dfs,bfs)

Flood fill算法—洪水覆盖算法

AcWing1113. 红与黑

acwing1113. 红与黑(bfs dfs)

视频讲解

bfs—Flood fill算法–最短路

#include<bits/stdc++.h> 

#define x first
#define y second

using namespace std;
const int N = 25;
typedef pair<int, int> PII;
int dx[4] = {-1, 0, 1, 0};
int dy[4] = {0, 1, 0, -1};
int n, m;
char g[N][N];


int bfs(int sx, int sy)
{
    queue<PII> q;
    q.push({sx, sy});//起点 
    g[sx][sy] = '#';//被走过 
    int res = 0;//记录个数 

    while(q.size())
    {
        auto t=q.front();
        q.pop();
        res++;//每遍历一个点 

        for(int i=0;i<4;i++)
        {
            int x=t.x+dx[i],y=t.y+dy[i];
            if (x>=0 && x<n && y>=0 && y<m && g[x][y]=='.')//未出界且可走 
            {
            	g[x][y]='#';//标记 
            	q.push({x,y});//加入队列 
			}
            
        }
    }
    return res;
}
int main()
{
    while(cin>>m>>n, n || m)
    {
        int x,y;
        for(int i=0;i<n;i++){
            for(int j=0;j<m;j++)
            {
            	cin>>g[i][j]; 
                if(g[i][j]=='@')
                {
                    x = i;
                    y = j;
                }
            }
        }
        cout<<bfs(x,y)<<endl;
    }
    return 0;
}

dfs—Flood fill算法—代码短

#include<bits/stdc++.h>
using namespace std;
int dx[]={-1,1,0,0},dy[]={0,0,-1,1};
const  int N=25;
int n,m;
char g[N][N];

int dfs(int x,int y)//参数---坐标 
{
	int res=1;//当前格子可以走 
	g[x][y]='#';
	
	for(int i=0;i<4;i++)
	{
		int a=x+dx[i],b=y+dy[i];
		if(a>=0 && a<n && b>=0 && b<m && g[a][b]=='.')
		{
			res+=dfs(a,b);
		}
	 } 
	return res;
}
int main()
{

	while(cin>>m>>n,m||n)//当在一行中读入的是两个零时,表示输入结束。 
	{
		int x,y;
		for(int i=0;i<n;i++){
			for(int j=0;j<m;j++)
			{
				cin>>g[i][j];
				if(g[i][j]=='@')
					x=i,y=j;
			}
		}
		cout<<dfs(x,y)<<endl;	
	 } 


	return 0;
}

AcWing2060. 奶牛选美—尽可能少的区域内涂色—Flood fill算法—枚举

AcWing2060. 奶牛选美
AcWing 2060. 奶牛选美详细证明

题目描述:在一个二维矩阵中,有两个点的集合,找到最短的距离(从一个集合到另一个集合中)距离的计算方式为(曼哈顿距离)
曼哈顿距离:两点间的曼哈顿距离 = | x1 - x2 | + | y1 - y2 |
输出最短距离

问题难点:
二维矩阵使用了字符类型存储
PII 上下左右的定义
dfs广度优先搜索的模板
floodfill算法的证明使用
Flood fill算法是从一个区域中提取若干个连通的点与其他相邻区域区分开(或分别染成不同颜色)的经典算法。

#include<iostream>
#include<bits/stdc++.h>
#include<vector> //数组 
using namespace std;
 
#define x first
#define y second

typedef pair<int,int> PII;
const int N = 55;//此处定义了数据范围 二维数组的大小 

int n,m;
char g[N][N]; 
vector<PII> points[2];//一共有两个点的集合 

int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};

void bfs(int x,int y,vector<PII>& ps)//洪水覆盖算法 
{
	g[x][y]='.';//将搜索过的点做标记 
	ps.push_back({x,y});//放入vector中去 
	
	for(int i=0;i<4;i++)
	{
		int a=x+dx[i];
		int b=y+dy[i];
		
		if(a>=0 && a<n && b>=0 && b<m && g[a][b] == 'X')//如果未超出界,且为x时 
		{
			bfs(a,b,ps);
		}
	}
}
int main()
{
	cin>>n>>m;
	for(int i=0;i<n;i++)
		for(int j=0;j<m;j++)
			cin>>g[i][j];
		 
	
	for(int i=0 ,k=0;i<n;i++)
		for(int j=0;j<m;j++)
			if(g[i][j]=='X')//遇到x时,将位于同一集合中的所有x放入一个vector中去 
				bfs(i,j,points[k++]);//洪水覆盖算法----bfs版本 
	
	int res=1e8;//返回的距离长度,先初始化一个很大的距离
	
	for(auto& a:points[0])//枚举两个集合中的所有点 
		for(auto& b:points[1])
			res=min(res,abs(a.x-b.x)+abs(a.y-b.y)-1);//最优解---曼哈顿距离 

	
	cout<<res;
	return 0;
}

201512-3-csp-画图—洪水覆盖算法

AcWing 3224. 画图
AcWing 3224. 画图(dfs:坐标反着读入,即可正常处理)

由数组坐标系(起点在左上角)---按数学坐标系输出(起点在左下角) 

在这里插入图片描述

DFS写法

#include<bits/stdc++.h>
using namespace std;
const int N=1e2+10;

char g[N][N];//地图 
bool st[N][N];//对某类型的标记 
int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};
int n,m;//n是高度(行),m是宽度(列)


void paint(int x1,int y1,int x2,int y2)//划线函数 
{
	if(x1==x2)//属于同行 
	{
		for(int j=y1;j<=y2;j++)
		{
			if(g[x1][j]=='|'||g[x1][j]=='+')
				g[x1][j]='+';
			else
				g[x1][j]='-';
		 }
	}
	else if(y1==y2)//同列时
	{
		for(int i=x1;i<=x2;i++)
		{
			if(g[i][y1]=='-'||g[i][y1]=='+')
				g[i][y1]='+';
			else
				g[i][y1]='|';
		 } 
	 }
	return ; 
	
 } 
 
void dfs(int x,int y,char C)//洪水覆盖算法 
{
	st[x][y]=true;//对同类型的填充做标记 
	g[x][y]=C;
	
	for(int i=0;i<4;i++)
	{
		int a=x+dx[i];
		int b=y+dy[i];
		if(a<0 || a>=n || b<0 || b>=m || st[a][b])//不越界---未走过 
			continue;
		if(g[a][b]=='-' || g[a][b]=='|' || g[a][b]=='+')//不撞墙 
			continue;
		dfs(a,b,C);
	 } 
 } 
int main()
{
	int q;
	cin>>m>>n>>q; 
	for(int i=0;i<n;i++)//---正常的数组输入---在输入坐标是将(x,y)对调 
		for(int j=0;j<m;j++)
			g[i][j]='.';
	while(q--)
	{
		int op;//操作类型
		cin>>op;
		if(op==0)//划线
		{
			int x1,y1,x2,y2;
			cin>>x1>>y1>>x2>>y2;
			if(x1>x2) swap(x1,x2);//保证小的坐标在前---x/y至少有一项,两个点的坐标相同--同行/同列 
			if(y1>y2) swap(y1,y2);
			paint(y1,x1,y2,x2);//交换x---y 
		}
		else//填充 
		{
			int x,y;
			char C;
			cin>>x>>y>>C;
			memset(st,0,sizeof(st));//这里需要对状态数组重新初始化一下 
			dfs(y,x,C);//交换x---y 
			
		} 
	}
	for(int i=n-1;i>=0;i--)//由数组坐标系(起点在左上角)---按数学坐标系输出(起点在左下角) 
	{
		for(int j=0;j<m;j++)
		{
			cout<<g[i][j];
		}
		cout<<endl;
	 } 

}

BFS写法

#include<bits/stdc++.h>
using namespace std;
const int N=1e2+10;
typedef pair<int,int> PII;
char g[N][N];//地图 
bool st[N][N];//对某类型的标记 
int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};
int n,m;//n是高度(行),m是宽度(列)


void paint(int x1,int y1,int x2,int y2)//划线函数 
{
	if(x1==x2)//属于同行 
	{
		for(int j=y1;j<=y2;j++)
		{
			if(g[x1][j]=='|'||g[x1][j]=='+')
				g[x1][j]='+';
			else
				g[x1][j]='-';
		 }
	}
	else if(y1==y2)//同列时
	{
		for(int i=x1;i<=x2;i++)
		{
			if(g[i][y1]=='-'||g[i][y1]=='+')
				g[i][y1]='+';
			else
				g[i][y1]='|';
		 } 
	 }
	return ; 
	
 } 
 
void bfs(int x,int y,char C)//洪水覆盖算法 
{
	queue<PII> q;
	q.push({x,y});
	st[x][y]=true;//对同类型的填充做标记 
	g[x][y]=C;
	while(q.size())
	{
		auto t=q.front();
		q.pop();
		for(int i=0;i<4;i++)
		{
			int a=t.first+dx[i];
			int b=t.second+dy[i];
			if(a<0 || a>=n || b<0 || b>=m || st[a][b])//不越界---未走过 
				continue;
			if(g[a][b]=='-' || g[a][b]=='|' || g[a][b]=='+')//不撞墙 
				continue;
			q.push({a,b});
			st[a][b]=true;//对同类型的填充做标记 
			g[a][b]=C;
		
	 	} 
	}
	
 } 
int main()
{
	int q;
	cin>>m>>n>>q; 
	for(int i=0;i<n;i++)//---正常的数组输入---在输入坐标是将(x,y)对调 
		for(int j=0;j<m;j++)
			g[i][j]='.';
	while(q--)
	{
		int op;//操作类型
		cin>>op;
		if(op==0)//划线
		{
			int x1,y1,x2,y2;
			cin>>x1>>y1>>x2>>y2;
			if(x1>x2) swap(x1,x2);//保证小的坐标在前---x/y至少有一项,两个点的坐标相同--同行/同列 
			if(y1>y2) swap(y1,y2);
			paint(y1,x1,y2,x2);//交换x---y 
		}
		else//填充 
		{
			int x,y;
			char C;
			cin>>x>>y>>C;
			memset(st,0,sizeof(st));//这里需要对状态数组重新初始化一下 
			bfs(y,x,C);//交换x---y 
			
		} 
	}
	for(int i=n-1;i>=0;i--)//由数组坐标系(起点在左上角)---按数学坐标系输出(起点在左下角) 
	{
		for(int j=0;j<m;j++)
		{
			cout<<g[i][j];
		}
		cout<<endl;
	 } 

}

DFS—深度优先搜索

AcWing 842. 排列数字

AcWing 842. 排列数字
AcWing 842. 排列数字–深度优先遍历代码+注释

#include<iostream>
using namespace std;
const int N = 10;
int path[N];//保存序列
int st[N];//数字是否被用过
int n;
void dfs(int u)
{
    if(u>n)//数字填完了,输出
    {
        for(int i=1;i<=n;i++)//输出方案
            cout<<path[i]<<" ";
        cout<<endl;
    }

    for(int i=1;i<=n;i++)//空位上可以选择的数字为:1 ~ n
    {
        if(!st[i])//如果数字 i 没有被用过
        {
            path[u] = i;//放入空位
            st[i]=1;//数字被用,修改状态
            dfs(u+1);//填下一个位
            st[i]=0;//回溯,取出 i
            path[u]=0;
        }
    }
}

int main()
{
    cin >> n;
    dfs(1);
}


AcWing 843. n-皇后问题

AcWing 843. n-皇后问题

代码分析

    对角线 dg[u+i]

,反对角线udg[n−u+i]中的下标 u+i和 n−u+i

    表示的是截距

下面分析中的(x,y)
相当于上面的(u,i)

    反对角线 y=x+b

, 截距 b=y−x,因为我们要把 b 当做数组下标来用,显然 b 不能是负的,所以我们加上 +n
(实际上+n+4,+2n都行),来保证是结果是正的,即 y - x + n
而对角线 y=−x+b
, 截距是 b=y+x

    ,这里截距一定是正的,所以不需要加偏移量

核心目的:找一些合法的下标来表示dg
或udg是否被标记过,所以如果你愿意,你取 udg[n+n−u+i] 也可以,只要所有(u,i)对可以映射过去就行

AcWing 1432. 棋盘挑战—对角线—八皇后问题

AcWing 1432. 棋盘挑战
在这里插入图片描述
在这里插入图片描述

dfs--天然保证了按照字典序
#include<bits/stdc++.h>
using namespace std;
const int N=15;

int n;
bool col[N];//用于标志这列是否被存放了
bool dg[N*2];//正对角线上是否有了元素  true:有 ;false :无 
//正对角线 ---存储b
//y=x+b  y=x+b  -> b=x-y; 
bool udg[N*2]; //反对角线上是否有了元素  true:有 ;false :无
//副对角线 ---存储b
//y=-x+b y=-x+b  ->b=x+y; 
int path[N];//方案 
int ans;
void dfs(int x)
{
	if(x>n)//搜索到第n行---搜索到答案 
	{
		ans++;
		if(ans<=3)
		{
			for(int i=1;i<=n;i++)	cout<<path[i]<<" ";
			cout<<endl;
		}
		return ;
	}
	
	for(int y=1;y<=n;y++)//枚举一下放到第几列 
	{
		if(!col[y]&&!dg[x+y]&&!udg[x-y+n])//该列、正对角线、副对角线上都没有数 
		{
			path[x]=y;
			col[y]=dg[x+y]=udg[x-y+n]=true;//标记一下 
			
			dfs(x+1);//搜索下一层 
			
			col[y]=dg[x+y]=udg[x-y+n]=false;//恢复现场---清空 
			path[x]=0;
		}
	}
}
int main()
{
	cin>>n;
	dfs(1);
	cout<<ans<<endl;
	return 0;
 } 


201709-4-csp-通信网络—正反向两次dfs—图论

AcWing 3250. 通信网络
AcWing 3250. 通信网络—两次dfs

#include<bits/stdc++.h>
using namespace std;
const int N =1e3+10;//顶点 
const int M =1e4+10;//边
bool st1[N];
bool st2[N];
//邻接表---正向存储 
int idx1;
int e1[M];
int ne1[M];
int h1[N];

void add1(int a,int b)
{
	e1[idx1]=b;
	ne1[idx1]=h1[a];
	h1[a]=idx1++;
}
//邻接表---逆向存储
int idx2;
int e2[M];
int ne2[M];
int h2[N];


void add2(int a,int b)
{
	e2[idx2]=b;
	ne2[idx2]=h2[a];
	h2[a]=idx2++;
}


int n,m;

void dfs1(int u)
{
	st1[u]=true;//当前点已经被遍历过了
	for(int i=h1[u];i!=-1;i=ne1[i])
	{
		int j=e1[i];
		if(!st1[j])
			dfs1(j);	
	 } 	
}
void dfs2(int u)
{
	st2[u]=true;//当前点已经被遍历过了
	for(int i=h2[u];i!=-1;i=ne2[i])
	{
		int j=e2[i];
		if(!st2[j])
			dfs2(j);	
	 } 	
}

int main()
{
	cin>>n>>m;
	memset(h1,-1,sizeof(h1));//邻接表---顶点表---初始化
	memset(h2,-1,sizeof(h2));
	for(int i=0;i<m;i++)
	{
		int a,b;
		cin>>a>>b;
		add1(a,b),add2(b,a);//邻接表  与  逆邻接表
	}
	
	int ans=0;
	for(int i=1;i<=n;i++)
	{
		memset(st1,0,sizeof(st1));
		memset(st2,0,sizeof(st2));
		dfs1(i);//正向搜索
		dfs2(i);//反向搜索
		int sum=0;
		for(int j=1;j<=n;j++)
		{
			if(st1[j]||st2[j])//只要有一个方向能搜索到就算可以
				sum++;
		}
		if(sum==n)
			ans++;
	}
	cout<<ans<<endl;
	return 0;
 } 



201312-5-csp-I’m stuck!—正反两个方向dfs

AcWing 3196. I’m stuck!

AcWing 3196. I’m stuck!:矩阵地图遍历 + 详细代码注释

#include<bits/stdc++.h>
using namespace std;
const int N=55;
char g[N][N];//存储地图 
bool st1[N][N];//true:从起点可以走到[i][j] 
bool st2[N][N];//true:从[i][j] 可以走到终点 
int R,C;
int ans;//返回符合答案的数目 
int dx[4]={-1,0,1,0};//上0-左1-下2-右3 
int dy[4]={0,1,0,-1};

bool check(int x,int y,int i)//判断是否能走 
{
	if(g[x][y]=='+'||g[x][y]=='S'||g[x][y]=='T')	return true;
	if(g[x][y]=='-'&& i%2!=0) return true;
	if(g[x][y]=='|'&& i%2==0)	return true;
	if(g[x][y]=='.'&& i==2) return true;
	return false;
}
void dfs1(int x,int y)//深度优先遍历,求出 S 能到的所有点
{
	st1[x][y]=true;
	for(int i=0;i<4;i++)
	{
		int a=x+dx[i];
		int b=y+dy[i];
		if(a<1||a>R || b<1||b>C ||g[a][b]=='#' ) 	continue;//走的位置不合法
		if(st1[a][b]) 	continue;//如果走过---跳过 
		if(check(x,y,i))//如果能走过去
			dfs1(a,b);
	 } 
}
void dfs2(int x,int y)//深度优先遍历,求出能走到 T 的所有点
{
	st2[x][y]=true;
	for(int i=0;i<4;i++)
	{
		int a=x+dx[i];
		int b=y+dy[i];
		if(a<1||a>R || b<1||b>C ||g[a][b]=='#' )	continue;//走的位置不合法
		if(st2[a][b])	continue;//如果走过---跳过
		if(check(a,b,i ^ 2))//如果能反向走过去---注意是那个点[a][b]---而不是[x][y]
			dfs2(a,b); 	
	}
 } 

int main()
{
	cin>>R>>C;
	for(int i=1;i<=R;i++)
		for(int j=1;j<=C;j++)
			cin>>g[i][j];
			
			
	int edx,edy;//记录下终点 
	for(int i=1;i<=R;i++)
		for(int j=1;j<=C;j++)
		{
			if(g[i][j]=='S')
				dfs1(i,j);//从起点开始dfs搜索可以到达的点 
			if(g[i][j]=='T')
			{
				edx=i,edy=j;
				dfs2(i,j);//从终点开始dfs搜索可以到达的点 
			}
		}
	if(!st1[edx][edy])//如果从起点到终点不可达直接输出
	{
		cout<<"I'm stuck!"<<endl;
		return 0; 
	} 
			
	for(int i=1;i<=R;i++)
		for(int j=1;j<=C;j++)
		{
			//cout<<st1[i][j]<<" "<<!st2[i][j]<<endl;
			if(st1[i][j] && !st2[i][j]) // S 能到 g[i][j], g[i][j] 不能到 T
				ans++;	 
		}
	cout<<ans<<endl;
	return 0;
}

201503-4-csp-网络延时—图论

AcWing 3215. 网络延时
详细讲解

#include<bits/stdc++.h>
using namespace std;
const int N=2e4+10;//这里交换机和计算机一起存储
const int M=N;
//邻接表存储图 
int idx;
int e[M];
int ne[M];
int h[N];

int n,m;
int ans;//最大深度+次大深度 
void add(int a,int b)
{
	e[idx]=b;
	ne[idx]=h[a];
	h[a]=idx++;
}


int dfs(int u)
{
	int d1=0,d2=0;//d1保存最大深度,d2保存次大深度
	
	for(int i=h[u];i!=-1;i=ne[i])
	{
		int j=e[i];
		int d=dfs(j);//返回下一层的最大深度
		
		if(d>d1)
			d2=d1,d1=d;
		else if(d>d2)
			d2=d; 
	 }
	 ans=max(ans,d1+d2);
	 return d1+1; 
 } 



int main()
{
	cin>>n>>m;
	memset(h,-1,sizeof(h));
	for(int i=2;i<=n;i++)
	{
		int p;
		cin>>p;
		add(p,i);//添加一条p->i的边---交换机->交换机 
	}
	for(int i=n+1;i<=n+m;i++)
	{
		int p;
		cin>>p;
		add(p,i);//添加一条p->i的边 ---交换机->计算机 
	 }
	 dfs(1);
	 cout<<ans<<endl; 
	return 0;
}

BFS—广度优先搜索

AcWing 844. 走迷宫—最短路(权重相同时)

AcWing 844. 走迷宫

#include<bits/stdc++.h>
using namespace std;
const int N=1e2+10;
typedef pair<int,int> PII;
#define x first
#define y second
int g[N][N]; 
int d[N][N];//存储从起点到达[i,j]位置的最短路径
int dx[4]={0,0,-1,1};
int dy[4]={-1,1,0,0}; 
int n,m;
int bfs()
{
	queue<PII> q;
	q.push({1,1});//将起点加入队列
	while(q.size()) 
	{
		auto t=q.front();//队首元素 
		q.pop();
		for(int i=0;i<4;i++)//将上下左右加入队列 
		{
			int X=t.x+dx[i];
			int Y=t.y+dy[i];
			if(X>=1 && X<=n && Y>=1 && Y<=m && g[X][Y]==0 && d[X][Y]==0)//未出界、不是墙、未走过、
			{
				d[X][Y]=d[t.x][t.y]+1;
				q.push({X,Y}); 
			 } 
		}
		
	}
	return d[n][m];//返回到达[n,m]的距离 
}
int main()
{

	cin>>n>>m;
	for(int i=1;i<=n;i++)
		for(int j=1;j<=m;j++)
			cin>>g[i][j];
	cout<<bfs()<<endl; 
	return 0;
 } 

201604-4-csp-游戏—最短路—拆点(通过升维来对状态点进行细分)

AcWing 3230. 游戏

详细讲解

#include<bits/stdc++.h>
using namespace std;
const int N=110;
const int M=310;//最多花费300秒的时间就能走到终点 
bool g[N][N][M];//---g[x][y][t]---t时刻,(x,y)坐标是否能走---true:表示不能走
bool st[N][N][M];//---st[x][y][t]:表示(x,y)坐标 在t时刻是否走过---为了不重复走 

int dx[]={-1,0,1,0};
int dy[]={0,1,0,-1};
int n,m,t;
struct Node{//结构体---方便队列存储信息 
	int x,y;
	int t;
};

int bfs()
{
	queue<Node> q;
	st[1][1][0]=true;//在0时刻---[1][1]位置已经走过 
	q.push({1,1,0});//将起点加入队列 
	
	while(q.size())
	{
		auto z=q.front();
		q.pop();
		
		for(int i=0;i<4;i++)
		{
			int x=z.x+dx[i];
			int y=z.y+dy[i];
			int t=z.t+1;
			
			if(x<1 || x>n || y<1 || y>m || g[x][y][t]) continue;//如果越界 或 [x][y]在t时刻不能走 
			if(st[x][y][t]) continue;//[x][y]在t时刻---已经经过该位置了 
				
			if(x==n && y==m)//到达终点 
				return t;
			st[x][y][t]=true;
			q.push({x,y,t});
		}
	}
	return -1;
}
int main()
{
	cin>>n>>m>>t;
	while(t--)
	{
		int r,c,a,b;
		cin>>r>>c>>a>>b;
		for(int i=a;i<=b;i++)
		{
			g[r][c][i]=true;//表示x,y这个格子在i时刻不能走
		}
	}
	int t=bfs();
	cout<<t<<endl;
	return 0;
}

201409-4-csp-最优配餐—最短路

AcWing 3205. 最优配餐
AcWing 3205. 最优配餐——多源BFS

#include<bits/stdc++.h>
using namespace std;
const int N =1e3+10;
const int INF =0x3f3f3f3f;
typedef long long LL;
typedef pair<int,int> PII;

bool st[N][N];//标记已经走过---和---原本就是障碍物 
int dist[N][N];//记录---多源起点到达该位置的最短距离
 
struct target{//存储买家的位置---订单数量 
	int x;
	int y;
	int c;
}Tg[N*N];

queue<PII> q;//bfs---队列---将坐标加入队列 
int n,m,k,d;
int dx[4]={-1,0,1,0};//上-右-下-左 
int dy[4]={0,1,0,-1};

void bfs()
{
	while(q.size())
	{
		auto t=q.front();
		q.pop();
		
		for(int i=0;i<4;i++)
		{
			int a=t.first+dx[i];
			int b=t.second+dy[i];
			if(a>=1 && a<=n && b>=1 && b<= n && !st[a][b])
			{
				if(dist[a][b]>dist[t.first][t.second]+1)
				{
					dist[a][b]=dist[t.first][t.second]+1;//最短路径算法---边权相同---bfs 
					st[a][b]=true;
					q.push({a,b});
				}
			}
		}
	}
}
int main()
{
	cin>>n>>m>>k>>d;
	memset(dist,INF,sizeof(dist));//初始化一下所有位置---为不可达 
	while(m--)
	{
		int a,b;
		cin>>a>>b;
		dist[a][b]=0;//-----------将多元起点---标记为起点---初始距离 
		q.push({a,b});//将起点加入队列 
	}
	for(int i=0;i<k;i++)
		cin>>Tg[i].x>>Tg[i].y>>Tg[i].c;
	
	while(d--)
	{
		int a,b;
		cin>>a>>b;
		st[a][b]=true;//表示该位置不可达 
		}	

	bfs();
	LL ans=0;
	
	for(int i=0;i<k;i++)
		ans+=dist[Tg[i].x][Tg[i].y]*Tg[i].c;

	cout<<ans<<endl;
 
	return 0;
 } 

201403-4-csp-无线网络–拆点(通过升维来对状态点进行细分)

AcWing 3200. 无线网络

详细题解

#include<bits/stdc++.h>
using namespace std;
const int N=210;//非特殊点+特殊点 
const int M=N*N;//无向图 
const int INF=0x3f3f3f3f; 
typedef pair<int,int> PII;//存储每个点的位置信息 
PII p[N];
int dist[N][N]; // 最短距离,从1->i---dist[i][j]---i:到达点的编号,j:路线中经过了几个特殊点 
#define x first
#define y second
typedef long long LL;

int n,m,k,r;

//邻接表存储图
int idx;
int e[M];
int ne[M];
int h[N];

void add(int a,int b)//边权都为1 
{
	e[idx]=b;
	ne[idx]=h[a];
	h[a]=idx++; 
 } 

bool check(PII a, PII b)
{
	LL dx=a.x-b.x;
	LL dy=a.y-b.y;
	return (dx*dx+dy*dy<=(LL)r*r);
 } 
int bfs()
{
	memset(dist,INF,sizeof(dist));
	dist[1][0]=0;//到达1号点,经过0个特殊点,路径距离=0 
	
	queue<PII> q;//----此时存储x:几号点,y经过几个特殊点 
	q.push({1,0});
	
	while(q.size())
	{
		auto t=q.front();
		q.pop();
		
		for(int i=h[t.x];i!=-1;i=ne[i])
		{
			int j=e[i];//以i为边-----------t.x->j
			int y=t.y;
			if(j>n)//说明此时选择的终点---是特殊点 
				y++;
			if(y<=k)
			{
				if(dist[j][y]>dist[t.x][t.y]+1)
				{
					dist[j][y]=dist[t.x][t.y]+1;
					q.push({j,y});
				}
			 } 
		}
	 }
	 
	 int ans=1e8;
	 for(int i=0;i<=k;i++)
	 {
	 	ans=min(ans,dist[2][i]);
	  }
	return ans-1; 
}
int main()
{
	cin>>n>>m>>k>>r;
	memset(h,-1,sizeof(h)); 
	for(int i=1;i<=n;i++) cin>>p[i].x>>p[i].y;//原有路由器 
	for(int i=n+1;i<=n+m;i++)	cin>>p[i].x>>p[i].y;//新增路由器
	

	//初始化---点与点之间的距离 
	for(int i=1;i<=n+m;i++)
	{
		for(int j=i+1;j<=n+m;j++)
		{
			if(check(p[i],p[j]))
			{
				add(i,j),add(j,i);//无向图--每条边存储两次 
			}
		}
	}
	
	
	cout<<bfs()<<endl;
	
	return 0;
 } 

201509-4-csp-高速公路

AcWing 3220. 高速公路
暴力bfs 60分

#include <bits/stdc++.h>

using namespace std;

const int N = 1010, M = N * N;
#define x first
#define y second
typedef pair<int,int> PII;


int h[N], e[M], ne[M], idx;
int n, m;
bool g[N][N];

void add(int a, int b){
    e[idx] = b, ne[idx] = h[a], h[a] = idx++;
}

queue<PII> q;

void bfs()
{
    while(q.size())
    {
        auto t = q.front();
        q.pop();

        for(int j = h[t.y]; j != -1; j = ne[j])
        {
            if(g[t.x][e[j]])continue;
            q.push({t.x, e[j]});
            g[t.x][e[j]] = true;
        }
    }
}

int main()
{
    cin >> n >> m;
    memset(h, -1, sizeof h);
    for(int i = 0; i < m;i  ++)
    {
        int a, b;
        cin >> a >> b;
        add(a, b);
        q.push({a, b});
    }

    bfs();
    int res = 0;
    for(int i = 1; i <= n; i ++)
    {
        for(int j = 1; j < i; j ++)
        {
            if(g[i][j] && g[j][i])res ++;

        }
    }

    cout << res << endl;
    return 0;
}

AcWing1101. 献给阿尔吉侬的花束—最短路

acwing1101. 献给阿尔吉侬的花束(bfs)

#include<iostream>
#include<bits/stdc++.h>
using namespace std;
#include<queue>//广度优先搜索 
#define x first
#define y second
typedef pair<int,int> PII;
const int N=210;
int dx[]={-1,1,0,0},dy[]={0,0,-1,1};

int n,m;
char g[N][N];
int dist[N][N];


int bfs(PII start)
{
	queue<PII> Q;
	Q.push(start);
	memset(dist,-1,sizeof(dist)); //初始化数组为-1 
	dist[start.x][start.y]=0;//起点的距离为0 
	
	while(Q.size())
	{
		auto t =Q.front();
		Q.pop();
		for(int i=0;i<4;i++)
		{
		int x=t.x+dx[i],y=t.y+dy[i];//上下左右的坐标 
		if(x>=0&&x<n&&y>=0&&y<m&&g[x][y]!='#'&&dist[x][y]==-1)//如果没有越界 
			{
				dist[x][y]=dist[t.x][t.y]+1;//更新距离 
				if(g[x][y]=='E')//到达终点 
					return dist[x][y];
				Q.push({x,y});//加入队列 
			}
			
		}
	}
return -1;//如果没有找到终点---当队列中没有点时----所有能走到的点都被标记了--(距离非0) 
}
int main()
{
	int T;
	cin>>T;
	while(T--)
	{
		cin>>n>>m;
		PII start;
		for(int i=0;i<n;i++)
			for(int j=0;j<m;j++)
			{
				cin>>g[i][j]; 
				if(g[i][j]=='S')//如果为S,则为起点 
					start={i,j};
			}
		int res=bfs(start);
		if(res==-1)	cout<<"oop!"<<endl;
		else cout<<res<<endl;
	}
	return 0;
}


AcWing 845. 八数码—至少需要进行多少次交换

AcWing 845. 八数码
详细讲解

#include<bits/stdc++.h>
using namespace std;
int dx[4]={1,-1,0,0};
int dy[4]={0,0,-1,1};

int bfs(string strat)
{
	string ed="12345678x";//定义结束状态
	queue<string> q;
	unordered_map<string,int> m;
	
	q.push(strat);
	m[strat]=0;//起点距离
	
	while(q.size())
	{
		auto t=q.front();
		q.pop();
		int dist=m[t];//如果是最终状态则返回距离
		if(t==ed)
			return dist; 
			
		int index=t.find('x');//查询x在字符串中的下标,然后转换为在矩阵中的坐标
		int x=index/3;
		int y=index%3;
		
		for(int i=0;i<4;i++)
		{
			int X=x+dx[i]; //求转移后x的坐标
			int Y=y+dy[i];
			if(X>=0 && X<3 && Y>=0 && Y<3) //当前坐标没有越界
			{
				swap(t[index],t[X*3+Y]);//转移x
				if(!m.count(t))//如果当前状态是第一次遍历,记录距离,入队
				{
					m[t]=dist+1;
					q.push(t);
				}
				swap(t[index],t[X*3+Y]);//还原状态,为下一种转换情况做准备
			}
		}
	 }
	 //无法转换到目标状态,返回-1
	 return -1; 
}
int main()
{
	string c,strat;//用字符串读入-----X 
	for(int i=1;i<=9;i++)
	{
		cin>>c;//字符串的输入是以空格截止 
		strat+=c;//字符串的拼接 
	}
	cout<<bfs(strat)<<endl; 
	return 0;
 } 

AcWing 3385. 玛雅人的密码

AcWing 3385. 玛雅人的密码

#include<bits/stdc++.h>
using namespace std;

int n;
int bfs(string start)
{
	queue<string> q;//存储字符串---广搜--队列
	unordered_map<string,int> dist;//存储到达字符串t的最近距离
	dist[start]=0;
	q.push(start);
	
	while(q.size())
	{
		auto t=q.front();
		q.pop();
		
		if(t.find("2012")!=-1)//在字符串中查找目标状态
			return dist[t];
			
		for(int i=1;i<n;i++)//广度搜索
		{
			string r=t;
			swap(r[i],r[i-1]);
			if(!dist.count(r))//如果是第一次搜索到这个字符串---加入队列
			{
				dist[r]=dist[t]+1;
				q.push(r);
			}
		}
	}
	return -1;
}
int main()
{
	string start;
	cin>>n;
	cin>>start;
	cout<<bfs(start)<<endl;//广度优先搜索
 } 

拓扑排序

AcWing 848. 有向图的拓扑序列

AcWing 848. 有向图的拓扑序列
AcWing 848. 拓扑排序−−思路介绍+图解模拟+详细代码注释

#include<bits/stdc++.h>
using namespace std;
const int N=1e5+10;
const int M=1e5+10;
//邻接表存储有向图
int idx;
int e[M];
int ne[M];
int h[N];


int cnt[N];//记录每个点的入度
vector<int> ans;//保存拓扑排序---点的顺序 

int n,m;

void add(int a,int b)
{
	e[idx]=b,ne[idx]=h[a],h[a]=idx++;
}
int topsort()
{
	queue<int> q;

	for(int i=1;i<=n;i++)//将初始时 入度为0 的点加入队列 
		if(cnt[i]==0)   
		    q.push(i);
	
	while(q.size())
	{
		auto t=q.front();
		q.pop();
		ans.push_back(t);//出队时,将其加入拓扑答案中
		
		for(int i=h[t];i!=-1;i=ne[i])
		{
			int j=e[i];// t->j的边
			cnt[j]--;//终点的入度--
			if(cnt[j]==0)
				q.push(j); 
		 } 
	 }
	  
	 if(ans.size()!=n)//ans中点的个数不等于n时,说明还有点不可以进行拓扑排序
	 	return -1;
	
}

int main()
{

	cin>>n>>m;
	memset(h,-1,sizeof(h));//经典错误---标准零分 
	for(int i=1;i<=m;i++)
	{
		int a,b;
		cin>>a>>b;
		add(a,b);
		cnt[b]++;//入度++ 	
	}
	
	auto t=topsort();

	if(t==-1)
		cout<<-1<<endl;
	else
		for(auto &x:ans)
			cout<<x<<" ";
	
	return 0;
}



最短路径之Dijkstra算法

在这里插入图片描述

AcWing 849. Dijkstra求最短路 I—(重边√有环√—非负—有向/无向图√)—应用于稠密图

Dijkstra求最短路 I:图解 详细代码(图解)

#include<bits/stdc++.h>
using namespace std;
const int N =500+10;
const int INF=0x3f3f3f3f;
int g[N][N];
bool st[N];
int dist[N];
int n,m;
int dijkstra()
{
	memset(dist,INF,sizeof(dist));
	dist[1]=0;//到1号点的距离为0 
	
	for(int i=1;i<n;i++)//再选择n-1个点 
	{
		int t=-1;
		for(int j=1;j<=n;j++)//选择最近的点
		{
			if(!st[j] && (t==-1||dist[j]<dist[t]))
				t=j; 
		 }
		 
		 st[t]=true;//状态数组---第一次选择的是起点---终点最终没有被选---终点是最后一个点
		 
		 for(int j=1;j<=n;j++)
		 {
		 	dist[j]=min(dist[j],dist[t]+g[t][j]);
		  } 
	 }
	 if(dist[n]!=INF)
	 	return dist[n];
	else
		return -1; 
}
int main()
{
	cin>>n>>m;
	memset(g,INF,sizeof(g));
	while(m--)
	{
		int x,y,c;
		cin>>x>>y>>c;
		g[x][y]=min(g[x][y],c);//盘重边 
	}
	cout<<dijkstra()<<endl; 
	return 0;
 } 

AcWing 850. Dijkstra求最短路 II------(重边√有环√—非负—有向/无向图√)—应用于稀疏图

在这里插入图片描述
请添加图片描述

#include<bits/stdc++.h>
using namespace std;
const int N=1e6+10;//边的最大数量 ---点的最大数量 
const int INF=0x3f3f3f3f;
typedef pair<int,int> PII;//存储每个点的信息---first:起点到i的距离,second:编号i
//点的信息
bool st[N]; 
int dist[N];

//邻接表存储方式 ---存储边
int idx;//边的编号
//编号为idx的边---的终点为e[i]
int e[N];//终点的编号---下标:边---存储:终点 
//上一条以a为起点的最大序号的边的序号是head[a] 
int ne[N];//起点的编号---下标:边---存储:起点(链接边表结点)
//编号为idx的边---的权值为w[i] 
int w[N];//存储第idx边的边权 ---下标:边---存储:边权

//---最后存进图的以x为起点的边的编号为h[x]
int head[N];//表头结点表---下标:点---存储:边



int n,m;
void add(int a,int b,int c)//输入的每条边都是由父节点 (a)子节点(b)和边权(c)组成的
{
	ne[idx]=head[a];//注意 ---(链接边表结点)上一条以a为起点的最大序号的边的序号是head[a] 
	e[idx]=b;
	w[idx]=c;
	head[a]=idx++;//现在以a为起点的最大序号的边的序号是idx了 
}
int dijkstra()
{
	memset(dist,INF,sizeof(dist));
	dist[1]=0;
	priority_queue<PII,vector<PII>,greater<PII>> heap;//定义小根堆
	heap.push({0,1});//把1号点放进去
	
	while(heap.size())
	{
		auto t=heap.top();
		heap.pop();//最小值出队---每次找到距离最小的点 
		
		int ver=t.second;
		int distance=t.first;
		
		
		if(st[ver])//如果更新过某点了---如果添加到S集合中了 
			continue;
		st[ver]=true;
		//用这个集合S所能到达的最小值更新---该点能到达的边的dist 
		for(int i=head[ver];i!=-1;i=ne[i])//i:边的编号 
		{
			int j=e[i];//这条边所能到达的终点---j:点的编号
			if(dist[j]>dist[ver]+w[i]) 
			{
				dist[j]=dist[ver]+w[i];//更新起点到任意一点的最短路
				heap.push({dist[j],j}); 
			}
		}
	 }
	 if(dist[n]!=INF)
	 	return dist[n];
	else
		return -1; 
}
int main()
{
	cin>>n>>m;
	memset(head,-1,sizeof(head));//把表头初始化为空节点 
	
	while(m--)
	{
		int a,b,c;
		cin>>a>>b>>c;
		add(a,b,c);//堆---不怕有重边
	 }
	 cout<<dijkstra()<<endl; 
	return 0;
 } 

bellman-ford—有边数限制的最短路问题

AcWing 853. 有边数限制的最短路

AcWing 853. 有边数限制的最短路
题干讲解

#include<bits/stdc++.h>
using namespace std;
const int N=500+10,M=1e4+10;
const int INF=0x3f3f3f3f;
//边结点数组 
struct{
	int a,b,w;
}edges[M];
int dist[N];
int backup[N];//辅助数组---防止串联 ---上一次迭代后 dist[] 数组的备份 
int n,m,k;

void bellman_ford()
{
	memset(dist,INF,sizeof(dist));
	dist[1]=0;
	//若在 n-1 次松弛后还能更新,则说明图中有负环,因此无法得出结果,否则就完成。
	for(int i=0;i<k;i++)//走k步 
	{
		memcpy(backup,dist,sizeof(dist));
		
		for(int j=0;j<m;j++)
		{
			int a=edges[j].a;
			int b=edges[j].b;
			int w=edges[j].w;
			dist[b]=min(dist[b],backup[a]+w);
		 } 
	}
}
int main()
{
	cin>>n>>m>>k;
	for(int i=0;i<m;i++)
	{
		int a,b,c;
		cin>>a>>b>>c;
		edges[i]={a,b,c};
	 }
	 
	 bellman_ford();
//INF是一个确定的值,并非真正的无穷大,会随着其他数值而受到影响,dist[n]大于某个与INF相同数量级的数即可	 
	 if(dist[n]>INF/2)
	 	cout<<"impossible"<<endl;
	else
		cout<<dist[n]<<endl;
	  
	return 0;
}

spfa

AcWing 851. spfa求最短路-带负权—有点小小万能的感觉?

AcWing 851. spfa求最短路

#include<bits/stdc++.h>
using namespace std;
const int N=1E5+10;
const int INF=0x3f3f3f3f;
bool st[N];//标记顶点是不是在队列中
int dist[N];//保存最短路径的值
//图的邻接表存储 
int idx;
int e[N];
int ne[N];
int w[N];
int h[N]; 

int n,m;
void add(int a,int b,int c)
{
	ne[idx]=h[a];
	e[idx]=b;
	w[idx]=c;
	h[a]=idx++;
 } 
int spfa()
{
	memset(dist,INF,sizeof(dist));
	dist[1]=0;
	queue<int> q;
	q.push(1);//将起点加入队列
	st[1]=true;
	
	while(q.size())
	{
		auto t=q.front();
		q.pop();
		st[t]=false;//从队列中取出来之后该节点st被标记为false,代表之后该节点如果发生更新可再次入队
		
		for(int i=h[t];i!=-1;i=ne[i])//遍历邻接表中的顶点结点---后的相连边
		{
			int j=e[i];//获得和i相连的点
			if(dist[j]>dist[t]+w[i])//如果可以距离变得更短,则更新距离
			{
				dist[j]=dist[t]+w[i];//更新距离
				
		//当前已经加入队列的结点,无需再次加入队列,即便发生了更新也只用更新数值即可,重复添加降低效率
				if(!st[j])//如果没在队列中
				{
					q.push(j);//入队
					st[j]=true;//打标记
				 } 
			}
		}
		
	}
    return dist[n];
}
int main()
{
	cin>>n>>m;
	memset(h,-1,sizeof(h));//注意---别忘了初始化---顶点表
	while(m--)
	{
		int a,b,c;
		cin>>a>>b>>c;
		add(a,b,c);//有向图  如果是无向图---+add(b,a,c) 
	}
	auto t=spfa();
	if(t==INF)
		cout<<"impossible"<<endl;
	else
		cout<<t<<endl;
	return 0; 
}

201609-4-csp-交通规划

AcWing 3235. 交通规划
详细讲解

#include<bits/stdc++.h>
using namespace std;

const int N=1e4+10;
const int M=2e5+10;//无向图存储---两条边
const int INF=0x3f3f3f3f;
bool st[N];
int dist[N];

//邻接表存储图
int e[M];//边 
int ne[M];
int w[M];
int h[N]; //点
int idx;
int n,m;

void add(int a,int b,int c)//向有向图中添加边---邻接表 
{
	e[idx]=b;
	w[idx]=c;
	ne[idx]=h[a];//注意---这里又写错了 
	h[a]=idx++;
 } 
 
void spfa()//-----spfa的代码还是不熟练
{
	memset(dist,INF,sizeof(dist));
	dist[1]=0; 
	queue<int> q;//注意---队列里存储的是--点的编号 
	q.push(1);
	st[1]=true;//表示在队列中 
	
	while(q.size())
	{
		auto t=q.front();
		q.pop();
		st[t]=false;
	
		
		for(int i=h[t];i!=-1;i=ne[i])
		{
			int j=e[i];//  t->j这条边
			if(dist[j]>dist[t]+w[i])
			{
				dist[j]=dist[t]+w[i];
				if(!st[j])//注意---这里需要判断一下是否需要加入队列 
				{
					st[j]=true;
					q.push(j);//更新过谁---我就拿这个更新过的点来更新别人
				}
			 } 
		 } 
		
	}
}
int main()
{
	cin>>n>>m;
	memset(h,-1,sizeof(h));
	
	while(m--)
	{
		int a,b,c;
		cin>>a>>b>>c;
		add(a,b,c);//无向图 
		add(b,a,c);
	}

	spfa();
	
	 //求每个点满足条件的邻边(每个点的满足条件的边中权值最小的边就是我们要找的边)
	int ans=0;
	for(int a=2;a<=n;a++)//从其余的n-1个点中选
	{
		int minw=INF;//用minw表示权值最小的边
		for(int i=h[a];i!=-1;i=ne[i])
		{
			int j=e[i];
			if(dist[a]==dist[j]+w[i])
			{
				minw=min(minw,w[i]);
			}
		}
		ans+=minw;//将每个点满足条件的权值最小边累加起来 
	 } 
	cout<<ans<<endl;
	 
	return 0;
 } 

201903-5-csp-317号子任务

AcWing 3276. 317号子任务

详细讲解

#include<bits/stdc++.h>
using namespace std;
const int N=1e4+10;
const int M=2e4+10;//无向图---每条边存储两次
const int INF=0x3f3f3f3f;
int idx;
int e[M];
int ne[M];
int w[M];
int h[N];

int dist[N];
int st[N];

int ds[N][1010];
int cnt;//记录到第几个行星发动机据点
int type[N];//记录据点的类型
int n,m,k;

void add(int a,int b,int c)
{
	e[idx]=b;
	w[idx]=c;
	ne[idx]=h[a];
	h[a]=idx++;
}
void spfa(int start)//每次从起点strat开始求---单源最短路径
{
	memset(dist,INF,sizeof(dist));//每次单元最短路径都有自己的dist st数组
	memset(st,0,sizeof(st));
	
	dist[start]=0;
	queue<int> q;
	q.push(start);
	st[start]=true;
	
	while(q.size())
	{
		auto t=q.front();
		q.pop();
		
		st[t]=false;
		
		for(int i=h[t];i!=-1;i=ne[i])
		{
			int j=e[i];//  t->j;
			if(dist[j]>dist[t]+w[i])
			{
				dist[j]=dist[t]+w[i];
				if(!st[j])
				{
					st[j]=true;
					q.push(j);
				}
			 } 
		}
	}
	
	for(int i=1;i<=n;i++)//将第cnt行星发动机据点---到---所有据点的最短路径存储下来
		ds[i][cnt]=dist[i];

}
int main()
{
	cin>>n>>m>>k;
	for(int i=1;i<=n;i++)
		cin>>type[i];
	memset(h,-1,sizeof(h));//因为图还是哪一个图---仅存储一次,从不同的点求单源最短路径
	while(m--)
	{
		int a,b,c;
		cin>>a>>b>>c;
		add(a,b,c),add(b,a,c);//无向图---将有向边存储两次---小心数组越界
	 }
	 
	 for(int i=1;i<=n;i++)
	 {
	 	if(type[i])
	 	{
	 		spfa(i);//从每个行星发动机开始进行spfa算法
	 		cnt++;
		 }
	  }
	  
	  
	  
	for(int i=1;i<=n;i++)//枚举所有的据点---找到---不是行星发动机的据点
	{
		int ans=0;
		sort(ds[i],ds[i]+cnt);//将第i个据点到所有行星发动机据点的距离排序
		
		for(int j=0;j<k && j<cnt;j++)
		{
			if(ds[i][j]!=INF)
				ans+=ds[i][j];
			else    //当排序后有一个行星发动机不可达,剩下的所有行星发动机都不可达
				break;
		 } 
		 cout<<ans<<endl;
	 } 
	
	return 0;
	
	 
 } 

AcWing 3255. 行车路线

AcWing 3255. 行车路线

AcWing 3255. 行车路线讲解

#include<bits/stdc++.h>
using namespace std;
const int N=510;//点数
const int M=2e5+10;//边数---注意无向边*2 
const int R=1e3+10;//纬度数---到达第i个点的路径上与i点直接相连的小道长度 

const int INF=0x3f3f3f3f;//为了方便输出最小花费 
typedef pair<int,int> PII;//[点的编号]---[到达i点的小道的长度] 
#define x first
#define y second
int dist[N][R];//dist[i][j]:      (从起点到达i点)且(与i点直接相连的小道长度为j)的最短距离 
bool st[N][R]; //spfa的状态数组
//邻接表存储图 
int idx;
int e[M];
int ne[M];
int h[N];
int w[M];
int type[M];

void add(int t,int a,int b,int c)
{
	type[idx]=t;
	w[idx]=c;
	e[idx]=b;
	ne[idx]=h[a];
	h[a]=idx++;
}

void spfa()// 求1号点到n号点的最短路距离
{
	memset(dist,INF,sizeof(dist));
	memset(st,0,sizeof(st));
	
	queue<PII> q;
	q.push({1,0});
	st[1][0]=true;
	dist[1][0]=0;
	
	while(q.size())
	{
		auto t=q.front();
		q.pop();
		st[t.x][t.y]=false;
		
		for(int i=h[t.x];i!=-1;i=ne[i])
		{
			int j=e[i];//t.x---->j的边(编号为i)
			if(type[i])//type为1时---小道 
			{
				if(dist[j][t.y+w[i]]>dist[t.x][t.y]-pow(t.y,2)+pow(t.y+w[i],2) && t.y+w[i]<=R)
				{
					dist[j][t.y+w[i]]=dist[t.x][t.y]-pow(t.y,2)+pow(t.y+w[i],2);
					if(!st[j][t.y+w[i]] && dist[j][t.y+w[i]]<=1e6)
					{
						st[j][t.y+w[i]]=true;
						q.push({j,t.y+w[i]});
					}
				}
			 }
			else//因为是大道---所以用dist[x][0] 
			{
			 	if(dist[j][0]>dist[t.x][t.y]+w[i])
				 {
				 	dist[j][0]=dist[t.x][t.y]+w[i];
				 	if(!st[j][0] && dist[j][0]<=1e6)
				 	{
				 		st[j][0]=true;
				 		q.push({j,0});
				 		
					 }
				  } 
			 } 
		}
	}
}
int main()
{
	int n,m;
	cin>>n>>m;
	memset(h,-1,sizeof(h)); 
	for(int i=1;i<=m;i++)
	{
		int t,a,b,c;
		cin>>t>>a>>b>>c;
		add(t,a,b,c);
		add(t,b,a,c);//无向图---两条边 
	}
	spfa();
	
	int ans=INF;
	for(int i=0;i<R;i++)
		ans=min(ans,dist[n][i]);
	cout<<ans<<endl;
	return 0;
}

AcWing 852. spfa判断负环

AcWing 852. spfa判断负环

#include<bits/stdc++.h>
using namespace std;
const int N=1E5+10;
const int INF=0x3f3f3f3f;
bool st[N];
int dist[N];
int cnt[N];//记录最短路径的长度 

int idx;
int e[N];
int ne[N];
int w[N];
int h[N]; 

int n,m;
void add(int a,int b,int c)
{
	ne[idx]=h[a];
	e[idx]=b;
	w[idx]=c;
	h[a]=idx++;
 } 
int spfa()
{
	memset(dist,INF,sizeof(dist));
	dist[1]=0;
	queue<int> q;
	for(int i=1;i<=n;i++)//---不同之处---将所有点都加入队列
    {
        st[i]=true;
        q.push(i);
    }
	
	while(q.size())
	{
		auto t=q.front();
		q.pop();
		st[t]=false;
		for(int i=h[t];i!=-1;i=ne[i])
		{
			int j=e[i];
			if(dist[j]>dist[t]+w[i])
			{
				dist[j]=dist[t]+w[i];
				cnt[j]=cnt[t]+1;//最短路径长度的变化
				
				if(cnt[j]>=n)
					return true;//有负环---负权回路 
				
	
				if(!st[j])
				{
					q.push(j);
					st[j]=true;
				 } 
			}
		}
		
	}
    return false;
}
int main()
{
	cin>>n>>m;
	memset(h,-1,sizeof(h));//注意---别忘了初始化---顶点表
	while(m--)
	{
		int a,b,c;
		cin>>a>>b>>c;
		add(a,b,c);//有向图  如果是无向图---+add(b,a,c) 
	}
	 if (spfa()) puts("Yes");
    else puts("No");

	return 0; 
}

Floyd

AcWing 854. Floyd求最短路

最短路径算法之floyd算法
AcWing 854. Floyd求最短路

#include<bits/stdc++.h>

using namespace std;

const int N = 210, INF = 1e9;

int n, m, Q;
int d[N][N];

void floyd()
{
    for (int k = 1; k <= n; k ++ )//从i到j---从i到k+从k到j 
        for (int i = 1; i <= n; i ++ )
            for (int j = 1; j <= n; j ++ )
                d[i][j] = min(d[i][j], d[i][k] + d[k][j]);
}

int main()
{
    scanf("%d%d%d", &n, &m, &Q);

    for (int i = 1; i <= n; i ++ )
        for (int j = 1; j <= n; j ++ )
            if (i == j) d[i][j] = 0;
            else d[i][j] = INF;

    while (m -- )
    {
        int a, b, c;
        scanf("%d%d%d", &a, &b, &c);
        d[a][b] = min(d[a][b], c);
    }

    floyd();

    while (Q -- )
    {
        int a, b;
        scanf("%d%d", &a, &b);

        int t = d[a][b];
        if (t > INF / 2) puts("impossible");
        else printf("%d\n", t);
    }

    return 0;
}


最小生成树

AcWing 858. Prim算法求最小生成树

AcWing 858. Prim算法求最小生成树

#include<bits/stdc++.h>
using namespace std;
const int N =500+10;
const int INF=0x3f3f3f3f;
int g[N][N];
bool st[N];
int dist[N];
int n,m;
int prim()
{
	memset(dist,INF,sizeof(dist));
	dist[1]=0;
	int res=0; //不同之处
	
	for(int i=0;i<n;i++)//选择n个点 ---与dijkstra(n-1次)不同,prim需要迭代n次
	{
		int t=-1;
		for(int j=1;j<=n;j++)
		{
			if(!st[j] && (t==-1||dist[j]<dist[t]))
				t=j; 
		 }
		 
		 
		 if (dist[t]==INF) return INF;//这里和dijkstra不同 
		 
		 st[t]=true;
		 
		 res+=dist[t];//这里和dijkstra不同 
		 
		 for(int j=1;j<=n;j++)
		 {
		 	dist[j]=min(dist[j],g[t][j]);//这里和dijkstra不同---更新方式不同 
		  } 
	 }
	 return res; 
}
int main()
{
	cin>>n>>m;
	memset(g,INF,sizeof(g));
	while(m--)
	{
		int x,y,c;
		cin>>x>>y>>c;
		g[x][y]=g[y][x]=min(g[x][y],c);//无向图 
	}
	 int res = prim();
    if (res == INF) 
        puts("impossible");
    else 
        printf("%d\n", res);
	return 0;
 } 

AcWing 859. Kruskal算法求最小生成树

AcWing 859. Kruskal算法求最小生成树

最小生成树之克鲁斯卡尔算法(Kruskal)

#include <cstring>
#include <iostream>
#include <algorithm>

using namespace std;

const int N = 100010, M = 200010, INF = 0x3f3f3f3f;

int n, m;
int p[N];

struct Edge
{
    int a, b, w;

    bool operator< (const Edge &W)const
    {
        return w < W.w;
    }
}edges[M];

int find(int x)
{
    if (p[x] != x) 
			p[x] = find(p[x]);
	return p[x];
}

int kruskal()
{
    sort(edges, edges + m);

    for (int i = 1; i <= n; i ++ ) p[i] = i;    // 初始化并查集

    int res = 0;//最小生成树的边权之和 
	int cnt = 0;//统计最小生成树的边的个数 
    for (int i = 0; i < m; i ++ )
    {
        int a = edges[i].a, b = edges[i].b, w = edges[i].w;

        a = find(a), b = find(b);
        if (a != b)
        {
            p[a] = b;
            res += w;
            cnt ++ ;
        }
    }

    if (cnt < n - 1) return INF;//如果最后边数不够时 
    return res;
}

int main()
{
    scanf("%d%d", &n, &m);

    for (int i = 0; i < m; i ++ )
    {
        int a, b, w;
        scanf("%d%d%d", &a, &b, &w);
        edges[i] = {a, b, w};
    }

    int t = kruskal();

    if (t == INF) puts("impossible");
    else printf("%d\n", t);

    return 0;
}


201412-4-csp-最优灌溉—非常裸的最小生成树问题

AcWing 3210. 最优灌溉

#include<bits/stdc++.h>
using namespace std;
const int N=1e3+10;
const int M=1e5+10;

struct Edge{//---存储边的信息
	int a;
	int b;
	int c;
	bool operator<(const Edge& t)const
	{
		return c<t.c;
	}
}edges[M];
int p[N];//并查集数组
int find(int x)
{
	if(x!=p[x])
		p[x]=find(p[x]);
	return p[x];
 } 
int n,m;
int kruskral()
{
	for(int i=1;i<=n;i++)//初始化
		p[i]=i;
	sort(edges,edges+m);//对边按权重排序---选择小边
	int ans=0;
	int cnt=0;//统计边的数量
	for(int i=0;i<m;i++)
	{
		int a=edges[i].a;
		int b=edges[i].b;
		int c=edges[i].c;
		
		if(find(a)!=find(b))
		{
			ans+=c;
			p[find(a)]=find(b);
			cnt++;
		}
	}
	if(cnt<n-1)//如果边数不够时---报错
	    return -1;
	return ans;
	
}
int main()
{
	cin>>n>>m;
	for(int i=0;i<m;i++)
	{
		int a,b,c;
		cin>>a>>b>>c;
		edges[i]={a,b,c};
	}
	auto t=kruskral();
	cout<<t<<endl;
	return 0;
}

201703-4-csp-地铁修建

AcWing 3245. 地铁修建

通过kruskal进行求解,加入的额外的更小权值的边并不影响结果
所以从小到大进行加入边权知道1与n节点在同一个连通块中,可达,此时的边权即是结果
#include<bits/stdc++.h>
using namespace std;
const int N=1e5+10;
const int M=2e5+10;

struct Edge{
	int a,b,c;
	bool operator<(const Edge& t) const
	{
		return c<t.c;
	}
}edges[M];
int p[N];
int find(int x)
{
	if(x!=p[x])//这里最好不要加if else
		p[x]=find(p[x]);
	return p[x];//一直找到根节点 
}
int n,m;
int maxw;
void kruskral()
{
	for(int i=1;i<=n;i++) p[i]=i;
	
	sort(edges,edges+m);


	for(int i=0;i<m;i++)
	{
		int a=edges[i].a;
		int b=edges[i].b;
		int c=edges[i].c;
		if(find(a)!=find(b))
		{
			p[find(a)]=find(b);
			if(find(1)==find(n))
			{
				maxw=c;//保存最大的边权 
				return ;
			}
			
		}
	 }

}
int main()
{
	cin>>n>>m;
	for(int i=0;i<m;i++)
	{
		int a,b,c;
		cin>>a>>b>>c;
		edges[i]={a,b,c};
	}
	
	kruskral();
	cout<<maxw<<endl;
	return 0;
}

201812-4-csp-数据中心

Wing 3270. 数据中心
Wing 3270. 数据中心(最小生成树之最大边权最小)

#include<bits/stdc++.h>
using namespace std;
const int N=1e5+10;
const int M=2e5+10;

struct Edge{
	int a,b,c;
	bool operator<(const Edge& t) const
	{
		return c<t.c;
	}
}edges[M];
int p[N];
int find(int x)
{
	if(x!=p[x])//这里最好不要加if else
		p[x]=find(p[x]);
	return p[x];//一直找到根节点 
}
int n,m;
int maxw;
void kruskral()
{
	for(int i=1;i<=n;i++) p[i]=i;
	
	sort(edges,edges+m);


	for(int i=0;i<m;i++)
	{
		int a=edges[i].a;
		int b=edges[i].b;
		int c=edges[i].c;
		if(find(a)!=find(b))
		{
			p[find(a)]=find(b);
				maxw=c;//保存最大的边权
		}
	 }
	return ;

}
int main()
{
	cin>>n;
	cin>>m;
	int root;
	cin>>root;
	for(int i=0;i<m;i++)
	{
		int a,b,c;
		cin>>a>>b>>c;
		edges[i]={a,b,c};
	}
	
	kruskral();
	cout<<maxw<<endl;
	return 0;
}

欧拉路径

201512-4-csp-送货

AcWing 3225. 送货

直接将邻接表放到set里做,自动判重、排序,保证输出的是最小字典序
判断欧拉路径的初始条件后直接dfs就行了

#include<bits/stdc++.h> 
using namespace std;
const int N = 10010, M = 100010;

int n, m;
set<int> g[N];//邻接表存储---集合实现 
int p[N];//并查集数组 
int ans[M], top;//保存了反向答案 

int find(int x)
{
    if (p[x] != x) p[x] = find(p[x]);
    return p[x];
}

void dfs(int u)
{
    while (g[u].size())//当还有临边时
    {
        int t = *g[u].begin();//边的终点
        g[u].erase(t), g[t].erase(u);//删除这条边
        dfs(t);
    }
    ans[ ++ top] = u;//记录答案
}

int main()
{
    scanf("%d%d", &n, &m);
    for (int i = 1; i <= n; i ++ ) p[i] = i;
    while (m -- )
    {
        int a, b;
        scanf("%d%d", &a, &b);
        g[a].insert(b), g[b].insert(a);
        p[find(a)] = find(b);
    }

    int s = 0;//欧拉路径---奇数条边点个数=0/2(其中一个是起点,一个是终点) 
    for (int i = 1; i <= n; i ++ )
        if (find(i) != find(1))
        {
            puts("-1");
            return 0;
        }
        else if (g[i].size() % 2) s ++ ;

    if (s != 0 && s != 2 || s == 2 && g[1].size() % 2 == 0)
    {
        puts("-1");
        return 0;
    }
    dfs(1);

    for (int i = top; i; i -- )
        printf("%d ", ans[i]);

    return 0;
}


染色法判定二分图

匈牙利算法

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

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

相关文章

NSSCTF做题(6)

[HCTF 2018]Warmup 查看源代码得到 开始代码审计 <?php highlight_file(__FILE__); class emmm { public static function checkFile(&$page) { $whitelist ["source">"source.php","hint"…

Java-API简析_java.util.Objects类(基于 Latest JDK)(浅析源码)

【版权声明】未经博主同意&#xff0c;谢绝转载&#xff01;&#xff08;请尊重原创&#xff0c;博主保留追究权&#xff09; https://blog.csdn.net/m0_69908381/article/details/133463511 出自【进步*于辰的博客】 因为我发现目前&#xff0c;我对Java-API的学习意识比较薄弱…

人工智能的学习算法

1956年&#xff0c;几个计算机科学家相聚在达特茅斯会议&#xff0c;提出了 “人工智能” 的概念&#xff0c;梦想着用当时刚刚出现的计算机来构造复杂的、拥有与人类智慧同样本质特性的机器。其后&#xff0c;人工智能就一直萦绕于人们的脑海之中&#xff0c;并在科研实验室中…

K折交叉验证——cross_val_score函数使用说明

在机器学习中&#xff0c;许多算法中多个超参数&#xff0c;超参数的取值不同会导致结果差异很大&#xff0c;如何确定最优的超参数&#xff1f;此时就需要进行交叉验证的方法&#xff0c;sklearn给我们提供了相应的cross_val_score函数&#xff0c;可对数据集进行交叉验证划分…

小程序是一种伪需求技术吗?

点击下方“JavaEdge”&#xff0c;选择“设为星标” 第一时间关注技术干货&#xff01; 免责声明~ 任何文章不要过度深思&#xff01; 万事万物都经不起审视&#xff0c;因为世上没有同样的成长环境&#xff0c;也没有同样的认知水平&#xff0c;更「没有适用于所有人的解决方案…

[NOIP2012 提高组] 开车旅行

[NOIP2012 提高组] 开车旅行 题目描述 小 A \text{A} A 和小 B \text{B} B 决定利用假期外出旅行&#xff0c;他们将想去的城市从 $1 $ 到 n n n 编号&#xff0c;且编号较小的城市在编号较大的城市的西边&#xff0c;已知各个城市的海拔高度互不相同&#xff0c;记城市 …

零基础一站式精通安卓逆向2023最新版(第一天):Android Studio的安装与配置

目录 一、Android Studio 开发环境的下载二、Android Studio 的安装与配置2.1 安装2.2 Android SDK 的管理 三、创建 Android 应用程序补充&#xff1a;安装完 Android Studio 后 SDK 目录下没有 tools 目录 一、Android Studio 开发环境的下载 通常情况下&#xff0c;为了提高…

对pyside6中的textedit进行自定义,实现按回车可以触发事件。

我的实现方法是&#xff0c;先用qt designer写好界面&#xff0c;如下图&#xff1a; 接着将其生成的ui文件编译成为py文件。 找到里面这几行代码&#xff1a; self.textEdit QTextEdit(self.centralwidget)self.textEdit.setObjectName(u"textEdit")self.textEdit…

Vue城市选择器示例(省市区三级)

Vue城市选择器&#xff08;省市区&#xff09; 读者可以参考下面的省市区三级联动代码思路&#xff0c;切记要仔细研究透彻&#xff0c;学习交流才是我们的本意&#xff0c;而非一成不变。切记切记&#xff01; 最近又重读苏子的词&#xff0c;颇为感慨&#xff0c;愿与诸君共…

2022年中国征信行业覆盖人群、参与者数量及征信业务查询量统计[图]

征信是指依法收集、整理、保存、加工自然人、法人及其他组织的信用信息&#xff0c;并对外提供信用报告、信用评估、信用信息咨询等服务&#xff0c;帮助客户判断、控制信用风险&#xff0c;进行信用管理的活动。 征信业主要范畴 资料来源&#xff1a;共研产业咨询&#xff08…

B. Comparison String

题目&#xff1a; 样例&#xff1a; 输入 4 4 <<>> 4 >><< 5 >>>>> 7 <><><><输出 3 3 6 2 思路&#xff1a; 由题意&#xff0c;条件是 又因为要使用尽可能少的数字&#xff0c;这是一道贪心题&#xff0c;所以…

初识多线程

一、多任务 现实中太多这样同时做多件事的例子了&#xff0c;例如一边吃饭一遍刷视频&#xff0c;看起来是多个任务都在做&#xff0c;其实本质上我们的大脑在同一时间依旧只做了一件事情。 二、普通方法调用和多线程 普通方法调用只有主线程一条执行路径 多线程多条执行路径…

uni-app_消息推送_华为厂商_unipush离线消息推送

文章目录 一、创建项目二、生成签名证书三、开通 unipush 推送服务四、客户端集成四、制作自定义调试基座五、开发者中心后台Web页面推送&#xff08;仅支持在线推送&#xff09;六、离线消息推送1、创建华为开发者账号2、开通推送服务3、创建项目4、添加应用5、添加SHA256证书…

【Linux】详解线程第三篇——线程同步和生产消费者模型

线程同步和生消模型 前言正式开始再次用黄牛抢票来讲解线程同步的思想通过条件变量来实现线程同步条件变量接口介绍初始化和销毁pthread_cond_waitsignal和broadcast 生产消费者模型三种关系用基本工程师思维再次理解基于生产消费者模型的阻塞队列版本一版本二多生多消 利用RAI…

2022年全球一次能源消费量:石油消耗量持续增加达190.69百亿亿焦耳,亚太地区消费量居首位[图]

一次性能源是指从自然界取得未经改变或转变而直接利用的能源。如原煤、原油、天然气、水能、风能、太阳能、海洋能、潮汐能、地热能、天然铀矿等。一次性能源又分为可再生能源和不可再生能源&#xff0c;前者指能够重复产生的天然能源&#xff0c;包括太阳能、风能、潮汐能、地…

响应式设计的实现方式

一. 什么是响应式 响应式网站设计是一种网络页面设计布局。页面的设计与开发应当根据用户行为以及设备环境&#xff08;系统平台&#xff0c;屏幕尺寸&#xff0c;屏幕定向等&#xff09;进行相应的响应和调整。 响应式网站常见特点&#xff1a; 1. 同时适配PC平板手机。 2…

排序篇(五)----非比较排序

排序篇(五)----非比较排序 基本思想&#xff1a; ​ 计数排序又称为鸽巢原理&#xff0c;是对哈希直接定址法的变形应用。 ​ 统计每个元素出现的次数&#xff0c;然后根据元素的大小顺序将它们放入正确的位置。 ​ 计数排序是一种小众的排序,它适合于数据密集的场景,按最大…

flink选择slot

flink选择slot 在这个类里修改 package org.apache.flink.runtime.resourcemanager.slotmanager.SlotManagerImpl; findMatchingSlot(resourceProfile)&#xff1a;找到满足要求的slot&#xff08;负责从哪个taskmanager中获取slot&#xff09;对应上图第8&#xff0c;9&…

百元开放式耳机推荐哪款、性价比最好的开放式耳机推荐

随着蓝牙耳机产业的高速发展&#xff0c;目前最热门的蓝牙耳机莫过于开放式的&#xff0c;跟传统的蓝牙耳机相比&#xff0c;开放式的耳机拥有久戴不累、安全舒适等优势&#xff0c;所谓的“开放式耳机”&#xff0c;就是指不用塞入耳朵内&#xff0c;也能听音乐的耳机&#xf…

noip2011铺地毯

[NOIP2011 提高组] 铺地毯 题目描述 为了准备一个独特的颁奖典礼&#xff0c;组织者在会场的一片矩形区域&#xff08;可看做是平面直角坐标系的第一象限&#xff09;铺上一些矩形地毯。一共有 n n n 张地毯&#xff0c;编号从 1 1 1 到 n n n。现在将这些地毯按照编号从小…