GitHub - jzplp/aoapc-UVA-Answer: 算法竞赛入门经典 例题和习题答案 刘汝佳 第二版
使用了算法竞赛入门经典第二版上介绍的方法:
一共有四个集合。先将前两个集合的元素两两相加,组成一个长度为n*n的数组,然后将这个数字排序好。这里耗时n^2和nlogn,最后时间复杂度为n^2。
然后再遍历后两个集合中的每个元素,两两相加,和为sum。如果在上面排序好的数组中找到一个值为-sum,那说明找到一个和为0的情况。由于排序好的数组中可能有重复元素,因此需要找到对应值的上界和下界。这里用二分查找方法,书中前面也讲过。这里的时间复杂度是n^2logn。
因此总复杂度是n^2logn。
AC代码
cpp
#include <stdio.h>
#include <stdlib.h>
int arr[4][4100];
int n;
int count = 0;
int addArr[17000000];
int compare(const void *left, const void *right)
{
return *(int *)left - *(int *)right;
}
int lowerBound(int num)
{
int beg = 0, end = n * n, mid;
while (end > beg)
{
mid = beg + (end - beg) / 2;
if (addArr[mid] >= num)
end = mid;
if (addArr[mid] < num)
beg = mid + 1;
}
return beg;
}
int upperBound(int num)
{
int beg = 0, end = n * n, mid;
while (end > beg)
{
mid = beg + (end - beg) / 2;
if (addArr[mid] > num)
end = mid;
if (addArr[mid] <= num)
beg = mid + 1;
}
return beg;
}
void computed()
{
int i, j, sum, upper, lower;
for (i = 0; i < n; ++i)
{
for (j = 0; j < n; ++j)
addArr[i * n + j] = arr[0][i] + arr[1][j];
}
qsort(addArr, n * n, sizeof(int), compare);
for (i = 0; i < n; ++i)
{
for (j = 0; j < n; ++j)
{
sum = arr[2][i] + arr[3][j];
upper = upperBound(-sum);
lower = lowerBound(-sum);
count += upper - lower;
}
}
}
int main()
{
int t, i, j;
scanf("%d", &t);
while (t--)
{
scanf("%d", &n);
for (i = 0; i < n; ++i)
{
for (j = 0; j < 4; ++j)
scanf("%d", &arr[j][i]);
}
count = 0;
computed();
printf("%d\n", count);
if (t != 0)
putchar('\n');
}
}