CF1458C Latin Square

对于一个排列 \(p_i\) ,将其表示为 \((i,p_i)\) , 那么它的逆排列可表示为 \((p_i,i)\)

这道题 \(i,j,a_{i,j}\) 均为排列,考虑用三元组 \((i,j,a_{i,j})\) 表示。(为了方便下标从 \(0\) 开始)

那么操作可表示为:

  • R \((i,j,k) \to (i,(j+1)\bmod n,k)\)
  • L \((i,j,k) \to (i,(j-1)\bmod n,k)\)
  • D \((i,j,k) \to ((i+1)\bmod n,j,k)\)
  • U \((i,j,k) \to ((i-1)\bmod n,j,k)\)
  • I \((i,j,k) \to (i,k,j)\)
  • C \((i,j,k) \to (k,j,i)\)

只需要记录每一维的偏移量与最终的位置即可。

时间复杂度 \(\mathcal O(n^2+m)\)

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

const int MAXN = 1000 , MAXM = 1e5;
int T , n , m , a[ MAXN + 5 ][ MAXN + 5 ][ 3 ] , b[ MAXN + 5 ][ MAXN + 5 ];
char op[ MAXM + 5 ];

int tag[ 3 ] , p[ 3 ];
int main( ) {
    scanf("%d",&T);
    while( T -- ) {
        scanf("%d %d",&n,&m);
        for( int i = 0 ; i < n ; i ++ )
            for( int j = 0 ; j < n ; j ++ ) {
                a[ i ][ j ][ 0 ] = i, a[ i ][ j ][ 1 ] = j;
                scanf("%d",&a[ i ][ j ][ 2 ]); a[ i ][ j ][ 2 ] --;
            }

        scanf("%s", op + 1 );
        for( int i = 0 ; i < 3 ; i ++ ) p[ i ] = i , tag[ i ] = 0;
        for( int i = 1 ; i <= m ; i ++ ) {
            if( op[ i ] == 'R' ) tag[ p[ 1 ] ] ++;
            if( op[ i ] == 'L' ) tag[ p[ 1 ] ] --;
            if( op[ i ] == 'D' ) tag[ p[ 0 ] ] ++;
            if( op[ i ] == 'U' ) tag[ p[ 0 ] ] --;
            if( op[ i ] == 'I' ) swap( p[ 1 ] , p[ 2 ] );
            if( op[ i ] == 'C' ) swap( p[ 0 ] , p[ 2 ] );
        } 
        for( int i = 0 ; i < 3 ; i ++ ) tag[ i ] = ( tag[ i ] % n + n ) % n;
        for( int i = 0 ; i < n ; i ++ )
            for( int j = 0 ; j < n ; j ++ )
                b[ ( a[ i ][ j ][ p[ 0 ] ] + tag[ 0 ] ) % n ][ ( a[ i ][ j ][ p[ 1 ] ] + tag[ 1 ] ) % n ] = ( a[ i ][ j ][ p[ 2 ] ] + tag[ 2 ] ) % n;
        for( int i = 0 ; i < n ; i ++ )
            for( int j = 0 ; j < n ; j ++ )
                printf("%d%c", b[ i ][ j ] + 1 , j == n - 1 ? '\n' : ' ' );
    }
    return 0;
}
posted @ 2022-10-25 22:11  chihik  阅读(12)  评论(0编辑  收藏  举报