LeetCode每日一练(两数之和)

举报
wangweijun 发表于 2022/03/29 22:35:11 2022/03/29
【摘要】 题目如下: 给你两个非空的链表,表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的,并且每个节点只能存储 一位 数字。请你将两个数相加,并以相同形式返回一个表示和的链表。 你可以假设除了...

题目如下:

给你两个非空的链表,表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的,并且每个节点只能存储 一位 数字。请你将两个数相加,并以相同形式返回一个表示和的链表。
你可以假设除了数字 0 之外,这两个数都不会以 0 开头。

在这里插入图片描述
题目很好理解,就是给你两个链表,比如243和564,需要逆序得到链表所代表的的数值,分别是342和465,将这两个数相加,得到结果807,再逆序存回一个链表并返回。

了解题目的意思之后,我们先来分析一下,这道题思路还是比较简单的,首先遍历两个链表,并对遍历结果进行逆序处理,得到两个数,然后相加得到结果,再逆序存入链表。

首先我们需要得到两个链表所代表的数值:

public static ListNode addTwoNumbers(ListNode l1, ListNode l2) {
        // 遍历l1链表得到数值
        StringBuilder sb = new StringBuilder();
        while (l1 != null) {
            sb.append(l1.val);
            l1 = l1.next;
        }
        // 将数值逆序
        String str1 = sb.reverse().toString();
        Integer num1 = Integer.valueOf(str1);

        // 清空sb
        sb.setLength(0);

        // 以同样的方式得到链表l2的数值
        while (l2 != null) {
            sb.append(l2.val);
            l2 = l2.next;
        }
        String str2 = sb.reverse().toString();
        Integer num2 = Integer.valueOf(str2);

        // 两数相加
        int result = num1 + num2;

        System.out.println(num1 + "+" + num2 + "=" + result);
        return null;
    }

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28

运行结果:

342+465=807

  
 
  • 1

得到结果之后,就创建一个新的链表,将结果逆序放入新链表中:

public static ListNode addTwoNumbers(ListNode l1, ListNode l2) {
        // 遍历l1链表得到数值
        StringBuilder sb = new StringBuilder();
        while (l1 != null) {
            sb.append(l1.val);
            l1 = l1.next;
        }
        // 将数值逆序
        String str1 = sb.reverse().toString();
        Integer num1 = Integer.valueOf(str1);

        // 清空sb
        sb.setLength(0);

        // 以同样的方式得到链表l2的数值
        while (l2 != null) {
            sb.append(l2.val);
            l2 = l2.next;
        }
        String str2 = sb.reverse().toString();
        Integer num2 = Integer.valueOf(str2);

        // 两数相加
        int result = num1 + num2;
        // 将结果整理成链表
        String strResult = String.valueOf(result);
        char[] chars = strResult.toCharArray();
        ListNode listNode = new ListNode();
        ListNode temp = listNode;
        for (int i = chars.length - 1; i >= 0; --i) {
            if (i == 0) {
                temp.val = Integer.parseInt(String.valueOf(chars[i]));
                // 对于尾结点,其指针域为空
                temp.next = null;
            } else {
                temp.val = Integer.parseInt(String.valueOf(chars[i]));
                temp.next = new ListNode();
                temp = temp.next;
            }
        }
        return listNode;
    }

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42

都是一些非常简单的链表操作,如果还不会链表的话,数据结构需要补补课了呦,遍历返回的链表就得到结果:

708

  
 
  • 1

怀着喜悦的心情将代码放到LeetCode上运行,结果没通过:
在这里插入图片描述
原来是测试数据太大了,导致int类型装不下,那就将其改为long类型:

......
Long num1 = Long.valueOf(str1);
Long num2 = Long.valueOf(str2);
Long result = num1 + num2;
......

  
 
  • 1
  • 2
  • 3
  • 4
  • 5

结果还是没有通过:
在这里插入图片描述
我的天,测试数据居然这么长,那没办法,我们将其改为BigDecimal就可以了,最终代码:

public static ListNode addTwoNumbers(ListNode l1, ListNode l2) {
        // 遍历l1链表得到数值
        StringBuilder sb = new StringBuilder();
        while (l1 != null) {
            sb.append(l1.val);
            l1 = l1.next;
        }
        // 将数值逆序
        String str1 = sb.reverse().toString();
        BigDecimal decimal1 = new BigDecimal(str1);

        // 清空sb
        sb.setLength(0);

        // 以同样的方式得到链表l2的数值
        while (l2 != null) {
            sb.append(l2.val);
            l2 = l2.next;
        }
        String str2 = sb.reverse().toString();
        BigDecimal decimal2 = new BigDecimal(str2);


        // 两数相加
        BigDecimal resultDecimal = decimal1.add(decimal2);
        // 将结果整理成链表
        String strResult = resultDecimal.toString();
        char[] chars = strResult.toCharArray();
        ListNode listNode = new ListNode();
        ListNode temp = listNode;
        for (int i = chars.length - 1; i >= 0; --i) {
            if (i == 0) {
                temp.val = Integer.parseInt(String.valueOf(chars[i]));
                temp.next = null;
            } else {
                temp.val = Integer.parseInt(String.valueOf(chars[i]));
                temp.next = new ListNode();
                temp = temp.next;
            }
        }
        return listNode;
    }

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42

测试通过:
在这里插入图片描述
可以看到执行用时是比较长的,只击败了18.79%的用户,我们来分析一下导致执行用时过长的原因,首先是对链表的遍历,我们一共遍历了两次链表,然后是链表的创建,这些都是非常耗时的操作,有没有办法能够只进行一次遍历就完成题目的要求呢?

题目表示链表的数字是按逆序方式存储的,这刚好给了我们一个便利的处理方式,我们可以同时遍历两个链表,并分别取出两个链表同一位置上的两个数值,相加后直接放到新创建的链表中,比如243和564链表:

在这里插入图片描述
由于数值是逆序存储,所以链表的第一个元素其实是数值的最后一个数,将2和5相加得到7,故结果的最后一位数为7,再将其存入新的链表,作为第一个结点即可:
在这里插入图片描述
然后l1和l2后移一位:
在这里插入图片描述
该位置上的两个数相加结果为10,这里就需要考虑到进位的问题,要让当前位置的数值为0,并向前一位进1,只需让相加的结果除以10,就能够得到进位;比如10除以10等于1,就向前进1,23除以10等于2,就向前进2。再让相加的结果模10就能够得到当前位置的结果数,比如10模10等于0,当前位置就是0,23模10等于3,当前位置就是3。由此可知,结果的倒数第二个数为0,将其存入新的链表:
在这里插入图片描述
l1、l2继续后移:
在这里插入图片描述
该位置上的两个数相加等于7,需要注意还得加上进位,所以结果的第一位为8,存入新链表:
在这里插入图片描述
此时l1、l2后移,结果为空,故计算结束,这样我们就通过一次遍历直接得到了结果链表。

当然,我们还需要考虑两个链表不一样长的情况,比如:
在这里插入图片描述
对于这种情况,前面两个数的求法都是一样的,5加2等于7,存入新链表,6加4等于10,向前进1,将0存入新链表,此时l1、l2再后移,l1为空,但l2仍然有值,这时候我们给l1一个占位,让其等于0,这样就能够继续相加,此时4加0,再加上进位等于5,所以最终结果为:
在这里插入图片描述
由此得到代码:

public ListNode addTwoNumbers(ListNode l1, ListNode l2) {
        // 新链表的头指针、尾指针均初始化为null
        ListNode head = null, tail = null;
        // 初始化进位为0
        int carry = 0;
        // 同时遍历l1和l2链表
        while (l1 != null || l2 != null) {
            // 处理两个链表不等长的情况,若某个链表遍历结束,结点为null,则让其值等于0
            int n1 = l1 != null ? l1.val : 0;
            int n2 = l2 != null ? l2.val : 0;
            // 两数相加,记得加上进位
            int sum = n1 + n2 + carry;
            // 采用尾插法建立新链表
            if (head == null) {
                head = tail = new ListNode(sum % 10); // 模10得到当前位置的数值
            } else {
                tail.next = new ListNode(sum % 10); // 模10得到当前位置的数值
                tail = tail.next;
            }

            // 计算进位
            carry = sum / 10;
            if (l1 != null) {
                l1 = l1.next;
            }
            if (l2 != null) {
                l2 = l2.next;
            }
        }
        // 若是两个链表遍历结束后,仍然有进位,则进位应成为新的一位
        if (carry > 0) {
            tail.next = new ListNode(carry);
        }
        return head;
    }

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35

因为计算得到的结果需要按顺序依次放入新链表,故而这里采用尾插法建立新的链表。

最后将代码提交到LeetCode上,测试通过:
在这里插入图片描述

文章来源: blizzawang.blog.csdn.net,作者:·wangweijun,版权归原作者所有,如需转载,请联系作者。

原文链接:blizzawang.blog.csdn.net/article/details/118612098

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。