P1966 [NOIP2013 提高组] 火柴排队 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn)
- 将两数组排序,(排序前要记录每个数对应的下标,之后会用到)
- 排好序之后两个数组就是理想的状态(即第一个数组对应第i大的数和第二个数组对应第i大的数对其,是最优解),需要知道得到这两个状态需要移动多少次,那么以first的idx(原第一个数组的下标)为下标,值为sec的idx(原第二个数组的下标)为值的数组a,a数组的逆序对就是答案,(要交换成当前理想状态需要的最少次数)
- 用归并排序求逆序对(就是当左边的数比右边的数值要大时,用ans累加)
#include <bits/stdc++.h>
using namespace std;
#define N 1e5
#define INF 2e9
#define MAX 100005
#define mod 99999997
#define ll long long
int ans;
void merge(long long int list[], long long int temp[], int mid, int left, int right)
{
int left_p = left, right_p = mid + 1, t_p = left;
while (left_p <= mid && right_p <= right)
{
if (list[left_p] <= list[right_p])
{
temp[t_p++] = list[left_p++];
}
else
{
temp[t_p++] = list[right_p++];
ans += mid - left_p + 1;
ans %= mod;
}
}
while (left_p <= mid)
temp[t_p++] = list[left_p++];
while (right_p <= right)
temp[t_p++] = list[right_p++];
for (int i = left; i <= right; i++)
{
list[i] = temp[i];
}
}
void mesort(long long int list[], long long int temp[], int n, int left, int right)
{
if (left < right)
{
int mid = (left + right) >> 1;
mesort(list, temp, n, left, mid);
mesort(list, temp, n, mid + 1, right);
merge(list, temp, mid, left, right);
}
}
void merg_sort(long long int list[], int n)
{
long long int *temp = (long long int *)malloc(sizeof(long long int) * (n + 1));
mesort(list, temp, n, 1, n);
}
struct node
{
int num, idx;
} first_[MAX], sec_[MAX];
ll a[MAX];
int n;
int main()
{
cin >> n;
for (int i = 1; i <= n; i++)
{
scanf("%d", &first_[i].num);
first_[i].idx = i;
}
for (int i = 1; i <= n; i++)
{
scanf("%d", &sec_[i].num);
sec_[i].idx = i;
}
sort(first_ + 1, first_ + n + 1, [](node a, node b)
{ return a.num < b.num; });
sort(sec_ + 1, sec_ + n + 1, [](node a, node b)
{ return a.num < b.num; });
for (int i = 1; i <= n; i++)
{
a[first_[i].idx] = sec_[i].idx;
}
merg_sort(a, n);
cout << ans % mod << endl;
}
归并求逆序对
int ans;
void merge(long long int list[], long long int temp[], int mid, int left, int right)
{
int left_p = left, right_p = mid + 1, t_p = left;
while (left_p <= mid && right_p <= right)
{
if (list[left_p] <= list[right_p])
{
temp[t_p++] = list[left_p++];
}
else
{
temp[t_p++] = list[right_p++];
ans += mid - left_p + 1;
ans %= mod;
}
}
while (left_p <= mid)
temp[t_p++] = list[left_p++];
while (right_p <= right)
temp[t_p++] = list[right_p++];
for (int i = left; i <= right; i++)
{
list[i] = temp[i];
}
}
void mesort(long long int list[], long long int temp[], int n, int left, int right)
{
if (left < right)
{
int mid = (left + right) >> 1;
mesort(list, temp, n, left, mid);
mesort(list, temp, n, mid + 1, right);
merge(list, temp, mid, left, right);
}
}
void merg_sort(long long int list[], int n)
{
long long int *temp = (long long int *)malloc(sizeof(long long int) * (n + 1));
mesort(list, temp, n, 1, n);
}
标签:temp,int,排队,list,mid,long,火柴,left From: https://www.cnblogs.com/Wang-Xianyi/p/16652161.html