4/150:寻找两个正序数组的中位数⭐ 5/150最长回文子串 6/100正则表达式匹配

题目:4/150寻找两个正序数组的中位数

给定两个大小分别为 m 和 n 的正序(从小到大)数组 nums1 和 nums2。请你找出并返回这两个正序数组的 中位数 。

算法的时间复杂度应该为 O(log (m+n)) 。
在这里插入图片描述

题解1:暴力

暴力思路简介,两个有序数组合并成一个,分奇偶得到中位数,需要注意的是,结果需要为double,且要除以2.0,注意边界问题
原来思路是一边合并一边比较是否已经merge到中位数位置,但实际当其中一个数组遍历完成后,需要复制另一个数组的剩余元素

class Solution {
public:
    double findMedianSortedArrays(vector<int>& nums1, vector<int>& nums2) {
        //分别依次遍历组成一个合并数组,在合并数组到第(m+n)/2的时候 可以取值 注意分奇偶
        int m = nums1.size();
        int n = nums2.size();
        vector<int> merge(m+n, 0);
        int i = 0, j = 0;
        int k = 0;
        double res = 0;
        while(i<m && j<n)
        {
            merge[k++] = nums1[i]<nums2[j]?nums1[i++]:nums2[j++];
        }

        while (i < m) {
            merge[k++] = (nums1[i++]); 
        }
        while (j < n) {
             merge[k++] = (nums2[j++]);
        }

        if((m+n)%2 == 0)
        {
            res =(merge[(m+n)/2] + merge[(m+n)/2 -1]) /2.0;
        }else
        {
             res =(merge[(m+n)/2]);
        }

        return res;

    }
};

看题解1:二分查找⭐ 需要再理解

二分查找的原理是检查中间元素和目标元素的大小,通过比较将查找范围缩小一半,过程在逻辑上重复,直到找到目标元素或者范围缩小到无法被继续划分 —— 如果中间元素大于目标元素,在前半部分继续二分;如果中间元素小于目标元素,在后半部分二分。时间复杂度为o(logn)
看题解整体思路能理解,但自己实现还是不会,需要重复再理解(但感觉也不好在一个题上纠结太久 先收藏⭐⭐⭐
还是需要注意除以2.0的问题
核心部分代码:

            int newIndex1 = min(index1 + k/2 -1, m-1);
            int newIndex2 = min(index2 + k/2-1, n-1);
            int pivot1 = nums1[newIndex1];
            int pivot2 = nums2[newIndex2];
            if(pivot1 <= pivot2)
            {
                k -= newIndex1 - index1+1;
                index1 = newIndex1+1;
            }else{
                k -= newIndex2 - index2+1;
                index2 = newIndex2+1;
            }

要找到第 k (k>1) 小的元素,那么就取 pivot1 = nums1[k/2-1] 和 pivot2 = nums2[k/2-1] 进行比较。nums1 中小于等于 pivot1 的元素有 nums1[0 … k/2-2] 共计 k/2-1 个,nums2 中小于等于 pivot2 的元素有 nums2[0 … k/2-2] 共计 k/2-1 个。取 pivot = min(pivot1, pivot2),两个数组中小于等于 pivot 的元素共计不会超过 (k/2-1) + (k/2-1) <= k-2 个。这样 pivot 本身最大也只能是第 k-1 小的元素。

如果 pivot = pivot1,那么 nums1[0 … k/2-1] 都不可能是第 k 小的元素。把这些元素全部 “删除”,剩下的作为新的 nums1 数组。同理pivot2, 删除一定小的元素,更新数组更新k值,即删除确定小的元素更新k值更新的比较区域

实现完整代码需要考虑边界的情况,以及写的时候容易将下标i和第i小的概念混淆,第i小要+1,

class Solution {
public:
    int getKthElement(const vector<int>& nums1, const vector<int>& nums2, int k) {
        int m = nums1.size();
        int n = nums2.size();
        int index1 = 0, index2 = 0;

        while (true) {
            // 边界情况
            if (index1 == m) {
                return nums2[index2 + k - 1];
            }
            if (index2 == n) {
                return nums1[index1 + k - 1];
            }
            if (k == 1) {
                return min(nums1[index1], nums2[index2]);
            }

            // 正常情况
            int newIndex1 = min(index1 + k / 2 - 1, m - 1);
            int newIndex2 = min(index2 + k / 2 - 1, n - 1);
            int pivot1 = nums1[newIndex1];
            int pivot2 = nums2[newIndex2];
            if (pivot1 <= pivot2) {
                k -= newIndex1 - index1 + 1;
                index1 = newIndex1 + 1;
            }
            else {
                k -= newIndex2 - index2 + 1;
                index2 = newIndex2 + 1;
            }
        }
    }

    double findMedianSortedArrays(vector<int>& nums1, vector<int>& nums2) {
        int totalLength = nums1.size() + nums2.size();
        if (totalLength % 2 == 1) {
            return getKthElement(nums1, nums2, (totalLength + 1) / 2);
        }
        else {
            return (getKthElement(nums1, nums2, totalLength / 2) + getKthElement(nums1, nums2, totalLength / 2 + 1)) / 2.0;
        }
    }
};

题目:5/150:最长回文子串

给你一个字符串 s,找到 s 中最长的回文子串。
如果字符串的反序与原始字符串相同,则该字符串称为回文字符串。
在这里插入图片描述

题解:用中心扩展法较好理解

每一个字符串作为中心,向两边扩展,保留最长长度与起始位置

class Solution {
public:
    string longestPalindrome(string s) {
        //以每个字符作为字符的中心,向左右进行扩展 中心可以为一个数 也可能是两个数作为中心
        int n = s.size();
        if(n<2)
        {return s;}
        int maxLen = 1;
        int center = 0;//维护一个起始点
        for(int i = 0; i<n; i++)
        {
            //对每个字符都进行判断,以此作为中心的最长回环数
            int len1 = expandAroundCenter(s, i, i);//单个字符为中心
            int len2 = expandAroundCenter(s, i, i+1);//两个字符为中心
            int len = max(len1, len2);
            if(maxLen<len)
            {
                maxLen = len;
                center = i;//维护当前以i为中心 长度为len的回文字符串
            }
        }
        //最长的回文即是 以i为中心 长度为maxLen的
        int start = center-(maxLen-1)/2;

        return s.substr(start, maxLen); 

    }

    int expandAroundCenter(string s, int left, int right)
        {
            int l = left, r = right;
            while(l>=0 && r<s.size() && s[l] == s[r])
            {
                l--;
                r++;
            }
            return r-l-1;//退出的时候不是回文了
        }
};
//试试暴力解法 中心扩展法

题目:6/100正则表达式匹配

给你一个字符串 s 和一个字符规律 p,请你来实现一个支持 ‘.’ 和 ‘*’ 的正则表达式匹配。

'.' 匹配任意单个字符
'*' 匹配零个或多个前面的那一个元素

所谓匹配,是要涵盖 整个 字符串 s的,而不是部分字符串。
在这里插入图片描述

题解:涉及到动态规划

选择从右往左扫描,s、p是否匹配,取决于最右端是否匹配,剩余的子串是否匹配,只是最右端有可能是特殊符号,需要分情况讨论
要不先PASS一下 12.6更新 对动态规划稍有理解

class Solution {
public:
      bool dp(string s,int i, string p, int j) {
      map<string, bool> memo;

        //1先写base case
        int m = s.size(), n = p.size();
        if(j == n)
        {
            return i == m;
        }
        if(i == m)
        {
            //是否能匹配空字符串
            //一定得是偶数个 且是以x*y*z*的形势
            if((n-j)%2 == 1)
            {
                return false;
            }
            for(; j+1<n; j+=2)
            {
                if(p[j+1] != '*')
                {
                    return false;
                }
            }
            return true;
        }
        //2 写状态转移方程 
        //用备忘录
        string key = to_string(i)+"_"+to_string(j);
        if(memo.count(key)) return memo[key];
        bool res = false;

        if(s[i] == p[j] || p[j] == '.')
        {
            if(j<n-1 && p[j+1] == '*')
            {
                res = dp(s, i, p, j+2)|| dp(s, i+1, p,j);

            }else{
                res = dp(s, i+1, p, j+1);
            }
            
        }else{
            if(j<n-1 && p[j+1] == '*')
            {
                res = dp(s, i, p, j+2);
            }else{
                res = false;
            }
        }

        memo[key] = res;
        return res;

    }

    bool isMatch(string s, string p) {
        int i = 0, j=0;
        return dp(s, i, p, j);
        
    }
};

但此种写法是超出时间限制的,不再耽搁,继续往下

  • 7
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值