Leetcode刷题记录——排序

归并排序:

  • 通过递归不断地将数组分为两半,直至分割后数组的左右边界相等,即数组中只有一个元素。进而将左右两半部分排序后合并,return后重复此操作,直至排序完成。

在这里插入图片描述

class Solution {
public:
    vector<int> sortArray(vector<int>& nums) {
        mergesort(nums,0,nums.size()-1);
        return nums;
    }
    void merge(vector<int>& nums,int l1,int r1,int l2,int r2)
    {
        int i=l1,j=l2,n=r1-l1+1+r2-l2+1,k=0;
        vector<int>temp(n);
        while(i<=r1&&j<=r2)
        {
            if(nums[i]<nums[j]) temp[k++]=nums[i++];
            else temp[k++]=nums[j++];
        }
        while(i<=r1) temp[k++]=nums[i++];
        while(j<=r2) temp[k++]=nums[j++];
        for(int i=0;i<n;i++)
        {
            nums[l1+i]=temp[i];
        }
    }
    void mergesort(vector<int>& nums,int left,int right)
    {
        if(left==right) return;
        int mid=left+(right-left)/2;
        mergesort(nums,left,mid);
        mergesort(nums,mid+1,right);
        merge(nums,left,mid,mid+1,right);
    }
};

排序时将两半部分中的最小元素依次放入中间数组中,完成排序后替换原数组

合并K个升序链表

  • 只是把数组换成了链表,思路相同,将K个链表排序问题转化为合并两个有序链表。

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     ListNode *next;
 *     ListNode() : val(0), next(nullptr) {}
 *     ListNode(int x) : val(x), next(nullptr) {}
 *     ListNode(int x, ListNode *next) : val(x), next(next) {}
 * };
 */
class Solution {
public:
    ListNode* mergeKLists(vector<ListNode*>& lists) {
        if(lists.size()==0) return nullptr;
        return mergelists(lists,0,lists.size()-1);
    }
    ListNode* merge(ListNode* l1,ListNode* l2)
    {
        if(!l1) return l2;
        if(!l2) return l1;
        if(l1->val<l2->val)
        {
            l1->next=merge(l1->next,l2);
            return l1;
        }
        else
        {
            l2->next=merge(l1,l2->next);
            return l2;
        }
    }
    ListNode* mergelists(vector<ListNode*>& lists,int left,int right)
    {
        if(left==right) return lists[left];
        int mid=left+(right-left)/2;
        ListNode* l1=mergelists(lists,left,mid);
        ListNode* l2=mergelists(lists,mid+1,right);
        return merge(l1,l2);
    }
};

快速排序:

  • 在数组中指定一个基准,先从右向左搜索,元素比基准值大则继续向左搜索,比基准值小则将该元素放到左边;然后从左向右搜索,元素比基准值小则继续向右搜索,比基准值大则将该元素放到右边;重复以上操作直至左右指针相遇,则该位置为基准位置,将基准值赋给当前位置。接着对基准左右两部分继续进行上述操作,完成快速排序。

在这里插入图片描述

class Solution {
public:
    vector<int> sortArray(vector<int>& nums) {
        quick_sort(nums,0,nums.size()-1);
        return nums;
    }
    void quick_sort(vector<int>& nums,int left,int right)
    {
        if(left>=right) return;
        int i=left,j=right;
        int pivot=rand()%(right-left+1)+left;
        swap(nums[left],nums[pivot]);
        pivot=nums[left];
        while(left<right)
        {
            while(left<right&&nums[right]>=pivot) right--;
            nums[left]=nums[right];
            while(left<right&&nums[left]<=pivot) left++;
            nums[right]=nums[left];
        }
        nums[left]=pivot;
        quick_sort(nums,i,left-1);
        quick_sort(nums,left+1,j);
    }
};

基准值随机在数组中选择以降低时间复杂度

数组中的第K个最大元素:

  • 在快速排序时判断所要查找的元素在基准的左侧还是右侧,只对该侧递归排序即可。

class Solution {
public:
    int findKthLargest(vector<int>& nums, int k) {
        int left=0,right=nums.size()-1;
        quick_sort(nums,left,right,k);
        return nums[right-k+1];
    }
    void quick_sort(vector<int>& nums,int left,int right,int k)
    {
        if(left>=right) return;
        int i=left,j=right;
        int pivot=rand()%(right-left+1)+left;
        swap(nums[left],nums[pivot]);
        pivot=nums[left];
        while(left<right)
        {
            while(left<right&&nums[right]>=pivot) right--;
            nums[left]=nums[right];
            while(left<right&&nums[left]<=pivot) left++;
            nums[right]=nums[left];
        }
        nums[left]=pivot;
        if(left>nums.size()-k) quick_sort(nums,i,left-1,k);
        else if(left<nums.size()-k) quick_sort(nums,left+1,j,k);
    }
};

若基准位置大于第K大元素的位置,则该元素在基准的左边,递归排序左边数组即可;基准位置小于第K大元素的位置同理可得只递归排序右边数组即可;若基准位置等于第K大元素的位置,则基准值即为所需元素,直接返回。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值