LeetCode题解:Merge Sorted Array

Merge Sorted Array


Given two sorted integer arrays A and B, merge B into A as one sorted array.

Note:
You may assume that A has enough space to hold additional elements from B. The number of elements initialized in A and B are m and n respectively.

思路:

我们当然可以立刻想到,申请一块新的长度为m+n的数组,进行归并排序,然后把数组拷贝(或者交换)回去。这个如果可以使用std::vector和swap的话,似乎也是一个有效的方法。但是题目中给定的参数是C数组,所以看来需要进行拷贝。这样归并的时间复杂度是O(m+n),同时拷贝的时间复杂度也是O(m+n)。

当然,如果这是面试题的话,那么问题可能没这么简单。可以做一个假设:面试官要求只能使用O(1)的空间。这样就不得不考虑利用数组A中的额外空间作为临时储存数据的地方。

一开始的考虑是,将merge时较大的数据保存在A数组额外的空间里,但是这样的话,需要考虑的情况很多,例如m和n之间的关系等等。

另外的考虑是,merge的数据保存在A额外的空间中,当额外空间用完之后,开始从A的起点开始继续,这时只要A的数据有被归并的话,起点数据是可以作废的。归并到最后,得到一个向左旋转n的数组。有关rotate和reverse之间的关系参见编程珠矶相关章节。

题解:

class Solution {
public:
    void merge(int A[], int m, int B[], int n) {
        // Iterator used through merging
        int merge_A = 0; 
        int merge_B = 0;
        
        // merge is done at the end of m-th element, A[]
        int merge_iter = m;
        
        while(merge_A < m || merge_B < n)
        {
            // check if we reached the end of merge region, if so, 
            // start from beginning
            if (merge_iter == m + n)
                merge_iter = 0;

            if (merge_A == m)  // merge A is done
                A[merge_iter++] = B[merge_B++];
            else if (merge_B == n) // merge B is done
                A[merge_iter++] = A[merge_A++];
            else
                A[merge_iter++] = (A[merge_A] < B[merge_B] ? A[merge_A++] : B[merge_B++]);
        }
        
        // merge is complete, now the array looks like
        // 4567 | 123
        // so we need to reverse it -- ref. Programming Pearls
        reverse(A, A+m);
        reverse(A+m, A+m+n);
        reverse(A, A+m+n);
    }
};


你可能感兴趣的:(LeetCode)