【算法】解题总结:剑指Offer 73 翻转单词序列、剑指Offer 18 删除链表的节点(2种思路)

JZ73 翻转单词序列

(中等)

题目

描述
牛客最近来了一个新员工Fish,每天早晨总是会拿着一本英文杂志,写些句子在本子上。同事Cat对Fish写的内容颇感兴趣,有一天他向Fish借来翻看,但却读不懂它的意思。例如,“nowcoder. a am I”。后来才意识到,这家伙原来把句子单词的顺序翻转了,正确的句子应该是“I am a nowcoder.”。Cat对一一的翻转这些单词顺序可不在行,你能帮助他么?在这里插入图片描述

示例1
输入:
“nowcoder. a am I”
返回值:
“I am a nowcoder.”

示例2
输入:
“”
返回值:
“”

思路

只需先定义一个存储结果的字符串变量,将传入的反序的参数字符串按空格分割开后得到一个字符串数组,逆向遍历这个数组,每次添加到之前预先定义的字符串变量的末尾即可,本题用 Java 来做十分得简单,但是归为了较难的题型,可能对于C/C++ 的代码实现者来说,确实有些额外的思考工作,除此之外我想不到还有什么能让这道题归为较难的原因。
在这里插入图片描述

实现

public class JZ73翻转单词序列 {
    public String ReverseSentence(String str) {
        if (str == null || str.length() == 0) {
            return str;
        }

        String[] words = str.split(" ");
        String[] res = new String[words.length];

        String reverse = "";
        for (int i = words.length - 1; i >= 0; i--) {
            reverse += words[i] + " ";
        }

        return reverse.substring(0, reverse.length() - 1);
    }
}

在这里插入图片描述

删除链表的节点

(中等)

题目

描述
给定单向链表的头指针和一个要删除的节点的值,定义一个函数删除该节点。返回删除后的链表的头节点。
1.此题对比原题有改动
2.题目保证链表中节点的值互不相同
3.该题只会输出返回的链表和结果做对比,所以若使用 C 或 C++ 语言,你不需要 free 或 delete 被删除的节点
数据范围:
0<=链表节点值<=10000
0<=链表长度<=10000

示例1
输入:
{2,5,1,9},5
返回值:
{2,1,9}
说明:
给定你链表中值为 5 的第二个节点,那么在调用了你的函数之后,该链表应变为 2 -> 1 -> 9

示例2
输入:
{2,5,1,9},1
返回值:
{2,5,9}
说明:
给定你链表中值为 1 的第三个节点,那么在调用了你的函数之后,该链表应变为 2 -> 5 -> 9

思路 1

删除结点我这里的思路是先进行一次遍历,遍历结束后要考虑三种情况:

  • 1、没找到:没找到则进行相应处理,可抛出异常,可打印提示,也可返回原链表,等等;
  • 2、找到了,是第一个结点:则返回链表的第二个结点作为头结点;
  • 3、找到了,不是第一个结点:则让待删除结点的前驱,指向待删除结点的后继,从而让 Java 的垃圾回收器回收掉待删除结点,从而达到删除的目的,并且最终要返回原链表的头结点。

实现 1

public class JZ18删除链表的节点 {
    public ListNode deleteNode(ListNode head, int val) {
        if (head == null) return null;

        ListNode temp = head;
        ListNode pre = null;
        while (temp != null && temp.val != val) {
            pre = temp;
            temp = temp.next;
        }
        //1 没找到
        if (temp == null) {
            System.out.println("未找到此结点~");
            return null;
        }
        //2 找到
        // 2.1 是第一个结点
        if (pre == null) return head.next;
        // 2.2 非第一个结点
        pre.next = temp.next;
        return head;
    }
}

在这里插入图片描述

思路 2

思路 2 与思路 1 类似,只是是从此结点的下一个结点为考虑点,从而去掉了指向当前结点前驱的 pre 变量,换了一种思维方式而已,删除结点的操作十分基础,也比较灵活,但大体脱不开这两类情况的判断,其中第二类情况又包含两种情况:

  • 1 待删除结点:是第一个结点
  • 2 待删除结点:是非第一个结点
    2.1 待删除结点:未找到
    2.2 待删除结点:找到

实现 2

public class JZ18删除链表的节点 {
    public ListNode deleteNode(ListNode head, int val) {
        if (head == null) return null;
        ListNode temp = head;
        //1 待删除结点:是第一个结点
        if (temp.val == val) return temp.next;
        //2 待删除结点:是非第一个结点
        while (temp.next != null && temp.next.val != val) {
            temp = temp.next;
        }
         //2.1 待删除结点:未找到
        if (temp.next == null) return head;
         //2.2 待删除结点:找到,为 temp.next
        temp.next = temp.next.next;
        return head;
    }
}

在这里插入图片描述

  • 3
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

超周到的程序员

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值