草庐IT

菜鸟刷题Day6

别动我的饭 2023-04-24 原文

⭐作者:别动我的饭
⭐专栏:菜鸟刷题
⭐标语:悟已往之不谏,知来者之可追

一.链表内指定区间反转:链表内指定区间反转_牛客题霸_牛客网 (nowcoder.com)

描述

将一个节点数为 size 链表 m 位置到 n 位置之间的区间反转,要求时间复杂度 O(n),空间复杂度O(1)
例如:给出的链表为1→2→3→4→5→NULL m=2,n=4,
返回 1→4→3→2→5→NULL


解题思路

如果只有一个节点或者m==n,那就直接返回head,因为不用反转。如果有多个节点,那就需要建立一个哨兵位标记住头节点,后续需要移动头节点。然后找到反转位置的前驱节点,再将反转位置赋值给head,将m到n之间的节点取下来头插就可以达到反转链表的效果。(head会随着不断头插向后挪动,需要用一个next指针记住head的下一个节点),此外要注意好区间范围的控制。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ZB1Dlp1l-1679744040656)(C:\Users\羽北冥\AppData\Roaming\Typora\typora-user-images\image-20230325131417710.png)]

struct ListNode* reverseBetween(struct ListNode* head, int m, int n ) {
    // write code here
    //如果只有一个节点或者没有节点就直接返回
    if(head==NULL||head->next==NULL||m==n)
        return head;

    //如果有多个节点,就建立一个哨兵位节点,找到反转位置拿下来头插
    struct ListNode*tmp=(struct ListNode*)malloc(sizeof(struct ListNode));
    tmp->next=head;
    struct ListNode*prev=tmp;
    for(int i=1;i<m;i++)
    {
        prev=prev->next;
    }

    //将需要反转链表的头部搞给head

   head=prev->next;

  
    for(int j=m;j<n;j++)
    {
        //将节点取下来头插
       struct ListNode*next=head->next;

      head->next=next->next;
      next->next=prev->next;
      prev->next=next;


    }

    head=tmp->next;
    free(tmp);
return head;
    
}

二.从链表中删去总和值为零的连续节点:1171. 从链表中删去总和值为零的连续节点 - 力扣(LeetCode)

描述

给你一个链表的头节点 head,请你编写代码,反复删去链表中由 总和 值为 0 的连续节点组成的序列,直到不存在这样的序列为止。

删除完毕后,请你返回最终结果链表的头节点。

你可以返回任何满足题目要求的答案(注意,下面示例中的所有序列,都是对 ListNode 对象序列化的表示。)

示例 1:

输入:head = [1,2,-3,3,1]
输出:[3,1]
提示:答案 [1,2,1] 也是正确的。

解题思路

首先要明确一点:这个数组不是绝对有序的(因为我当时考虑了哈哈)。

核心就是只要前面的数值相加结果等于零,那么前面所有的节点都可以舍弃。但是你直接从零位置开始累加的话不一定会得到前缀和能为零,所以这里可以考虑使用嵌套循环,也就是说如果从第一个位置累加不能为零,那么就从第二个位置再累加一次。直接走完所有的节点都不能累加为零,就说明所有的数都是正数。

struct ListNode* removeZeroSumSublists(struct ListNode* head){

	struct ListNode*phead=(struct ListNode*)malloc(sizeof(struct ListNode));
    phead->next=head;
	//用双指针嵌套循环
    struct ListNode*cur=phead;
    struct ListNode*curr=cur->next;
    int sum=0;

    while(cur)
    {
        sum=0;//这里必须在更新一下保证第二次循环不被影响
       
       	curr=cur->next;//这里怎么能不更新curr
        while(curr)
        {
            sum+=curr->val;

            if(sum==0)
            {
                cur->next=curr->next;//不释放节点,直接更新cur位置       
            }
            curr=curr->next;
        }
        cur=cur->next;
    
    }
    return phead->next;
}

三.链表求和:面试题 02.05. 链表求和 - 力扣(LeetCode)

描述

给定两个用链表表示的整数,每个节点包含一个数位。这些数位是反向存放的,也就是个位排在链表首部。编写函数对这两个整数求和,并用链表形式返回结果。

示例:

输入:(7 -> 1 -> 6) + (5 -> 9 -> 2),即617 + 295
输出:2 -> 1 -> 9,即912

解题思路

我最开始想着设定两个变量然后分别循环遍历两个链表,将两个链表中得到的值存储给变量n1,n2,最后两者相加得到sum,再对sum做文章。但是这样要走两次循环,后面我有思考了一下发现可以直接相加。

除n1和n2以外,在设定一个carry变量用来保存进位(对于加法来说如果这两个数相加大于十,则要往前进一位,再将这一位加给十位相加得到的结果),可以直接将这三个变量相加的结果存放到链表中。需要注意的是链表最后的节点相加可能超过十,所以出了循环以后要对carry判断一下,如果carry不为零,则还要开一个节点存放carry

此外设置一个head和一个tail指针,在第一次插入的时候初始化head,后续只动tail指针,最后用head做返回值。

struct ListNode* addTwoNumbers(struct ListNode* l1, struct ListNode* l2)
{
    int n1,n2=0;
    int carry=0,sum=0;
    struct ListNode*head=NULL,*tail=NULL;
    
    while(l1||l2)//如果两个都是空就没有意义了
    {
        n1=l1?l1->val:0;//如果l1不是空,就返回它的值否则返回0
        n2=l2?l2->val:0;
        
        sum=n1+n2+carry;
        
        if(head==NULL)
        {
            head=tail=(struct ListNode*)malloc(sizeof(struct ListNode));
            tail->val=sum%10;//sum的值可能超过十,我们只需要存sum的个位就行
            tail->next=NULL;
        }
        else
        {
            //不是第一次插入,只动尾节点
            tail->next=(struct ListNode*)malloc(sizeof(struct ListNode));
            tail->next->val=sum%10;
            tail=tail->next;
            tail->next=NULL;
        }
        
        carry=sum/10;//更新carry
        
        //如果了l1和l2不为空就继续往下走
        if(l1)
            l1=l1->next;
        if(l2)
            l2=l2->next;
    }
    
    if(carry>0)
    {
         tail->next=(struct ListNode*)malloc(sizeof(struct ListNode));
         tail->next->val=carry;
         tail->next->next=NULL;
    }
return head;
}

四.括号的最大嵌套深度:1614. 括号的最大嵌套深度 - 力扣(LeetCode)

描述

如果字符串满足以下条件之一,则可以称之为 有效括号字符串(valid parentheses string,可以简写为 VPS):

字符串是一个空字符串 “”,或者是一个不为 “(” 或 “)” 的单字符。
字符串可以写为 AB(A 与 B 字符串连接),其中 A 和 B 都是 有效括号字符串 。
字符串可以写为 (A),其中 A 是一个 有效括号字符串 。
类似地,可以定义任何有效括号字符串 S 的 嵌套深度 depth(S):

depth(“”) = 0
depth© = 0,其中 C 是单个字符的字符串,且该字符不是 “(” 或者 “)”
depth(A + B) = max(depth(A), depth(B)),其中 A 和 B 都是 有效括号字符串
depth(“(” + A + “)”) = 1 + depth(A),其中 A 是一个 有效括号字符串
例如:“”、“()()”、“()(()())” 都是 有效括号字符串(嵌套深度分别为 0、1、2),而 “)(” 、“(()” 都不是 有效括号字符串 。

给你一个 有效括号字符串 s,返回该字符串的 s 嵌套深度 。

示例 1:

输入:s = "(1+(2*3)+((8)/4))+1"
输出:3
解释:数字 8 在嵌套的 3 层括号中。

解题思路

这类题目用栈会比较好处理,但是C语言如果要使用栈的话,又要徒手写一个,这样太耗费时间了。所以这里可以采用一个数组通过下标的控制来达到模拟栈的效果。

如果是左括号就将其入栈,如果是右括号就将左括号出栈。也就是说如果是 “( ”则size–,如果是 “ )”则size++,以此来表示栈内容量的变化。在不断入栈出栈的过程中,size的最大值就是括号的最大嵌套深度,因为s是一个有效的括号字符串。

#define MAX(a,b) ((a)>(b)?(a):(b))

int maxDepth(char * s)
{
	int len=strlen(s);
    int size=0;
    for(int i=0;i<len;i++)
    {
        if(s[i]=="(")
            size++;
        ans=MAX(size,ans);
        if(s[i]==")")
            size--;
	}
    return ans;
}

其实就是拥有左括号数的最大值


最近铃芽之旅上线了不知道各位有没有去看(又有多少老铁是一个人去看的,斜眼笑),昨天我可是特意给你们放了假(好吧,其实是昨天课太多了,而我又被某些题目卡了三个小时所以才没来得及,菜鸡流泪)。

人们总是高估短期努力能够带来的提升,却忽略长期坚持能带来的改变。今天是第六天了,你还有坚持吗?

有关菜鸟刷题Day6的更多相关文章

  1. ruby-on-rails - rails : Find tasks that were created on a certain day? - 2

    我有一个任务列表(名称、starts_at),我试图在每日View中显示它们(就像iCal)。deftodays_tasks(day)Task.find(:all,:conditions=>["starts_atbetween?and?",day.beginning,day.ending]end我不知道如何将Time.now(例如“2009-04-1210:00:00”)动态转换为一天的开始(和结束),以便进行比较。 最佳答案 deftodays_tasks(now=Time.now)Task.find(:all,:conditio

  2. ruby-on-rails - 为什么 Rails 菜鸟不应该使用 Gem Devise? - 2

    按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visitthehelpcenter指导。关闭9年前。我是ruby​​onrails菜鸟。相比之下,我的HTMLCSSjavascript和jQuery相当不错。最近我使用MichaelHartl的教程进入了RubyonRails:http://ruby.railstutorial.org/ruby-on-rails-tutorial-book.但是,唉,我正在尝试构建自己的项目并使用gemdevise作为进

  3. 什么是0day漏洞?如何预防0day攻击? - 2

    什么是0day漏洞?0day漏洞,是指已经被发现,但是还未被公开,同时官方还没有相关补丁的漏洞;通俗的讲,就是除了黑客,没人知道他的存在,其往往具有很大的突发性、破坏性、致命性。0day漏洞之所以称为0day,正是因为其补丁永远晚于攻击。所以攻击者利用0day漏洞攻击的成功率极高,往往可以达到目的并全身而退,而防守方却一无所知,只有在漏洞公布之后,才后知后觉,却为时已晚。“后知后觉、反应迟钝”就是当前安全防护面对0day攻击的真实写照!为了方便大家理解,中科三方为大家梳理当前安全防护模式下,一个漏洞从发现到解决的三个时间节点:T0:此时漏洞即0day漏洞,是已经被发现,还未被公开,官方还没有相

  4. ruby - Rails 比较 date.end_of_day.to_datetime 和 date.to_datetime.end_of_day 返回的日期对象值时返回 false - 2

    ruby1.9.3dev(2011-09-23修订版33323)[i686-linux]轨道3.0.20最近为什么在与DateTimeonRails相关的RSpecs项目上工作我发现在给定日期以下语句发出的值date.end_of_day.to_datetime和date.to_datetime.end_of_day虽然它们表示相同的日期时间,但比较时返回false。为了确认这一点,我打开了Rails控制台并尝试了以下操作1.9.3dev:053>monday=Time.now.monday=>2013-02-2500:00:00+05301.9.3dev:054>monday.cla

  5. Ruby,从 Date.day_fraction_to_time 获取小时、秒和时间 - 2

    我找到了这个方法here.start=DateTime.nowsleep15stop=DateTime.now#minutesputs((stop-start)*24*60).to_ihours,minutes,seconds,frac=Date.day_fraction_to_time(stop-start)我有以下错误:`':privatemethod`day_fraction_to_time'calledforDate:Class(NoMethodError)我检查了/usr/lib/ruby/1.9.1/date.rb并找到了它:defday_fraction_to_time(

  6. Ruby strftime : Day without leading zero, %e 不工作 - 2

    我正在尝试用没有前导零的日期来格式化日期使用%d它工作正常,但前导零date_time.strftime("%d/%m/%y")result:04/01/11我搜索了一下,发现我应该使用%e而不是%d,但是执行以下操作会得到一个空字符串。date_time.strftime("%e/%m/%y")result:这跟Ruby的版本有关系吗?我在Windows机器上使用v1.8.7。更重要的是,是否有另一种方法可以在没有前导零的情况下完成一天(比gsub更方便)? 最佳答案 如果你想删除月份或日期的前导零,只需在格式前添加一个减号,如下

  7. ruby-on-rails - Rails : Date. 今天 - 1.day - 2

    使用rails控制台,我只是被这个咬住了:假设今天是12月11日。Date.today-1.day#December10(nospaces)Date.today-1.day#December10(aspaceonbothsidesoftheminussign)Date.today-1.day#December11whaaaat?Date.today-5.days#Stilldecember11!有人能解释一下这是怎么回事吗?我有点担心这在代码中很容易被遗漏。关于如何对此进行编码还有其他建议吗? 最佳答案 您看到的差异是由ruby​​

  8. day1-数组part01| 704. 二分查找、27. 移除元素 - 2

    数组理论基础数组是存放在连续内存空间上的相同类型数据的集合。数组下标从0开始数组内存空间的地址是连续的c++中vector和array的区别1、vector是顺序容器,其利用连续的内存空间来存储元素,但是其内存空间大小是能够改变的。2、array是顺序容器,其也是利用连续的内存空间来存储元素,但它的内存空间是固定大小的,申请之后就无法改变。3、vector的底层是array实现的二维数组二维数组在内存的空间地址是连续的704|二分查找思路1、把整个数组一分为二;2、判断目标值在左区间还是右区间,若在左区间,则修改右区间指针的位置;若在右区间,则修改新区间的左区间位置3、重复上述过程,直到lef

  9. day44|● 完全背包● 518. 零钱兑换 II ● 377. 组合总和 Ⅳ - 2

    518.零钱兑换II1.代码classSolution{public:intchange(intamount,vector&coins){vectorf(amount+1,0);f[0]=1;for(inti=0;i2.动规五部曲1.确定dp数组和其下标含义由题目说可知求选择钱票得到总和为target的方案数,dp[j]相当于选择物品体积相加为i的方案数2.递推公式每次加入物品,都有可能到达体积j,所以在每次加上这个物品到达j时加上这个方案数f[j]+=f[j-coins[i]];3.初始化因为在for循环和dp公式中没有确切的值,肯定需要初始化,初始化第一个就可以保证后面的推导出来了,f[0

  10. 代码随想录day2|有序数组的平方、长度最小的子数组、螺旋矩阵 - 2

    前言:今天去校医院拔了两颗牙,太痛了,今天写的博客就比较水。1、有序数组的平方(双指针法)classSolution{public:vectorsortedSquares(vector&nums){intk=nums.size()-1;vectorresult(nums.size(),0);//创造一个数组result长度与nums相同for(inti=0,j=nums.size()-1;i2、长度最小的子数组(滑动窗口)classSolution{public:intminSubArrayLen(inttarget,vector&nums){intresult=INT32_MAX;//返回值

随机推荐