leetcode解题思路分析(四)22-28题

2024-09-05 05:18
文章标签 leetcode 22 28 解题 思路 分析

本文主要是介绍leetcode解题思路分析(四)22-28题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  1. 括号生成
    给出 n 代表生成括号的对数,请你写出一个函数,使其能够生成所有可能的并且有效的括号组合。

很容易想到采用回溯法解决该题,通过画出树分析递归规律可得如下代码

class Solution {
public:void backtrace(int left, int right, int n, string& s, vector<string>& res) {if (left == n && right == n) {res.push_back(s);return;}if (left < n) {s += "(";backtrace(left + 1, right, n, s, res);s.pop_back();}if (right < left) {s += ")";backtrace(left, right + 1, n, s, res);s.pop_back();}}vector<string> generateParenthesis(int n) {vector<string> res;string s;backtrace(0, 0, n, s, res);return res;}
};
  1. 合并k个排序链表
    合并 k 个排序链表,返回合并后的排序链表。请分析和描述算法的复杂度。

本题在做过合并2个链表之后就变得很简单:采用二分法逐个合并直至剩下一个即可

/*** Definition for singly-linked list.* struct ListNode {*     int val;*     ListNode *next;*     ListNode(int x) : val(x), next(NULL) {}* };*/
class Solution {
public:ListNode* mergeKLists(vector<ListNode*>& lists) {int size = lists.size();if (size == 0) {return nullptr;}if (size == 1) {return lists[0];}while (size > 1){               for (int i = 0; i < size / 2; i++){lists[i] = mergeTwoLists(lists[i], lists[i + size / 2]);}if (size % 2){lists[size / 2] = lists[size - 1];size = size / 2 + 1;               }else{size = size / 2;}}return lists[0];}ListNode* mergeTwoLists(ListNode* l1, ListNode* l2) {ListNode *preHead, *ptr;preHead = new ListNode(-1);ptr = preHead;while (l1 != NULL && l2 != NULL){if (l1->val <= l2->val){         ptr->next = l1;    l1 = l1->next;                                 }else{ptr->next = l2;l2 = l2->next;}ptr = ptr->next;}ptr->next = (l1 == NULL? l2 : l1);return preHead->next;}
};
  1. 两两交换链表中的节点
    给定一个链表,两两交换其中相邻的节点,并返回交换后的链表。
    你不能只是单纯的改变节点内部的值,而是需要实际的进行节点交换。

非常简单的链表操作题

/*** Definition for singly-linked list.* struct ListNode {*     int val;*     ListNode *next;*     ListNode(int x) : val(x), next(NULL) {}* };*/
class Solution {
public:ListNode* swapPairs(ListNode* head) {ListNode *preHead = new ListNode(-1);ListNode *tmp, *ahead;preHead->next = head;ahead = preHead;while (head != NULL && head->next != NULL){tmp = head->next;ahead->next = tmp;head->next = tmp->next;tmp->next = head;ahead = head;            head = head->next;}return preHead->next;}
};
  1. K 个一组翻转链表
    给你一个链表,每 k 个节点一组进行翻转,请你返回翻转后的链表。
    k 是一个正整数,它的值小于或等于链表的长度。
    如果节点总数不是 k 的整数倍,那么请将最后剩余的节点保持原有顺序。

本题是24题的提高,为了翻转K个一组,则需要遍历k个,让每一个指针指向前一个节点,然后最前面的节点指向下一组的开始。因此考虑迭代解决

/*** Definition for singly-linked list.* struct ListNode {*     int val;*     ListNode *next;*     ListNode(int x) : val(x), next(NULL) {}* };*/
class Solution {
public:ListNode* reverseKGroup(ListNode* head, int k) {int d = 0;auto node = head;while (node != NULL) {if (++d >= k) break;node = node->next;}if (d < k) return head;ListNode* prev = NULL;ListNode* curr = head;for (int i = 0; i < k; ++i) {auto node = curr->next;curr->next = prev;prev = curr;curr = node;}head->next = reverseKGroup(curr, k);return prev;}
};
  1. 删除排序数组中的重复项
    给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度。
    不要使用额外的数组空间,你必须在原地修改输入数组并在使用 O(1) 额外空间的条件下完成。

对已经排序好的数组,我们采取双指针法,检测到不同的元素再和相同元素第二位进行互换,然后指针前进即可

class Solution {
public:int removeDuplicates(vector<int>& nums) {if (nums.size() == 0) return 0;int i = 0;for (int j = 1; j < nums.size(); j++) {if (nums[j] != nums[i]) {i++;nums[i] = nums[j];}}return i + 1;}
};
  1. 移除元素
    给定一个数组 nums 和一个值 val,你需要原地移除所有数值等于 val 的元素,返回移除后数组的新长度。不要使用额外的数组空间,你必须在原地修改输入数组并在使用 O(1) 额外空间的条件下完成。
    元素的顺序可以改变。你不需要考虑数组中超出新长度后面的元素。

本题可以采取和上题类似的策略:对每个重复的元素则赋值给后面不同的元素。但是注意这里需要保证不会重复赋值,因此比较可行的方法是进行节点的交换

class Solution {
public:int removeElement(vector<int>& nums, int val) {int size = nums.size();if (size == 0)return 0;int ptr = 0;int end = 0;for (int i = 0; i < size; i++){if (nums[i] == val){if (end < i)end = i;if (end == size)return ptr;while (nums[end] == val){end++;if (end == size){return ptr;}}int tmp = nums[ptr];nums[ptr] = nums[end];nums[end] = tmp;end++;ptr++;}else{ptr++;}}return ptr;}
};

但是上述做法的代码看起来颇为繁琐,其实可以更进一步的简化思想:我们只要每次把不同于val的值赋值在前列就可以完成了

class Solution {
public:int removeElement(vector<int>& nums, int val) {if (nums.size() == 0) return 0;int i = 0;for (int j = 0; j < nums.size(); j++) {if (nums[j] != val) {nums[i] = nums[j];                i++;}}return i;}
};

还有没有办法优化性能呢?是有的。上述方法存在一个普遍问题:对于检索过的val的值最后还会检查一遍。但是其实是不需要检查的:我们可以将重复的值替换为末尾的值, 并且不再检查它

class Solution {
public:int removeElement(vector<int>& nums, int val) {int i = 0;int n = nums.size();while (i < n) {if (nums[i] == val) {nums[i] = nums[n - 1];// reduce array size by onen--;} else {i++;}}return n;}
};
  1. 实现 strStr()
    给定一个 haystack 字符串和一个 needle 字符串,在 haystack 字符串中找出 needle 字符串出现的第一个位置 (从0开始)。如果不存在,则返回 -1。

本题采用暴力检索的方式:找到一个相同元素则开始继续后续检查,如果检查完不一样则继续向后检索

class Solution {
public:int strStr(string haystack, string needle) {int i = 0, j = 0;while(i < haystack.size() && j < needle.size()){if(haystack[i] == needle[j]) {i++;j++;}else {i = i - j +1 ;j = 0;}}if(j == needle.size()) return i - j;return -1;}
};

这篇关于leetcode解题思路分析(四)22-28题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1138048

相关文章

哈希leetcode-1

目录 1前言 2.例题  2.1两数之和 2.2判断是否互为字符重排 2.3存在重复元素1 2.4存在重复元素2 2.5字母异位词分组 1前言 哈希表主要是适合于快速查找某个元素(O(1)) 当我们要频繁的查找某个元素,第一哈希表O(1),第二,二分O(log n) 一般可以分为语言自带的容器哈希和用数组模拟的简易哈希。 最简单的比如数组模拟字符存储,只要开26个c

性能分析之MySQL索引实战案例

文章目录 一、前言二、准备三、MySQL索引优化四、MySQL 索引知识回顾五、总结 一、前言 在上一讲性能工具之 JProfiler 简单登录案例分析实战中已经发现SQL没有建立索引问题,本文将一起从代码层去分析为什么没有建立索引? 开源ERP项目地址:https://gitee.com/jishenghua/JSH_ERP 二、准备 打开IDEA找到登录请求资源路径位置

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验

SWAP作物生长模型安装教程、数据制备、敏感性分析、气候变化影响、R模型敏感性分析与贝叶斯优化、Fortran源代码分析、气候数据降尺度与变化影响分析

查看原文>>>全流程SWAP农业模型数据制备、敏感性分析及气候变化影响实践技术应用 SWAP模型是由荷兰瓦赫宁根大学开发的先进农作物模型,它综合考虑了土壤-水分-大气以及植被间的相互作用;是一种描述作物生长过程的一种机理性作物生长模型。它不但运用Richard方程,使其能够精确的模拟土壤中水分的运动,而且耦合了WOFOST作物模型使作物的生长描述更为科学。 本文让更多的科研人员和农业工作者

MOLE 2.5 分析分子通道和孔隙

软件介绍 生物大分子通道和孔隙在生物学中发挥着重要作用,例如在分子识别和酶底物特异性方面。 我们介绍了一种名为 MOLE 2.5 的高级软件工具,该工具旨在分析分子通道和孔隙。 与其他可用软件工具的基准测试表明,MOLE 2.5 相比更快、更强大、功能更丰富。作为一项新功能,MOLE 2.5 可以估算已识别通道的物理化学性质。 软件下载 https://pan.quark.cn/s/57

leetcode-24Swap Nodes in Pairs

带头结点。 /*** Definition for singly-linked list.* public class ListNode {* int val;* ListNode next;* ListNode(int x) { val = x; }* }*/public class Solution {public ListNode swapPairs(L

leetcode-23Merge k Sorted Lists

带头结点。 /*** Definition for singly-linked list.* public class ListNode {* int val;* ListNode next;* ListNode(int x) { val = x; }* }*/public class Solution {public ListNode mergeKLists

衡石分析平台使用手册-单机安装及启动

单机安装及启动​ 本文讲述如何在单机环境下进行 HENGSHI SENSE 安装的操作过程。 在安装前请确认网络环境,如果是隔离环境,无法连接互联网时,请先按照 离线环境安装依赖的指导进行依赖包的安装,然后按照本文的指导继续操作。如果网络环境可以连接互联网,请直接按照本文的指导进行安装。 准备工作​ 请参考安装环境文档准备安装环境。 配置用户与安装目录。 在操作前请检查您是否有 sud

线性因子模型 - 独立分量分析(ICA)篇

序言 线性因子模型是数据分析与机器学习中的一类重要模型,它们通过引入潜变量( latent variables \text{latent variables} latent variables)来更好地表征数据。其中,独立分量分析( ICA \text{ICA} ICA)作为线性因子模型的一种,以其独特的视角和广泛的应用领域而备受关注。 ICA \text{ICA} ICA旨在将观察到的复杂信号

C++ | Leetcode C++题解之第393题UTF-8编码验证

题目: 题解: class Solution {public:static const int MASK1 = 1 << 7;static const int MASK2 = (1 << 7) + (1 << 6);bool isValid(int num) {return (num & MASK2) == MASK1;}int getBytes(int num) {if ((num &