2018年全国多校算法寒假训练营练习比赛(第五场)解题报告

这篇博客记录了2018年全国多校算法寒假训练营练习比赛的解题报告,包括A到H共8道题目。各题目涉及逆序数计算、数列操作、字符串问题、集合分配、电灯泡问题、数字变换、博弈问题及数组区间操作等算法问题。博主提供了每道题目的思路和解法,如堆排列、线段树、字符串处理等。
摘要由CSDN通过智能技术生成

A-逆序数

https://www.nowcoder.com/acm/contest/77/A

题目描述

在一个排列中,如果一对数的前后位置与大小顺序相反,即前面的数大于后面的数,那么它们就称为一个逆序。一个排列中逆序的总数就称为这个排列的逆序数。比如一个序列为4 5 1 3 2, 那么这个序列的逆序数为7,逆序对分别为(4, 1), (4, 3), (4, 2), (5, 1), (5, 3), (5, 2),(3, 2)。

输入描述:

第一行有一个整数n(1 <= n <= 100000), 然后第二行跟着n个整数,对于第i个数a[i],(0 <= a[i] <= 100000)。

输出描述:

输出这个序列中的逆序数

思路: 堆排列的过程正好可以计算逆序数。

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

const int maxn = 1e5+10;
int arr[maxn], tmp[maxn];
long long ans;

void merge(int *a, int st, int mid, int ed) {
    int i = st, j = mid + 1, k = st;
    while (i <= mid && j <= ed) {
        if (a[i] <= a[j]) {
            tmp[k++] = a[i++];
        } else {
            ans += j - k;
            tmp[k++] = a[j++];
        }
    }

    while (i <= mid) {
        tmp[k++] = a[i++];
    }
    while (j <= ed) {
        tmp[k++] = a[j++];
    }
    for (int i = st; i <= ed; i++) {
        a[i] = tmp[i];
    }
}

void merge_sort(int *a, int st, int ed) {
    if (st < ed) {
        int mid = (st+ed) / 2;
        merge_sort(a, st, mid);
        merge_sort(a, mid+1, ed);
        merge(a, st, mid, ed);
    }
}

int main(void) {
    int n;
    scanf("%d", &n);
    for (int i = 0; i < n; i++) {
        scanf("%d", &arr[i]);
    }
    merge_sort(arr, 0, n-1);
    cout << ans << endl;
} 

B-Big Water Problem

题目描述

给一个数列,会有多次询问,对于每一次询问,会有两种操作:
1:给定两个整数x, y, 然后在原数组的第x位置上加y;
2:给定两个整数l,r,然后输出数组从第l位加到第r位数字的和并换行

输入描述:

第一行有两个整数n, m(1 <= n, m <= 100000)代表数列的长度和询问的次数
第二行n个数字,对于第i个数字a[i],(0<=a[i]<=100000)。
接下来m行,每一行有三个整数f, x, y。第一个整数f是1或者是2,代表操作类型,如果是1,接下来两个数x,y代表第x的位置上加y,如果是2,则求x到y的和,保证数据合法。

输出描述:

输出每次求和的结果并换行

思路: 线段树单点更新模板题

#include <iostream>
#include <cstring>
#include <algorithm>
using namespace std;
const int maxn = 1e5+10;
int tree[maxn<<2];

inline void pushUp(int node) {
    tree[node] = tree[node<<1] + tree[node<<1|1];
}

void build(int node, int l, int r) {
    if (l == r) cin >> tree[node];
    else {
        build(node<<1, l, (l+r) / 2);
        build((node<<1) +1, (l+r) / 2 + 
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值