• 782. Transform to Chessboard


    An N x N board contains only 0s and 1s. In each move, you can swap any 2 rows with each other, or any 2 columns with each other.

    What is the minimum number of moves to transform the board into a "chessboard" - a board where no 0s and no 1s are 4-directionally adjacent? If the task is impossible, return -1.

    Examples:
    Input: board = [[0,1,1,0],[0,1,1,0],[1,0,0,1],[1,0,0,1]]
    Output: 2
    Explanation:
    One potential sequence of moves is shown below, from left to right:
    
    0110     1010     1010
    0110 --> 1010 --> 0101
    1001     0101     1010
    1001     0101     0101
    
    The first move swaps the first and second column.
    The second move swaps the second and third row.
    
    
    Input: board = [[0, 1], [1, 0]]
    Output: 0
    Explanation:
    Also note that the board with 0 in the top left corner,
    01
    10
    
    is also a valid chessboard.
    
    Input: board = [[1, 0], [1, 0]]
    Output: -1
    Explanation:
    No matter what sequence of moves you make, you cannot end with a valid chessboard.

    Note:

    • board will have the same number of rows and columns, a number in the range [2, 30].
    • board[i][j] will be only 0s or 1s.

    Approach #1: Array. [Math]

    class Solution {
        public int movesToChessboard(int[][] b) {
            int N = b.length, rowSum = 0, colSum = 0, rowSwap = 0, colSwap = 0;
            for (int i = 0; i < N; ++i) for (int j = 0; j < N; ++j)
                if ((b[0][0] ^ b[i][0] ^ b[0][j] ^ b[i][j]) == 1) return -1;
            for (int i = 0; i < N; ++i) {
                rowSum += b[0][i];
                colSum += b[i][0];
                if (b[i][0] == i % 2) rowSwap++;
                if (b[0][i] == i % 2) colSwap++;
            }
            if (rowSum != N/2 && rowSum != (N+1)/2) return -1;
            if (colSum != N/2 && colSum != (N+1)/2) return -1;
            if (N % 2 == 1) {
                if (colSwap % 2 == 1) colSwap = N - colSwap;
                if (rowSwap % 2 == 1) rowSwap = N - rowSwap;
            } else {
                colSwap = Math.min(N-colSwap, colSwap);
                rowSwap = Math.min(N-rowSwap, rowSwap);
            }
            return (colSwap + rowSwap) / 2;
        }
    }
    

      

    Analysis:

    In a valid chess board, there are 2 and only 2 kinds of rows and one is inverse to the other. For example if there is a row 0101001 in the board, any other row must be either 0101001 or 1010110.

    The same for colums.

    A corollary is that, any rectangle inside the board with corners top left, top right, bottom left, bottom right must be 4 zeros or 2 zeros 2 ones or 4 ones.

    Another important property is that every row and column has half ones. Assume the board is N * N:

    If N = 2  * K, every row and every colum has K ones and K zeros.

    If N = 2 * K + 1, every row and every col has K ones and K + 1 zeros or K + 1 ones and K zeros.

    Since the swap process does not break this property, for a fiven board to be valid, this property must hold.

    These two conditions are necessary and sufficient condition for a calid chessboard.

    Once we know it is a valid cheese board, we start to count swaps.

    Basic on the property above, when we arange the first row, we are actually moving all columns.

    I try to arrange one row into 01010 and 10101 and I count the number of swap.

    In case of N even, I take the minimum swaps, because both are possible.

    In case of N odd, I take the even swaps.

    Because when we make a swap, we move 2 columns or 2 rows at the same time.

    So col swaps and row swaps shoule be same here.

    Reference:

    https://leetcode.com/problems/transform-to-chessboard/discuss/114847/C%2B%2BJavaPython-Solution-with-Explanation

    永远渴望,大智若愚(stay hungry, stay foolish)
  • 相关阅读:
    Tomcat下bootstrap启动分析
    Ubuntu读取/root/.profile时发现错误:mesg:ttyname fa
    【转载】Activiti delete process definition by key
    Ubuntu16解锁root
    Activiti源码:ActivitiEventSupport类中eventListeners的设计
    [转载] Activiti Tenant Id 字段释疑
    Linux下使用NTFS格式移动硬盘
    Linux dd命令制作U盘启动盘
    Activiti源码:StandaloneInMemProcessEngineConfiguration与SpringProcessEngineConfiguration
    Activiti源码学习:ExecutionListener与TaskListener的区别
  • 原文地址:https://www.cnblogs.com/h-hkai/p/10662760.html
Copyright © 2020-2023  润新知