알고리즘 leetcode|80. 순서 배열 II에서 중복 제거 (녹이 심함)



80. 순서 배열 II에서 중복 항목을 제거합니다.

순서가 지정된 배열이 제공되면 반복되는 요소를 제자리에서nums 삭제 하여 두 번 이상 나타나는 요소가 두 번만 나타나도록 하고 삭제 후 배열의 새 길이를 반환하십시오.

추가 배열 공간을 사용하는 대신 -place 하고 O(1) 추가 공간을 사용하여 수행해야 합니다.

설명 :

반환된 값은 정수인데 출력 응답은 배열인 이유는 무엇입니까?

입력 배열은 " 참조 " 에 의해 전달 됩니다. 즉, 함수의 입력 배열에 대한 수정 사항이 호출자에게 표시됩니다.

내부 동작을 다음과 같이 상상할 수 있습니다.

// nums 是以“引用”方式传递的。也就是说,不对实参做任何拷贝
int len = removeDuplicates(nums);

// 在函数里修改输入数组对于调用者是可见的。
// 根据你的函数返回的长度, 它会打印出数组中 该长度范围内 的所有元素。
for (int i = 0; i < len; i++) {
    print(nums[i]);
}

예시 1:

输入:
	
	nums = [1,1,1,2,2,3]
	
输出:
	
	5, nums = [1,1,2,2,3]
	
解释:
	
	函数应返回新长度 length = 5, 并且原数组的前五个元素被修改为 1, 1, 2, 2, 3。 不需要考虑数组中超出新长度后面的元素。

예시 2:

输入:
	
	nums = [0,0,1,1,1,1,2,3,3]
	
输出:
	
	7, nums = [0,0,1,1,2,3,3]
	
解释:
	
	函数应返回新长度 length = 7, 并且原数组的前五个元素被修改为 0, 0, 1, 1, 2, 3, 3。不需要考虑数组中超出新长度后面的元素。

힌트:

  • 1 <= 숫자.길이 <= 3 * 10 4
  • -10 4 <= 숫자[i] <= 10 4
  • nums는 오름차순으로 정렬됩니다.

분석하다:

  • 이 알고리즘 질문에 직면한 두 번째 마스터는 다시 한 번 깊은 생각에 빠졌습니다.
  • 이 질문은 26. 순서 배열에서 중복 항목 삭제 와 매우 유사합니다 . 차이점은 이 질문에서는 동일한 요소를 최대 한 번 반복할 수 있다는 것입니다.
  • 중복된 항목을 삭제하려면 매핑 테이블 등의 데이터 구조를 사용하여 개수를 계산하는 것이 더 직관적인 아이디어이지만 문제는 일정 수준의 추가 공간만 사용할 수 있으므로 장비에만 의존할 수는 없고 실력에만 의존할 수 있다는 것입니다.
  • 질문에는 아주 중요한 정보, 즉 순서가 있는데, 이는 같은 숫자가 옆에 있을 것이라는 뜻, 즉 현재 첨자와 이전 첨자는 기껏해야 같은 숫자를 가질 수 있다는 뜻이고, 현재 첨자는 이전 첨자와 동일한 것으로 판단되며 표시된 숫자가 동일하면, 즉 nums[i]와 가 nums[i - 2]같으면 해당 요소를 삭제해야 빠르게 중복 여부를 판단할 수 있습니다.
  • 요구 사항은 제자리에서 수정하는 것이므로 새 배열의 길이가 필요하고(반환 값 자체에도 새 배열의 길이가 필요함) 순회 첨자, 즉 이중 포인터도 필요합니다. 한 번만 통과해야 하며 일정한 수준의 추가 공간만 필요하고 매우 효율적입니다.
  • 또한, 배열 요소가 3보다 작은 경우(또는 배열 길이가 2보다 작거나 같은 경우) 확실히 질문의 의미에 부합하므로 별도의 처리 없이 바로 반환될 수 있습니다.

답변:

녹:

impl Solution {
    
    
    pub fn remove_duplicates(nums: &mut Vec<i32>) -> i32 {
    
    
        let n = nums.len();
        if n <= 2 {
    
    
            return n as i32;
        }
        let (mut slow, mut fast) = (2, 2);
        while fast < n {
    
    
            if nums[slow - 2] != nums[fast] {
    
    
                nums[slow] = nums[fast];
                slow += 1;
            }
            fast += 1;
        }
        return slow as i32;
    }
}

가다:

func removeDuplicates(nums []int) int {
    
    
	n := len(nums)
	if n <= 2 {
    
    
		return n
	}
	slow, fast := 2, 2
	for fast < n {
    
    
		if nums[slow-2] != nums[fast] {
    
    
			nums[slow] = nums[fast]
			slow++
		}
		fast++
	}
	return slow
}

C++:

class Solution {
    
    
public:
    int removeDuplicates(vector<int>& nums) {
    
    
        const int n = nums.size();
        if (n <= 2) {
    
    
            return n;
        }
        int slow = 2, fast = 2;
        while (fast < n) {
    
    
            if (nums[slow - 2] != nums[fast]) {
    
    
                nums[slow] = nums[fast];
                ++slow;
            }
            ++fast;
        }
        return slow;
    }
};

파이썬:

class Solution:
    def removeDuplicates(self, nums: List[int]) -> int:
        slow = 2
        for fast in range(2, len(nums)):
            if nums[fast] != nums[slow - 2]:
                nums[slow] = nums[fast]
                slow += 1
        return slow


자바:

class Solution {
    
    
    public int removeDuplicates(int[] nums) {
    
    
        final int n = nums.length;
        if (n <= 2) {
    
    
            return n;
        }
        int slow = 2, fast = 2;
        while (fast < n) {
    
    
            if (nums[slow - 2] != nums[fast]) {
    
    
                nums[slow] = nums[fast];
                ++slow;
            }
            ++fast;
        }
        return slow;
    }
}

글 읽어주셔서 정말 감사합니다~
[좋아요] [모으기] [댓글] 3연속 오신 것을 환영합니다~
포기하는 것도 어렵지 않지만, 끈기가 멋있을 것 같아요~
우리 모두 조금이라도 발전할 수 있었으면 좋겠습니다 매일~
이 글은 흰모자 부사령관이 쓴 글입니다 : https://le-yi.blog.csdn.net/Original blog~


추천

출처blog.csdn.net/leyi520/article/details/132802590