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 +