Advertisement

【acwing算法基础课笔记】01-基础算法-习题课

阅读量:

  1. 786 求第k个数 - 快速选择
  2. 788 逆序对的数量 - 归并排序
  3. 790 数的三次方根 - 浮点数二分
  4. 796 子矩阵的和 - 前缀和
  5. 798 差分矩阵 - 差分
  6. 799 无重复字符的最长字串

给定一个浮点数 n,求它的三次方根。

790 数的三次方根 - 浮点数二分

输入格式

共一行,包含一个浮点数 n。

输出格式

共一行,包含一个浮点数,表示问题的解。

注意,结果保留 6 位小数。

数据范围

−10000≤n≤10000

输入样例:

1000.00

输出样例:

10.000000

复制代码
    #include<bits/stdc++.h>
    using namespace std;
     
    int main(){
    double x;cin>>x;
    double l=-10000,r=10000;
    while(r-l>1e-8){//-7的话可能四舍五入不对,比有效数多两位保险
        double mid=(r+l)/2;
        if(mid*mid*mid>=x)r=mid;
        else l=mid;
    }
    printf("%.6f",r);
    }
    
    
      
      
      
      
      
      
      
      
      
      
      
      
      
    

796 子矩阵的和 - 前缀和

题目描述

输入一个 n 行 m 列的整数矩阵,再输入 q 个询问,每个询问包含四个整数 x1,y1,x2,y2,表示一个子矩阵的左上角坐标和右下角坐标。
对于每个询问输出子矩阵中所有数的和。

输入格式

第一行包含三个整数 n,m,q。
接下来 n 行,每行包含 m 个整数,表示整数矩阵。
接下来 q 行,每行包含四个整数 x1,y1,x2,y2,表示一组询问。

输出格式

共 q 行,每行输出一个询问的结果。

数据范围

1≤n,m≤1000,
1≤q≤200000,
1≤x1≤x2≤n,
1≤y1≤y2≤m,
-1000≤矩阵内元素的值≤1000

输入样例:

复制代码
    3 4 3
    1 7 2 4
    3 6 2 8
    2 1 2 3
    1 1 2 2
    2 1 3 4
    1 3 3 4
    
    
      
      
      
      
      
      
      
    

输出样例:

复制代码
    17
    27
    21
    
    
      
      
      
    

解 - 前缀和

复制代码
    #include <iostream>
    
    using namespace std;
    
    const int N = 1010; // 设置矩阵最大大小
    
    int a[N][N], s[N][N]; // a:原始矩阵,s:a的前缀和矩阵
    
    int main()
    {
    int n, m, q;
    cin >> n >> m >> q;
    for (int i = 1; i <= n; ++i) // 从下标1开始,避免边界问题
        for (int j = 1; j <= n; ++j)
        {
            cin >> a[i][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; // 查询的子矩阵左上右下坐标
        cin >> x1 >> y1 >> x2 >> y2;
        cout << s[x2][y2] - s[x2][y1 - 1] - s[x1 - 1][y2] + s[x1 - 1][y1 - 1];
    }
    }
    
    
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
    

798 差分矩阵 - 差分

题目描述

入一个 n 行 m 列的整数矩阵,再输入 q 个操作,每个操作包含五个整数 x1,y1,x2,y2,c,其中 (x1,y1) 和 (x2,y2) 表示一个子矩阵的左上角坐标和右下角坐标。

每个操作都要将选中的子矩阵中的每个元素的值加上 c 。

请你将进行完所有操作后的矩阵输出。

输入格式

第一行包含整数 n,m,q。

接下来 n 行,每行包含 m 个整数,表示整数矩阵。

接下来 q 行,每行包含 5 个整数 x1,y1,x2,y2,c, 表示一个操作。

输出格式

共 n 行,每行 m 个整数,表示所有操作进行完毕后的最终矩阵。

数据范围

1≤n,m≤1000,
1≤q≤100000,
1≤x1≤x2≤n,
1≤y1≤y2≤m,
-1000≤c≤1000,
-1000≤矩阵内元素的值≤1000

复制代码
    #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[x1+1][y1] -= c;
    b[x1][y1+1] -= c;
    b[x1+1][y1+1] += c;
    }
    
    int main(){
    // 输入 a 前缀和数组
    cin >> n >> m >> q;
    for(int i=1;i<=n;++i)
        for(int j=1;j<=m;++j)
            cin >> a[i][j];
    
    // 构建 b 差分数组
    for(int i=1;i<=n;++i)
        for(int j=1;j<=m;++j)
            insert(i,j,i,j,a[i][j]);
    
    // 对a的区间整体操作O(n)映射到b的几个点操作,n次O(1)操作
    while(q--){
        int x1,y1,x2,y2,c;
        cin >> x1 >> x2 >> x2 >> y2 >> c;
        insert(x1,y1,x2,y2,c);
    }
    // 转换 b 数组到 a 输出
    for(int i=1;i<=n;++i)
        for(int j=1;j<=m;++j){
            b[i][j] += b[i-1][j-1] - b[i-1][j] - b[i][j-1] + b[i-1][j-1];
            cout << b[i][j];
        }
    
    }
    
    
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
    

799 无重复字符的最长字串

https://leetcode.cn/problems/longest-substring-without-repeating-characters/description/

给定一个字符串 s ,请你找出其中不含有重复字符的 最长 子串 的长度。

示例 1:

输入: s = “abcabcbb”
输出: 3
解释: 因为无重复字符的最长子串是 “abc”,所以其长度为 3。

示例 2:

输入: s = “bbbbb”
输出: 1
解释: 因为无重复字符的最长子串是 “b”,所以其长度为 1。
示例 3:

输入: s = “pwwkew”
输出: 3
解释: 因为无重复字符的最长子串是 “wke”,所以其长度为 3。

请注意,你的答案必须是 子串 的长度,“pwke” 是一个子序列,不是子串。

力扣的方法一:滑动窗口 解答写的很清楚

模板

快速排序算法模板 —— 模板题 AcWing 785. 快速排序
复制代码
    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);
    }
    
    
    
    
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
    

归并排序算法模板 —— 模板题 AcWing 787. 归并排序
复制代码
    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];
    }
    
    
    
    
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
    

整数二分算法模板 —— 模板题 AcWing 789. 数的范围
复制代码
    bool check(int x) {} 
    
    
    int bsearch_1(int l, int r)
    {
    while (l < r)
    {
        int mid = l + r >> 1;
        if (check(mid)) r = mid;    
        else l = mid + 1;
    }
    return l;
    }
    
    int bsearch_2(int l, int r)
    {
    while (l < r)
    {
        int mid = l + r + 1 >> 1;
        if (check(mid)) l = mid;
        else r = mid - 1;
    }
    return l;
    }
    
    
    
    
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
    

浮点数二分算法模板 —— 模板题 AcWing 790. 数的三次方根
复制代码
    bool check(double x) {} 
    
    double bsearch_3(double l, double r)
    {
    const double eps = 1e-6;   
    while (r - l > eps)
    {
        double mid = (l + r) / 2;
        if (check(mid)) r = mid;
        else l = mid;
    }
    return l;
    }
    
    
    
    
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
    

高精度加法 —— 模板题 AcWing 791. 高精度加法
复制代码
    vector<int> add(vector<int> &A, vector<int> &B)
    {
    if (A.size() < B.size()) return add(B, A);
    
    vector<int> C;
    int t = 0;
    for (int i = 0; i < A.size(); i ++ )
    {
        t += A[i];
        if (i < B.size()) t += B[i];
        C.push_back(t % 10);
        t /= 10;
    }
    
    if (t) C.push_back(t);
    return C;
    }
    
    
    
    
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
    

高精度减法 —— 模板题 AcWing 792. 高精度减法
复制代码
    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;
    }
    
    
    
    
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
    

高精度乘低精度 —— 模板题 AcWing 793. 高精度乘法
复制代码
    vector<int> mul(vector<int> &A, int b)
    {
    vector<int> C;
    
    int t = 0;
    for (int i = 0; i < A.size() || t; i ++ )
    {
        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;
    }
    
    
    
    
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
    

高精度除以低精度 —— 模板题 AcWing 794. 高精度除法
复制代码
    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 %= b;
    }
    reverse(C.begin(), C.end());
    while (C.size() > 1 && C.back() == 0) C.pop_back();
    return C;
    }
    
    
    
    
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
    

一维前缀和 —— 模板题 AcWing 795. 前缀和
复制代码
    S[i] = a[1] + a[2] + ... a[i]
    a[l] + ... + a[r] = S[r] - S[l - 1]
    
    
      
      
    

二维前缀和 —— 模板题 AcWing 796. 子矩阵的和
复制代码
    S[i, j] = 第i行j列格子左上部分所有元素的和
    以(x1, y1)为左上角,(x2, y2)为右下角的子矩阵的和为:
    S[x2, y2] - S[x1 - 1, y2] - S[x2, y1 - 1] + S[x1 - 1, y1 - 1]
    
    
      
      
      
    

一维差分 —— 模板题 AcWing 797. 差分
复制代码
    给区间[l, r]中的每个数加上c:B[l] += c, B[r + 1] -= c
    
    
      
    

二维差分 —— 模板题 AcWing 798. 差分矩阵
复制代码
    给以(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
    
    
      
      
    

位运算 —— 模板题 AcWing 801. 二进制中 1 的个数
复制代码
    求n的第k位数字: n >> k & 1
    返回n的最后一位1:lowbit(n) = n & -n
    
    
      
      
    

双指针算法 —— 模板题 AcWIng 799. 最长连续不重复子序列, AcWing 800. 数组元素的目标和
复制代码
    for (int i = 0, j = 0; i < n; i ++ )
    {
    while (j < i && check(i, j)) j ++ ;
    }
    常见问题分类:
    (1) 对于一个序列,用两个指针维护一段区间
    (2) 对于两个序列,维护某种次序,比如归并排序中合并两个有序序列的操作
    
    
      
      
      
      
      
      
      
    

离散化 —— 模板题 AcWing 802. 区间和
复制代码
    vector<int> alls; 
    sort(alls.begin(), alls.end()); 
    alls.erase(unique(alls.begin(), alls.end()), alls.end());   
    
    
    int find(int x) 
    {
    int l = 0, r = alls.size() - 1;
    while (l < r)
    {
        int mid = l + r >> 1;
        if (alls[mid] >= x) r = mid;
        else l = mid + 1;
    }
    return r + 1; 
    }
    
    
    
    
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
    

区间合并 —— 模板题 AcWing 803. 区间合并
复制代码
    void merge(vector<PII> &segs)
    {
    vector<PII> res;
    
    sort(segs.begin(), segs.end());
    
    int st = -2e9, ed = -2e9;
    for (auto seg : segs)
        if (ed < seg.first)
        {
            if (st != -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});
    
    segs = res;
    }
    
    
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
      
    

全部评论 (0)

还没有任何评论哟~