第一讲 基础算法

本文深入探讨了快速排序、归并排序、二分查找等经典算法,包括它们的基本思想、实现细节和应用实例。通过这些算法,我们可以高效地对数据进行排序和查找,提高程序的运行效率。

AcWing 785. 快速排序

快速排序本质上是分治法

  1. 分解成子问题
  2. 递归处理子问题
  3. 合并子问题
    在这里插入图片描述
#include<iostream>
using namespace std;

const int N = 1e6 + 10;

int n;
int q[N];

void quick_sort(int q[], int l, int r)
{
	//递归的边界情况,到这个时候子问题不能继续分解
    if(l >= r) return;
	
	//分解为子问题
    int x = q[(l + r) >> 1], i = l - 1, j = r + 1;
    while(i < j)
    {
        do i ++ ; while(q[i] < x);
        do j -- ; while(q[j] > x);
        if(i < j) swap(q[i], q[j]); 
    }
	
	//递归处理子问题
    quick_sort(q, l, j);
    quick_sort(q, j + 1, r);
}

int main()
{
    scanf("%d", &n);
    for(int i = 0; i < n; i ++ ) scanf("%d", &q[i]);

    quick_sort(q, 0, n - 1);

    for(int i = 0; i < n; i ++ ) printf("%d ", q[i]);

    return 0;
}

AcWing 786. 第k个数

减治法,分治法的一种形式
每次只需要判断k在左区间还是右区间,一直递归查找k所在区间,另外的一个区间则舍去
最后只剩一个数时,只会有数组[k]一个数,返回数组[k]的值就是答案

#include<iostream>
using namespace std;

const int N = 1e5 + 10;

int n, k;
int q[N];

int quick_sort(int l, int r, int k)
{
    if(l == r) return q[l];

    int x = q[l], i = l - 1, j = r + 1;
    while(i < j)
    {
        while(q[ ++ i] < x);
        while(q[ -- j] > x);
        if(i < j) swap(q[i], q[j]);
    }
	
	//判断k所在的区间,sl表示左区间的长度
    int sl = j - l + 1;
    
	//k在左区间
    if(k <= sl) return quick_sort(l, j, k);		
    //k在右区间,k - sl表示 k 在右区间的位置
    else return quick_sort(j + 1, r, k - sl);	
}

int main()
{
    cin >> n >> k;
    for(int i = 0; i < n; i ++ ) cin >> q[i];
    cout << quick_sort(0, n - 1, k) << endl;

    return 0;
}

AcWing 787. 归并排序

分治法的典型

  1. 分解成子问题
  2. 递归处理子问题
  3. 合并子问题
    在这里插入图片描述
#include<iostream>
using namespace std;

const int N = 1e6 + 10;
int n;
int q[N], tmp[N];		//tmp作为辅助数组

void merge_sort(int q[], int l, int r)
{
	//子问题不能再接着分解了
	if(l >= r) return;

	//分解子问题
	int mid = l + r >> 1;
	merge_sort(q, l, mid);
	merge_sort(q, mid + 1, r);
	
	//合并子问题
	int k = 0, i = l, j = mid + 1;
	
    while(i <= mid && j <= r)
	{
		if(q[i] <= q[j]) tmp[k ++ ] = q[i ++ ];
		else tmp[k ++ ] = q[j ++ ];
	}

	while(i <= mid) tmp[k ++ ] = q[i ++ ];
	while(j <= r) tmp[k ++ ] = q[j ++ ];

	for(i = l, j = 0; i <= r; i ++, j ++ ) q[i] = tmp[j];
}

int main()
{
	scanf("%d", &n);
	for(int i = 0; i < n; i ++ ) scanf("%d", &q[i]);

	merge_sort(q, 0, n - 1);
	for(int i = 0; i < n; i ++ ) printf("%d ", q[i]);

	return 0;
}

AcWing 788. 逆序对的数量

#include<iostream>
using namespace std;

typedef long long LL;

const int N = 1e6 + 10;

int n;
int q[N], tmp[N];

LL merge_sort(int l, int r)
{
    if(l >= r) return 0;

    int mid = l + r >> 1;
    LL res = merge_sort(l, mid) + merge_sort(mid + 1, r);

    int k = 0, i = l, j = mid + 1;
    while(i <= mid && j <= r)
        if(q[i] <= q[j]) tmp[k ++ ] = q[i ++ ];
        else
        {
            tmp[k ++ ] = q[j ++ ];
            res += mid - i + 1;
        }

    while(i <= mid) tmp[k ++ ] = q[i ++ ];
    while(j <= r) tmp[k ++ ] = q[j ++ ];

    for(int i = l, j = 0; i <= r; i ++, j ++ ) q[i] = tmp[j];

    return res;
}

int main()
{
    cin >> n;

    for(int i = 0; i < n; i ++ ) cin >> q[i];

    cout << merge_sort(0, n - 1) << endl;

    return 0;
}

AcWing 789. 数的范围

一道典型的二分法处理的题目

当想找不满足性质的边界值(蓝色区域的右边界值)

  1. 找中间值 mid = (l+r+1)/2
  2. if(check(mid))等于true或者是false
  3. check(m)是检查m是在不满足性质的区间(检查是不是在蓝色区间)
  4. 更新l或者r

此时check函数检查mid是否在蓝色区间内
if check(mid)== true :说明mid在蓝色区间内,那么应该在区间[mid, r]寻找蓝色区间的右边界值,l = mid
if check(mid)== false:说明mid在红色区间内,那么应该在区间[l, mid - 1]寻找蓝色区间的右边界值,r = mid - 1

当想找满定在质的五界值(红色区域的差边界值)

  1. 找中间值mid = (l+r)/2
  2. if(check(mid))等于true或者是false
  3. check(m)是检查m是在满足性质的区间(检查是不是在红色区间)
  4. 更新l或者r

此时check函数检查mid是否在红色区间内
if check(mid)== false:说明mid在蓝色区间内,那么,应该在区间[mid + 1, r]中寻找红色区间的左边界值,即l= mid + 1
if check(mid)== true:说明mid在红色区间内,那么,应该在区间[l, mid]中寻找红色区间的左边界值,即r = mid

在这里插入图片描述

#include<iostream>
using namespace std;

const int N = 1e5 + 10;

int n, m;
int q[N];

int main()
{
    scanf("%d%d", &n, &m);
    for(int i = 0; i < n; i ++ ) scanf("%d", &q[i]);

    while(m -- )
    {
        int x;
        scanf("%d", &x);

        int l = 0, r = n - 1;
        
        //检查左边界
        while(l < r)
        {
            int mid = l + r >> 1;
            if(q[mid] >= x) r = mid;
            else l = mid + 1;
        }
        if(q[l] != x) cout << "-1 -1" << endl;
        else
        {
            cout << l << " ";

            int l = 0, r = n - 1;
            //检查右边界
            while(l < r)
            {
                int mid = l + r + 1 >> 1;
                if(q[mid] <= x) l = mid;
                else r = mid - 1;
            }
            cout << r << endl;
        }
    }
    return 0;
}

AcWing 790. 数的三次方根

二分法处理问题

#include<iostream>
using namespace std;

int main()
{
	double x;
	cin >> x;

	double l = -10000, r = 10000;
	while(r - l > 1e-8)
	{
		double mid = (l + r) / 2;
		if(mid * mid * mid >= x) r = mid;
		else l = mid;
	}
	printf("%lf", l);
	return 0;
}

AcWing 791. 高精度加法

模拟一下列竖式进行加法运算,使用一个变量carry,进行进位的操作,
在这里插入图片描述

#include<iostream>
#include<vector>
using namespace std;

vector<int> add(vector<int> &A, vector<int> &B)
{
	vector<int> C;

	int t;
	for(int i = 0; i < A.size() || i < B.size(); i ++ )
	{
		if(i < A.size()) t += A[i];
		if(i < B.size()) t += B[i];
		C.push_back(t % 10);
		t /= 10;	
	}
	if(t) C.push_back(1);
	return C;
}

int main()
{
	string a, b;
	vector<int> A, B;

	cin >> a >> b;
	for(int i = a.size() - 1; i >= 0; i -- ) A.push_back(a[i] - '0');
	for(int i = b.size() - 1; i >= 0; i -- ) B.push_back(b[i] - '0');

	auto C = add(A, B);
	for(int i = C.size() - 1; i >= 0; i -- ) printf("%d", C[i]);

	return 0;
}

AcWing 792. 高精度减法

#include<iostream>
#include<vector>
using namespace std;

bool cmp(vector<int> &A, vector<int> &B)
{
	if(A.size() != B.size()) return A.size() > B.size();

	for(int i = A.size() - 1; i >= 0; i -- )
		if(A[i] != B[i])
			return A[i] > B[i];

	return true;
}

vector<int> sub(vector<int> &A, vector<int> &B)
{
	vector<int> C;
	for(int i = 0, t = 0; i < A.size(); i ++ )
	{
		t = A[i] - t;
		if(i < B.size()) t -= B[i];
		C.push_back((t + 10) % 10);
		if(t < 0) t = 1;
		else t = 0;
	}
	while(C.size() > 1 && C.back() == 0) C.pop_back();
	return C; 
}

int main()
{
	string a, b;
	vector<int> A, B;
	
	cin >> a >> b;
	
	for(int i = a.size() - 1; i >= 0; i -- ) A.push_back(a[i] - '0');
	for(int i = b.size() - 1; i >= 0; i -- ) B.push_back(b[i] - '0');

	if(cmp(A, B))
	{
		auto C = sub(A, B);
		for(int i = C.size() - 1; i >= 0; i -- ) printf("%d", C[i]);
	}
	else
	{
		auto C = sub(B, A);
		printf("-");
		for(int i = C.size() - 1; i >= 0; i -- ) printf("%d", C[i]);
	}
	
	return 0;
}

AcWing 793. 高精度乘法

#include<iostream>
#include<vector>

using namespace std;

vector<int> mul(vector<int> &A, vector<int> &B)
{
	vector<int> C(A.size() + B.size());

	for(int i = 0; i < A.size(); i ++ )
		for(int j = 0; j < B.size(); j ++ )
			C[i + j] += A[i] * B[j];
	
	for(int i = 0, t = 0; i < C.size() || t; i ++ )
	{
		t += C[i];
		if(i >= C.size()) C.push_back(t % 10);
		else C[i] = t % 10;

		t /= 10;
	}
	while(C.size() > 1 && C.back() == 0) C.pop_back();

	return C;
}

int main()
{
	string a, b;
	cin >> a >> b;

	vector<int> A, B;

	for(int i = a.size() - 1; i >= 0; i -- ) A.push_back(a[i] - '0');
	for(int i = b.size() - 1; i >= 0; i -- ) B.push_back(b[i] - '0');

	auto C = mul(A, B);

	for(int i = C.size() - 1; i >= 0; i -- ) cout  << C[i];
	puts("");

	return 0;
}

AcWing 794. 高精度除法

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

vector<int> div(vector<int> &A, int b, int &r)
{
	vector<int> C;
	r = 0;
	for(int i = A.size() - 1; i >= 0; i -- )
	{
		r = r * 10 + A[i];
		C.push_back(r / b);
		r = r % b;
	}
	reverse(C.begin(), C.end());
	while(C.size() > 1 && C.back() == 0) C.pop_back();
	return C;
}

int main()
{
	string a;
	int b;

	cin >> a >> b;
	vector<int> A;
	for(int i = a.size() - 1; i >= 0; i -- ) A.push_back(a[i] - '0');

	int r;
	auto C = div(A, b, r);

	for(int i = C.size() - 1; i >= 0; i -- ) printf("%d", C[i]);
	cout << endl << r << endl;

	return 0;
}

AcWing 795. 前缀和

原数组: a[1], a[2], a[3], a[4], a[5], …, a[n]
前缀和:S[i] = a[1] + a[2] + a[3] + … + a[i]
作用:可以快速求出数组中某一段连续区间的和
用法:求数组a[l],a[l + 1], a[l + 2], ..., a[r]的和,即S[r] - S[l - 1]

#include<iostream>
using namespace std;

const int N = 1e5 + 10;

int n, m;
int a[N], s[N];

int main()
{
	scanf("%d%d", &n, &m);
	for(int i = 1; i <= n; i ++ ) scanf("%d", &a[i]);

	for(int i = 1; i <= n; i ++ ) s[i] = s[i - 1] + a[i];

	while(m -- )
	{
		int l, r;
		scanf("%d%d", &l, &r);
		printf("%d\n", s[r] - s[l - 1]);
	}
    
	return 0;
}

AcWing 796. 子矩阵的和

S[i, j]表示从a[1, 1]a[i, j] 这个矩阵的元素的和
S [ i , j ] = S [ i , j − 1 ] + S [ i − 1 , j ] − S [ i − 1 , j − 1 ] + a [ i , j ] S[i,j]=S[i,j−1]+S[i−1,j]−S[i−1,j−1]+a[i,j] S[i,j]=S[i,j1]+S[i1,j]S[i1,j1]+a[i,j]

(x1,y1)(x2, y2)这个子矩阵的计算公式
S [ x 2 , y 2 ] − S [ x 1 − 1 , y 2 ] − S [ x 2 , y 1 − 1 ] + S [ x 1 − 1 , y 1 − 1 ] S[x2,y2]−S[x1−1,y2]−S[x2,y1−1]+S[x1−1,y1−1] S[x2,y2]S[x11,y2]S[x2,y11]+S[x11,y11]
在这里插入图片描述
在这里插入图片描述

#include<iostream>
using namespace std;

const int N = 1010;

int m, n, q;
int a[N][N], s[N][N];

int main()
{
	scanf("%d%d%d", &n, &m, &q);
	for(int i = 1; i <= n; i ++ )
		for(int j = 1; j <= m; j ++ )
			scanf("%d", &a[i][j]);

	for(int i = 1; i <= n; i ++ )
		for(int j = 1; j <= m; j ++ )
			s[i][j] = s[i - 1][j] + s[i][j - 1] - s[i - 1][j - 1] + a[i][j];

	while(q -- )
	{
		int x1, y1, x2, y2;
		scanf("%d%d%d%d", &x1, &y1, &x2, &y2);
		printf("%d\n", s[x2][y2] - s[x1 - 1][y2] - s[x2][y1 - 1] + s[x1 - 1][y1 - 1]);
	}

	return 0;
}

AcWing 797. 差分

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

a [ 0 ] = 0 ; a[0 ]= 0; a[0]=0;
b [ 1 ] = a [ 1 ] − a [ 0 ] ; b[1] = a[1] - a[0]; b[1]=a[1]a[0];
b [ 2 ] = a [ 2 ] − a [ 1 ] ; b[2] = a[2] - a[1]; b[2]=a[2]a[1];
b [ 3 ] = a [ 3 ] − a [ 2 ] ; b[3] =a [3] - a[2]; b[3]=a[3]a[2];

b [ n ] = a [ n ] − a [ n − 1 ] ; b[n] = a[n] - a[n-1]; b[n]=a[n]a[n1];
由公式可以得知,数组a是数组b的前缀和数组,反过来,数组b是数组a的差分数组

应用场景:
给定区间[l ,r],让我们把a数组中的[l, r]区间中的每一个数都加上c,即
a [ l ] + c , a [ l + 1 ] + c , a [ l + 2 ] + c , . . . , a [ r ] + c ; a[l] + c , a[l+1] + c , a[l+2] + c ,... , a[r] + c; a[l]+c,a[l+1]+c,a[l+2]+c,...,a[r]+c;
如果单纯使用for循环去做加法,那么m次加上c的时间复杂度则为 O ( n ∗ m ) O(n * m) O(nm)

这个时候要将时间复杂度降低就可以使用差分数组了
a数组是b数组的前缀和数组,比如对b数组的b[i]的修改,会影响到a数组中从a[i]及往后的每一个数。
首先让差分b数组中的 b [ l ] + c b[l] + c b[l]+c ,a数组变成 a [ l ] + c , a [ l + 1 ] + c , . . . , a [ n ] + c a[l] + c ,a[l+1] + c,..., a[n] + c a[l]+c,a[l+1]+c,...,a[n]+c;

然后为了避免对数组中坐标r之后的值造成影响, b [ r + 1 ] − c b[r+1] - c b[r+1]c, a数组变成 a [ r + 1 ] − c , a [ r + 2 ] − c , . . . , a [ n ] − c a[r+1] - c,a[r+2] - c,...,a[n] - c a[r+1]c,a[r+2]c,...,a[n]c;
在这里插入图片描述
因此一维差分对a数组的[l, r]区间同时加上一个数c,只需要对其差分数组b,做出 b [ l ] + = c b[l] += c b[l]+=c, b [ r + 1 ] − = c b[r + 1] -= c b[r+1]=c 的操作,时间复杂度为 O ( 1 ) O(1) O(1),m次操作的总体时间复杂度则从 O ( n ∗ m ) O(n*m) O(nm)降低到 O ( m ) O(m) O(m)

#include<iostream>
using namespace std;

const int N = 1e5 + 10;

int n, m;
int a[N], b[N];

void insert(int l, int r, int c)
{
	b[l] += c;
	b[r + 1] -= c;
}

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

	for(int i = 1; i <= n; i ++ ) scanf("%d", &a[i]);
	for(int i = 1; i <= n; i ++ ) insert(i, i, a[i]);

	while(m -- )
	{
		int l, r, c;
		scanf("%d%d%d", &l, &r, &c);
		insert(l, r, c);
	}

	for(int i = 1; i <= n; i ++ ) b[i] += b[i - 1];
	for(int i = 1; i <= n; i ++ ) printf("%d ", b[i]);
	return 0;
}

AcWing 798. 差分矩阵

与一维前缀和延伸至二维前缀和的过程类似
一维差分延伸至二维差分
核心操作:
b [ x 1 ] [ y 1 ] + = c ; b[x1][y1] += c; b[x1][y1]+=c;
b [ x 2 + 1 ] [ y 1 ] − = c ; b[x2 + 1][y1] -= c; b[x2+1][y1]=c;
b [ x 1 ] [ y 2 + 1 ] − = c ; b[x1][y2 + 1] -= c; b[x1][y2+1]=c;
b [ x 2 + 1 ] [ y 2 + 1 ] + = c ; b[x2 + 1][y2 + 1] += c; b[x2+1][y2+1]+=c;
在这里插入图片描述

#include<iostream>
using namespace std;

const int N = 1010;

int n, m, q;
int a[N][N], b[N][N];

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

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

	for(int i = 1; i <= n; i ++ )
		for(int j = 1; j <= m; j ++ )
			scanf("%d", &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];
	
	for(int i = 1; i <= n; i ++ )
	{
		for(int j = 1; j <= m; j ++ )
			printf("%d ", b[i][j]);
		puts("");
	}
	return 0;
}

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

双指针算法。
使用一个数组s维护当前找到的区间 [i, j]中数字的个数。
如果出现重复数字,那么指针 i 向后移动一位,并且哈希表中这个数字的出现次数减一,直到没有重复字符为止,使用了一个while循环实现
如果没有重复数组,那么指针 j 向后移动一位,同时s中对应的数字出现次数加一
在这个过程中使用res记录子序列长度的最大值。
在这里插入图片描述

#include<iostream>
using namespace std;

const int N = 1e6 + 10;

int a[N], s[N];
int n;

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

	int res = 0;
	for(int i = 0, j = 0; i < n; i ++ )
	{
		s[a[i]] ++ ;
		while(s[a[i]] > 1)
		{
			s[a[j]] -- ;
			j ++ ;
		}
		res = max(res, i - j + 1);
	}
	cout << res << endl;

	return 0;
}

AcWing 800. 数组元素的目标和

i从 0开始 从前往后遍历
jm - 1开始 从后向前遍历,j指针不会回退

#include<iostream>
using namespace std;

const int N = 1e5 + 10;

int n, m, x;
int a[N], b[N];

int main()
{
	scanf("%d%d%d", &n, &m, &x);
	
	for(int i = 0; i < n; i ++ ) scanf("%d", &a[i]);
	for(int i = 0; i < m; i ++ ) scanf("%d", &b[i]);

	for(int i = 0, j = m - 1; i < n; i ++ )
	{
		while(j > 0 && a[i] + b[j] > x) j -- ;
		if(j >= 0 && a[i] + b[j] == x)
			cout << i << " " << j << endl;
	}
	return 0;
}

AcWing 2816. 判断子序列

  1. j指针用来扫描整个b数组,i指针用来扫描a数组。若发现a[i]==b[j],则让i指针后移一位。
  2. 整个过程中,j指针不断后移,而i指针只有当匹配成功时才后移一位,若最后若i==n,则说明匹配成功。在这里插入图片描述
#include<iostream>
using namespace std;

const int N = 1e5 + 10;

int n, m;
int a[N], b[N];

int main()
{
	cin >> n >> m;

	for(int i = 0; i < n; i ++ ) cin >> a[i];
	for(int i = 0; i < m; i ++ ) cin >> b[i];

	int j = 0;
	for(int i = 0; i < m; i ++ )
		if(b[i] == a[j])
			j ++ ;
	
	if(j == n) cout << "Yes" << endl;
	else cout << "No" << endl;

	return 0;
}

AcWing 801. 二进制中1的个数

按位与运算
在这里插入图片描述

#include<iostream>
using namespace std;

int lowBit(int x)
{
	return x & -x;
}

int main()
{
	int n;
	cin >> n;
	while(n -- )
	{
		int x;
		cin >> x;
		
		int res = 0;
		while(x) 
		{
			x -= lowBit(x);
			res ++ ;
		}

		cout << res << " ";
	}
	return 0;
}

AcWing 802. 区间和

题解

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

typedef pair<int, int> PII;
const int N = 3e5 + 10;

int n, m;
int a[N], s[N];

vector<int> all;
vector<PII> add, query;

int find(int x)
{
    int l = 0, r = all.size() - 1;
    while(l < r)
    {
        int mid = l + r >> 1;
        if(all[mid] >= x) r = mid;
        else l = mid + 1;
    }
    return r + 1;
}

int main()
{
    cin >> n >> m;

    for(int i = 0; i < n; i ++ )
    {
        int x, c;
        cin >> x >> c;
        add.push_back({x, c});
        all.push_back(x);
    }

    for(int i = 0; i < m; i ++ )
    {
        int l, r;
        cin >> l >> r;
        query.push_back({l, r});

        all.push_back(l);
        all.push_back(r);
    }

    sort(all.begin(), all.end());
    all.erase(unique(all.begin(), all.end()), all.end());

    for(auto item : add)
    {
        int x = find(item.first);
        a[x] += item.second;
    }

    for(int i = 1; i <= all.size(); i ++ ) s[i] = s[i - 1] + a[i];

    for(auto item : query)
    {
        int l = find(item.first), r = find(item.second);
        cout << s[r] - s[l - 1] << endl;
    }

    return 0;
}

AcWing 803. 区间合并

思想方法类似于贪心

  1. 先给每个区间按左端点排序
  2. 遍历每个区间
    如果第二个区间起点小于第一个区间终点,那么就是有交集,将第一个区间的终点更新为第二个区间的终点
    如果没有交集,那么将前面更新完毕的区间存储起来
    在这里插入图片描述
#include<iostream>
#include<algorithm>
#include<vector>
using namespace std;

typedef pair<int, int> PII;

int n;
vector<PII> seg;

void merge(vector<PII> &Seg)
{
    vector<PII> res;
    int st = -2e9, ed = -2e9;
    sort(Seg.begin(), Seg.end());
    for(auto seg : Seg)
    {
        if(ed < seg.first)
        {
            if(ed != -2e9) res.push_back({st, ed});
            st = seg.first;
            ed = seg.second;
        }
        else ed = max(ed, seg.second);
    }
    if(st != -2e9) res.push_back({st, ed});
    Seg = res;  
}

int main()
{
    cin >> n;
    for(int i = 0; i < n; i ++ )
    {
        int l, r;
        cin >> l >> r;
        seg.push_back({l, r});
    }
    merge(seg);
    cout << seg.size() << endl;
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值