第1讲 基础算法

快速排序

在这里插入图片描述


#include <iostream>

using namespace std;

const int N = 100010;

int q[N];

void quick_sort(int q[], int l, int r)
{
    if (l >= r) return;        //这个表示  区间 只有 一个数了 就 不用 排序,直接退出就好

    int i = l - 1, j = r + 1, x = q[l + r >> 1];            //x表示 q中 的 中间的 值
    while (i < j)
    {
        do i ++ ; while (q[i] < x);         //i是从左边开始 遍历,直到找到q[i]>=x停止;j是从右边 开始 遍历,直到找到q[j]<=x为止; 我们要保证左边的值都小于x,右边的值都大于x
        do j -- ; while (q[j] > x);
        if (i < j) swap(q[i], q[j]);        //如果i<j,交换
    }

    quick_sort(q, l, j);            //先给 左边 递归 处理
    quick_sort(q, j + 1, r);        //再给 右边 递归 处理
}

int main()
{
    int n;
    scanf("%d", &n);            //输入n

    for (int i = 0; i < n; i ++ ) scanf("%d", &q[i]);       //读入 所有 的 数字

    quick_sort(q, 0, n - 1);            //对q,进行 0~n-1中的 一个排序

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

    return 0;
}


第k个数


#include<iostream>
using namespace std;

const int maxn=100010;

int n,k;

int q[maxn];

void quick_sort(int q[],int l,int r)
{
    if(l>=r)
        return ;
    int i=l-1,j=r+1,x=q[l+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()
{
    cin >> n >> k;

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

    quick_sort(q,0,n-1);

    cout << q[k-1];

    return 0;
}

归并排序

在这里插入图片描述


#include <iostream>

using namespace std;

const int N = 1e5 + 10;

int a[N], tmp[N];

void merge_sort(int q[], int l, int r)
{
    if (l >= r) return;     //如果 区间 中的 元素 只有 1个 或者 没有 直接 返回

    int mid = l + r >> 1;           //归并 排序 第1步:先确定 中点

    merge_sort(q, l, mid), merge_sort(q, mid + 1, r);       //归并 排序 第2步:归并排序 左边 或者 右边

    int k = 0, i = l, j = mid + 1;      //tmp表示 临时 数组,k表示 temp中的 元素的 个数
    //i,j是 两个 指针

    while (i <= mid && j <= r)      //i小于 左半边的边界;j小于 右半边的 边界
        if (q[i] <= q[j])       //将小的数字 放到 临时 数组中
            tmp[k ++ ] = q[i ++ ];      //q[i]小 将 q[i]放到 临时 数组中
        else tmp[k ++ ] = q[j ++ ];     //q[j]小 将 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];      //现在 需要 更新 q 这个 数组,j去遍历 temp数组;i去遍历q数组
}

int main()
{
    int n;
    scanf("%d", &n);
    for (int i = 0; i < n; i ++ ) scanf("%d", &a[i]);       //输入n个数字

    merge_sort(a, 0, n - 1);        //从0~n-1进行  归并 排序

    for (int i = 0; i < n; i ++ ) printf("%d ", a[i]);      //输出 a[i]

    return 0;
}


逆序对的数量

在这里插入图片描述

#include <iostream>

using namespace std;

typedef long long LL;

const int N = 1e5 + 10;

int a[N], tmp[N];

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

    int mid = l + r >> 1;

    LL res = 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
        {
            res += mid - i + 1;
            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];

    return res;
}

int main()
{
    int n;
    scanf("%d", &n);                //输入n 
    for (int i = 0; i < n; i ++ ) scanf("%d", &a[i]);       //输入n个 数字

    cout << merge_sort(a, 0, n - 1) << endl;        //直接merge_sort

    return 0;
}

数的范围

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


#include <iostream>

using namespace std;

const int N = 100010;

int n, m;
int q[N];

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

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

        int l = 0, r = n - 1;               //这个就是 找到 第一个 x的 位置,使用第1中 模板
        while (l < r)
        {
            int mid = l + r >> 1;
            if (q[mid] >= x) r = mid;
            else l = mid + 1;
        }

        if (q[r] != x) cout << "-1 -1" << endl;
        else        //否则,就是 q[l]==x
        {
            cout << l << ' ';       //输出 起点

            int l = 0, r = n - 1;           //这个就是 需要 找到 最后一个 x的 位置,使用第2个中的 模板
            while (l < r)
            {
                int mid = l + r + 1 >> 1;
                if (q[mid] <= x) l = mid;
                else r = mid - 1;
            }

            cout << l << endl;
        }
    }

    return 0;
}


数的三次方根


#include <iostream>

using namespace std;

int main()
{
    double x;       //设置一个double类型的 x
    
    cin >> x;   //输入x

    double l = -100, r = 100;
    while (r - l > 1e-8)
    {
        double mid = (l + r) / 2;
        if (mid * mid * mid >= x) r = mid;     //说明 应该 在 左半边 寻找
        else l = mid;
    }

    printf("%.6lf\n", l);
    return 0;
}

高精度加法

#include <iostream>
#include <vector>

using namespace std;

vector<int> add(vector<int> &A, vector<int> &B)
{
    if (A.size() < B.size()) return add(B, A);

    vector<int> C;
    int t = 0;             //t表示 进位
    for (int i = 0; i < A.size(); i ++ )
    {
        t += A[i];         //A[i]是头一个 数字 必须 要 加的
        if (i < B.size()) t += B[i];        //如果 在 长度 之内,继续加B[i]
        C.push_back(t % 10);            //将 余数 放到 数组C中
        t /= 10;        //t/=10,表示 进位
    }

    if (t) C.push_back(t);      //如果t还是 有的话,继续  进位
    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 -- ) cout << C[i];     //然后 倒着 输出 出来就是 结果
    cout << endl;

    return 0;
}


高精度减法

6,color_FFFFFF,t_70)

#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();       //如果A,B长度 不同,如果A的长度>B的长度,返回true,否则返回 false

    //这里表示A,B的长度 相同,从高位开始 比较,如果A的某一位不等于B的某一位,则如果A[i]>B[i],则直接返回true
    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;               //t表示 当前这一位的值
        if (i < B.size()) t -= B[i];
        C.push_back((t + 10) % 10);         //如果t<0,则需要借一位则为10
        if (t < 0)          //t<0,表示 有进位,则t为1,因为 再次循环的话,需要-t
            t = 1;
        else       //否则,表示 没有 进位
            t = 0;
    }
    
    //下面这一步 是 去掉 前导0  比如123-120 得到的 结果 是003,所以 这一步的 作用是 去掉0
    while (C.size() > 1 && C.back() == 0) C.pop_back();             //C.back()表示C的最后 一位,如果为0,则直接 干掉
    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');

    vector<int> C;

    if (cmp(A, B)) C = sub(A, B);           //如果A>B,则直接 减
    else C = sub(B, A), cout << '-';            //如果A<B,则B-A,完了,加上 负号

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

    return 0;
}


高精度乘法


#include <iostream>
#include <vector>

using namespace std;


vector<int> mul(vector<int> &A, int b)
{
    vector<int> C;

    int t = 0;          //t表示 当前 这一位的 数字 
    for (int i = 0; i < A.size() || t; i ++ )         //t不为0
    {
        if (i < A.size()) t += A[i] * b;
        C.push_back(t % 10);
        t /= 10;
    }

    while (C.size() > 1 && C.back() == 0) C.pop_back();

    return C;
}


int main()
{
    string a;
    int b;          

    cin >> a >> b;      //输入a,b 用 a 乘 b

    vector<int> A;
    for (int i = a.size() - 1; i >= 0; i -- ) A.push_back(a[i] - '0');         //将A放进去

    auto C = mul(A, b);         //直接 相乘

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

    return 0;
}


高精度除法

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

using namespace std;

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

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

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

    int r;      //r表示 余数
    auto C = div(A, B, r); 

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

    cout << endl << r << endl;

    return 0;
}

前缀和


#include<iostream>
using namespace std;

const int maxn=1000010;

int n,m;

int a[maxn],s[maxn];

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

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

    while(m--)
    {
        int l,r;
        cin >> l >> r;
        cout << s[r]-s[l-1] << endl;
    }

    return 0;
}

子矩阵的和

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

#include <iostream>

using namespace std;

const int N = 1010;

int n, m, q;
int 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", &s[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];     //s[i][j]+=s[i-1][j]+s[i][j-1]-s[i-1][j-1]

    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;
}


差分

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

#include <iostream>

using namespace std;

const int N = 100010;

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]);      //输入n个数字

    for (int i = 1; i <= n; i ++ ) insert(i, i, a[i]);
    //此时b[i]表示的是a[i]的差分,就是b[i]=a[i]-a[i-1],反过来a是b的前缀和,比如a[3]=b[1]+b[2]+b[3]


    while (m -- )
    {
        int l, r, c;
        scanf("%d%d%d", &l, &r, &c);
        insert(l, r, c);        //l到r这个 区间 每个 数字 加上 c
    }

    for (int i = 1; i <= n; i ++ ) b[i] += b[i - 1];        //由于b是a的差分,所以

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

    return 0;
}


差分矩阵

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


#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);        //输入n行,m列,q个询问

    for (int i = 1; i <= n; i ++ )
        for (int j = 1; j <= m; j ++ )
            scanf("%d", &a[i][j]);                    //输入 a[i][j]

    for (int i = 1; i <= n; i ++ )          //这是对b数组的  一种 小操作
        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;           //输入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;
}


最长连续不重复子序列

在这里插入图片描述

在这里插入图片描述

自己根据 例子 写写 就出来了


#include <iostream>

using namespace std;

const int N = 100010;

int n;
int q[N], s[N];     //s数组 存放的是 每个 数字 出现 多少次

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

    int res = 0;
    for (int i = 0, j = 0; i < n; i ++ )
    {
        s[q[i]] ++ ;        //在s中 将 q[i] 对应的值 ++
        while (j < i && s[q[i]] > 1) s[q[j ++ ]] -- ;
        res = max(res, i - j + 1);
    }

    cout << res << endl;

    return 0;
}


数组元素的目标和


#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 ++ )        //i去遍历 a数组,j去遍历b数组
    {
        while (j >= 0 && a[i] + b[j] > x)
            j -- ;
        if (j >= 0 && a[i] + b[j] == x) 
            cout << i << ' ' << j << endl;
    }

    return 0;
}


判断子序列

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

#include <iostream>
#include <cstring>

using namespace std;

const int N = 100010;

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

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

    int i = 0, j = 0;
    while (i < n && j < m)
    {
        if (a[i] == b[j])
            i ++ ;
        j ++ ;                          //从前往后 扫描 b这个 数组
    }

    if (i == n) puts("Yes");
    else puts("No");

    return 0;
}


二进制中1的个数

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

#include <iostream>

using namespace std;

int main()
{
    int n;
    scanf("%d", &n);        //输入n
    
    while (n -- )
    {
        int x, s = 0;
        scanf("%d", &x);        //x表示当前输入的 数字

        for (int i = x; i; i -= i & -i) 
            s ++ ;

        printf("%d ", s);
    }

    return 0;
}

区间和

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



//下标就是 映射的值

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

using namespace std;

typedef pair<int, int> PII;

const int N = 300010;  //n,m的取值范围是10^5,由于n个位置和m个查询,一个是n+2m,m个查询有两个下标,有可能查询的这两个下标不在n给出的中

int n, m;           //这个值 是 n和m
int a[N], s[N];         //s数组表示 前缀和
//a数组 是  存的 数

vector<int> alls;       //alls存储的是 所有 离散化后的值

vector<PII> add, query;     //将 插入和 查询的 操作读入到 add和query中

int find(int x)     //求x离散化后的 结果
{
    int l = 0, r = alls.size() - 1;
    //通过 二分 的 方式 找到 大于x的最小的数字
    while (l < r)
    {
        int mid = l + r >> 1;
        if (alls[mid] >= x) r = mid;
        else l = mid + 1;
    }
    return l + 1;           //映射 为 从1开始
}



int main()
{
    cin >> n >> m;                 //输入n和m

    for (int i = 0; i < n; i ++ )
    {
        int x, c;
        cin >> x >> c;          
        add.push_back({x, c});      //将所有操作 放入 add中   在下标为x的位置加上c
        alls.push_back(x);      //这个x其实 就是 离散化后的值,放入到alls中
    }

    for (int i = 0; i < m; i ++ )
    {
        int l, r;
        cin >> l >> r;          //读入所有的 左右 区间
        query.push_back({l, r});        //将要查询的结果放入到  query中

        alls.push_back(l);
        alls.push_back(r);      //同时也要将 左右下标 放入到 alls中
    }

    // 去重
    sort(alls.begin(), alls.end());         //先对alls进行 排序
    alls.erase(unique(alls.begin(),alls.end()), alls.end());       //去除掉alls中的 重复 元素

    // 处理插入操作
    for (auto item : add)
    {
        int x = find(item.first);           //x其实就是 离散化 后的  下标
        a[x] += item.second;        //在这个位置 加上 item.second
    }

    // 预处理前缀和
    for (int i = 1; i <= alls.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;
}




区间合并

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


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

const int maxn=100010;

int n;

vector<pair<int,int>> seg;

void merge(vector<pair<int,int>> &seg)
{
    vector<pair<int,int>> res;

    sort(seg.begin(),seg.end());

    int st=-2e9,ed=-2e9;

    for(auto item:seg)
        if(ed<item.first)
        {
                if(st!=-2e9)
                {
                    res.push_back({st,ed});
                }
                st=item.first,ed=item.second;
        }
        else
        {
            ed=max(ed,item.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();

    return 0;
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值