乘风破浪:LeetCode真题_002_Add Two Numbers

乘风破浪:LeetCode真题_002_Add Two Numbers

一、前言

    这次的题目是关于链表方面的题目,把两个链表对应节点相加,还要保证进位,每个节点都必须是十进制的0~9。因此主要涉及到链表,指针方面的知识,以及活学活用的编程能力。

二、LeetCode真题_002_Add Two Numbers

2.1 问题介绍

2.2 分析与解决

看到这样的问题,我们首先要分析清题意,之后画出一个原理图,然后就便于解决了。可以看到主要是包括了进位的问题,因此我们每一次相加的时候需要考虑到进位,并且得到的结果需要取余数作为该节点的结果,并且取整数作为下一位的进位,这里要非常注意,因为是十以内的两位数相加,因此进位最多为9+9+1=19,取整之后为1,因此没有进位则进位为零,有进位,则进位为1。

 1 public ListNode addTwoNumbers(ListNode l1, ListNode l2) {
 2     ListNode dummyHead = new ListNode(0);
 3     ListNode p = l1, q = l2, curr = dummyHead;
 4     int carry = 0;
 5     while (p != null || q != null) {
 6         int x = (p != null) ? p.val : 0;
 7         int y = (q != null) ? q.val : 0;
 8         int sum = carry + x + y;
 9         carry = sum / 10;
10         curr.next = new ListNode(sum % 10);
11         curr = curr.next;
12         if (p != null) p = p.next;
13         if (q != null) q = q.next;
14     }
15     if (carry > 0) {
16         curr.next = new ListNode(carry);
17     }
18     return dummyHead.next;
19 }

  下面看看我们的算法:

1 public class ListNode {
2     int val;
3     ListNode next;
4
5     ListNode(int val) {
6         this.val = val;
7     }
8 }

   使用上面的数据结构进行的算法:

 1 public class Solution {
 2
 3     public ListNode addTwoNumbers(ListNode l1, ListNode l2) {
 4
 5         if (l1 == null) {
 6             return l2;
 7         }
 8
 9         if (l2 == null) {
10             return l1;
11         }
12
13         ListNode p1 = l1;
14         ListNode p2 = l2;
15         ListNode root = new ListNode(0); // 头结点
16         ListNode r = root;
17         root.next = l1;
18
19         int carry = 0; // 初始进位
20         int sum;
21         while (p1 != null && p2 != null) {
22             sum = p1.val + p2.val + carry;
23             p1.val = sum % 10; // 本位的结果
24             carry = sum / 10; // 本次进位
25
26             r.next = p1;
27             r = p1; // 指向下一个相加的结点
28             p1 = p1.next;
29             p2 = p2.next;
30
31         }
32
33         if (p1 == null) {
34             r.next = p2;
35         } else {
36             r.next = p1;
37         }
38
39         // 最后一次相加还有进位
40         if (carry == 1) {
41             // 开始时r.next是第一个要相加的结点
42             while (r.next != null) {
43                 sum = r.next.val + carry;
44                 r.next.val = sum % 10;
45                 carry = sum / 10;
46                 r = r.next;
47             }
48
49             // 都加完了还有进位,就要创建一个新的结点
50             if (carry == 1) {
51                 r.next = new ListNode(1);
52             }
53         }
54
55         return root.next;
56     }
57 }

三、总结

系统给出的算法是比较不错的,不但简单而且整洁,便于理解,并且考虑到了所有的可能情况,值得我们去学习和模仿。

原文地址:https://www.cnblogs.com/zyrblog/p/10208020.html

时间: 2024-08-30 08:54:41

乘风破浪:LeetCode真题_002_Add Two Numbers的相关文章

乘风破浪:LeetCode真题_008_String to Integer (atoi)

乘风破浪:LeetCode真题_008_String to Integer (atoi) 一.前言 将整型转换成字符串,或者将字符串转换成整型,是经常出现的,也是必要的,因此我们需要熟练的掌握,当然也有很多工具来实现了,但是在这个基础上加入一些其他的因素就是考点的所在了. 二.String to Integer (atoi) 2.1 问题理解 2.2 问题分析和解决     看到这个问题,我们就需要遍历字符串,然后判断开始的时候是不是空格,+,-,或者数字,如果不是的话就是不合理的,如果是,则继

乘风破浪:LeetCode真题_007_Reverse Integer

乘风破浪:LeetCode真题_007_Reverse Integer 一.前言 这是一个比较简单的问题了,将整数翻转,主要考察了取整和取余,以及灵活地使用long型变量防止越界的问题. 二.Reverse Integer 2.1 问题理解 2.2 问题分析与解决    可以看到通过简单地取整和取余运算就能得到答案,但是需要注意越界问题,使用long在Java中8个字节的特性来完成越界检查和处理.    我们的算法: public class Solution { /** * <pre> *

乘风破浪:LeetCode真题_013_Roman to Integer

乘风破浪:LeetCode真题_013_Roman to Integer 一.前言 上一节我们讨论了如何把阿拉伯数字转换成罗马数字,现在我们需要思考一下如何把罗马数字转换成阿拉伯数字,其实我们仔细观擦这些结构就会发现罗马数字如果前面的比后面的小,就需要用后面的减去前面的.而且如果有这样的运算,也只是两个字符拼接而成的,这为我们解题提供了思路. 二.Roman to Integer 2.1 问题 2.2 分析与解决 根据题意,我们可以明白只需要从开始到结尾遍历这些罗马数字,如果发现前一个小于后一个

乘风破浪:LeetCode真题_021_Merge Two Sorted Lists

乘风破浪:LeetCode真题_021_Merge Two Sorted Lists 一.前言 关于链表的合并操作我们是非常熟悉的了,下面我们再温故一下将两个有序链表合并成一个的过程,这是基本功. 二.Merge Two Sorted Lists 2.1 问题 2.2  分析与解决 这是比较简单的,将两个已经有序的链表合并成一个,只需要两个指针不断的遍历和比较,判断谁的大以此来变换指针即可. /** * Definition for singly-linked list. * public cl

乘风破浪:LeetCode真题_024_Swap Nodes in Pairs

乘风破浪:LeetCode真题_024_Swap Nodes in Pairs 一.前言 这次还是链表的操作,不过我们需要交换链表奇数和偶数位置上的节点,因此要怎么做呢? 二.Swap Nodes in Pairs 2.1 问题      要求是不能修改节点的元素,并且使用的空间也是常数级别的,不能使用更高的内存空间. 2.2 分析与解决      通过分析我们可以使用简单的遍历算法,通过四个指针来解决,当然我们也可以使用递归算法来解答. 使用正常遍历算法: public class Solut

乘风破浪:LeetCode真题_026_Remove Duplicates from Sorted Array

乘风破浪:LeetCode真题_026_Remove Duplicates from Sorted Array 一.前言     我们这次的实验是去除重复的有序数组元素,有大体两种算法. 二.Remove Duplicates from Sorted Array 2.1 问题      题目大意理解,就是对数组进行元素去重,然后返回去处重复之后的长度,无论我们对数组做了什么的修改,都没有关系的,只要保证再返回的长度之内的数组正确性即可.因为最后是根据长度来遍历的,因此我们不用担心. 2.2 分析

乘风破浪:LeetCode真题_025_Reverse Nodes in k-Group

乘风破浪:LeetCode真题_025_Reverse Nodes in k-Group 一.前言 将一个链表按照一定的长度切成几部分,然后每部分进行翻转以后再拼接成一个链表是比较困难的,但是这也能锻炼我们的思维能力. 二.Reverse Nodes in k-Group 2.1 问题 2.2 分析与解决     最简单的想法,我们可以将链表分成几部分,每一个部分分开考虑,比如使用头插法,正好可以将顺序颠倒一下,或者我们通过某种方式使得顺序发生改变,然后再结合起来. /** * Definiti

乘风破浪:LeetCode真题_027_Remove Element

乘风破浪:LeetCode真题_027_Remove Element 一.前言 这次是从数组中找到一个元素,然后移除该元素的所有结果,并且返回长度. 二.Remove Element 2.1 问题 2.2 分析与解决     这个题和上一题是非常相似的,只不过这次是从数组中找到给定的元素,并且删除该元素,同时返回剩余数组的长度,超过长度的元素不用管,存不存在都可以.于是我们想到了和上次一样的方法,用一个指针指向开始,一个指向结尾,开始的向后移动,如果遇到需要删除的元素,则用最后的元素替代,最后的

乘风破浪:LeetCode真题_031_Next Permutation

乘风破浪:LeetCode真题_031_Next Permutation 一.前言 这是一道经典的题目,我们实在想不出最好的方法,只能按照已有的方法来解决,同时我们也应该思考一下为什么要这样做?是怎么想到的?这比我们记住步骤更加的有用. 二.Next Permutation 2.1 问题 2.2 分析与解决 排列(Arrangement),简单讲是从N个不同元素中取出M个,按照一定顺序排成一列,通常用A(M,N)表示.当M=N时,称为全排列(Permutation).从数学角度讲,全排列的个数A