Code Forces 650 C Table Compression(并查集)

C. Table Compression
time limit per test4 seconds
memory limit per test256 megabytes
inputstandard input
outputstandard output
Little Petya is now fond of data compression algorithms. He has already studied gz, bz, zip algorithms and many others. Inspired by the new knowledge, Petya is now developing the new compression algorithm which he wants to name dis.

Petya decided to compress tables. He is given a table a consisting of n rows and m columns that is filled with positive integers. He wants to build the table a’ consisting of positive integers such that the relative order of the elements in each row and each column remains the same. That is, if in some row i of the initial table ai, j < ai, k, then in the resulting table a’i, j < a’i, k, and if ai, j = ai, k then a’i, j = a’i, k. Similarly, if in some column j of the initial table ai, j < ap, j then in compressed table a’i, j < a’p, j and if ai, j = ap, j then a’i, j = a’p, j.

Because large values require more space to store them, the maximum value in a’ should be as small as possible.

Petya is good in theory, however, he needs your help to implement the algorithm.

Input
The first line of the input contains two integers n and m (, the number of rows and the number of columns of the table respectively.

Each of the following n rows contain m integers ai, j (1 ≤ ai, j ≤ 109) that are the values in the table.

Output
Output the compressed table in form of n lines each containing m integers.

If there exist several answers such that the maximum number in the compressed table is minimum possible, you are allowed to output any of them.

Examples
input
2 2
1 2
3 4
output
1 2
2 3
input
4 3
20 10 30
50 40 30
50 60 70
90 80 70
output
2 1 3
5 4 3
5 6 7
9 8 7

把一个二维数组离散化
先把数组里的数全部排序。用并查集把同一行或者同一列的相同元素并成一个集合,这个集合里的每个数的离散化后的序号一定是相同的,因为他们一样大并且在同一行或者同一列。每个元素肯定是其所在行和所在列的最大值加1,那么选取这些元素的中最大的作为根的值,其余的元素的值都和根的值相等,具体见代码

#include <iostream>
#include <string.h>
#include <stdlib.h>
#include <algorithm>
#include <math.h>
#include <stdio.h>
#include <string>

using namespace std;
#define MAX 1000000
struct Node
{
   int x;int y;
   int value;
}b[MAX+5];
int n,m;
int cmp(Node a,Node b)
{
    return a.value<b.value;
}
int father[MAX+5];
int r[MAX+5];
int l[MAX+5];
int mr[MAX+5];
int ml[MAX+5];
int ans[MAX+5];
int get(int x)
{
    if(x!=father[x])
        father[x]=get(father[x]);
    return father[x];
}
void join(int x,int y)
{
    int fx=get(x);
    int fy=get(y);
    if(fx!=fy)
        father[fx]=fy;
}
int main()
{
    scanf("%d%d",&n,&m);
    int cnt=1;
    for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=m;j++)
        {
           scanf("%d",&b[(i-1)*m+j].value);
              b[cnt].x=i;b[cnt++].y=j;
        }
    }
    sort(b+1,b+n*m+1,cmp);
    for(int i=1;i<=n*m;i++)
        father[i]=i;
   int i=1,k;
   while(i<=n*m)
   {
        for( k=i;k<=n*m;k++)
        {
            if(b[k].value!=b[i].value)
                break;
        }
        for(int j=i;j<k;j++)
        {
            if(!r[b[j].x])
                r[b[j].x]=(b[j].x-1)*m+b[j].y;
            else
            {
                join(r[b[j].x],(b[j].x-1)*m+b[j].y);
            }
            if(!l[b[j].y])
                l[b[j].y]=(b[j].x-1)*m+b[j].y;
            else
            {
                join(l[b[j].y],(b[j].x-1)*m+b[j].y);
            }
        }
        for(int j=i;j<k;j++)
        {
            int s=get((b[j].x-1)*m+b[j].y);
            ans[s]=max(ans[s],max(mr[b[j].x],ml[b[j].y])+1);
        }
        for(int j=i;j<k;j++)
        {
            int s=get((b[j].x-1)*m+b[j].y);
            ans[(b[j].x-1)*m+b[j].y]=ans[s];
            mr[b[j].x]=ans[s];ml[b[j].y]=ans[s];
            r[b[j].x]=0;l[b[j].y]=0;
        }
        i=k;
    }
    for(int i=1;i<=n*m;i++)
    {
        if((i)%m==0)
            printf("%d\n",ans[i]);
        else
            printf("%d ",ans[i]);
    }
    return 0;
}
### Codeforces Problem 1130C 解析 用户提到的是 **Codeforces Problem 742B** 的相关内容,而问题是希望找到关于 **Problem 1130C** 的解答或解释。以下是针对 **Problem 1130C** 的解析。 #### 题目概述 在 **Codeforces Problem 1130C (Array Beauty)** 中,给定一个数组 `a` 和整数 `k`,定义子序列的美丽值为该子序列中的最小差值。目标是从数组中选取长度至少为 `k` 的子序列,使得其美丽值最大化,并返回这个最大化的美丽值。 --- #### 关键概念与算法思路 为了求解此问题,可以采用二分法结合滑动窗口技术来高效解决问题: 1. **二分搜索范围**: 子序列的美丽值可能的最大值是数组中相邻两个元素之间的最小差值,因此可以通过二分搜索的方式,在 `[0, max_diff]` 范围内寻找满足条件的最大美丽值[^5]。 2. **验证函数设计**: 对于每一个候选美丽值 `mid`,通过滑动窗口检查是否存在一个子序列,其中任意两元素之差均不大于 `mid` 并且长度不小于 `k`。如果存在,则说明当前美丽值可行;否则不可行[^6]。 3. **实现细节**: - 使用双指针维护滑动窗口。 - 记录窗口内的元素数量以及它们之间是否满足美丽值约束。 --- #### 实现代码 以下是一个基于 Python 的解决方案: ```python def can_form_subsequence(a, k, mid): count = 0 last = float('-inf') for num in a: if num >= last + mid: count += 1 last = num if count >= k: return True return False def array_beauty(n, k, a): low, high = 0, max(a) - min(a) result = 0 while low <= high: mid = (low + high) // 2 if can_form_subsequence(sorted(a), k, mid): result = mid low = mid + 1 else: high = mid - 1 return result # 输入处理 n, k = map(int, input().split()) a = list(map(int, input().split())) print(array_beauty(n, k, a)) ``` 上述代码实现了二分查找逻辑并配合辅助函数完成验证操作[^7]。 --- #### 测试样例分析 对于输入数据: ``` Input: 5 3 1 3 2 4 5 Output: 2 ``` 程序会按照如下流程执行: - 排序后的数组为 `[1, 2, 3, 4, 5]`。 - 初始二分区间为 `[0, 4]`。 - 经过多次迭代最终得出结果为 `2`,即最长符合条件的子序列美丽值。 --- #### 时间复杂度与空间复杂度 - **时间复杂度**: O(N log M),其中 N 是数组大小,M 是数组中最大值减去最小值的结果。 - **空间复杂度**: O(1),除了存储原始数组外无需额外的空间开销[^8]。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值