ListNode链表加和题目

参考地址:

https://www.jianshu.com/p/0d0dbfcbc1c3

给出两个 非空 的链表用来表示两个非负的整数。其中,它们各自的位数是按照 逆序 的方式存储的,并且它们的每个节点只能存储 一位 数字。
如果,我们将这两个数相加起来,则会返回一个新的链表来表示它们的和。
您可以假设除了数字 0 之外,这两个数都不会以 0 开头。
示例:
输入:(2 -> 4 -> 3) + (5 -> 6 -> 4)
输出:7 -> 0 -> 8
原因:342 + 465 = 807

首先完成链表定义,然后分析题目逻辑

package javaTest.Licode.twonumbersum;

/**
 * code
 */
public class code {

    public static void main(String[] args) {
        ListNode l1 = new ListNode(4);
        l1.next = new ListNode(5);
        l1.next.next = new ListNode(6);

        ListNode l2 = new ListNode(7);
        l2.next = new ListNode(8);
        l2.next.next = new ListNode(9);
    }

    public ListNode addTwoNumbers(ListNode l1, ListNode l2) {

        return null;
    }

    class ListNode {
        int val;
        ListNode next;
    
        ListNode(int x) {
            val = x;
        }
    }
}

初步理解从链表最深处开始计算如果加和大于10就向前面一处加一 保留各位

加一加在哪 我认为任意一个链表就可以 所以加在L1

问题就回到了取得 链表深度 以及取得对应深度的值上面

下面是完整版

package javaTest.Licode.twonumbersum;

/**
 * code
 */
public class code {

    public static void main(String[] args) {
        ListNode l1 = new ListNode(4);
        l1.next = new ListNode(5);
        l1.next.next = new ListNode(6);

        ListNode l2 = new ListNode(7);
        l2.next = new ListNode(8);
        l2.next.next = new ListNode(9);


        

        
        ListNode l3 = addTwoNumbers(l1, l2); 

        //l1第一位异常加一是因为最后一步计算加一加在了第一位,已经在结果l3中修改
        while (l1 != null) {
            System.out.print(l1.val);
            l1 = l1.next;
            if (l1 != null) {
                System.out.print("->");
            }
        }
        System.out.println("//");
        while (l2 != null) {
            System.out.print(l2.val);
            l2 = l2.next;
            if (l2 != null) {
                System.out.print("->");
            }
        }
        System.out.println("//");
        while (l3 != null) {
            System.out.print(l3.val);
            l3 = l3.next;
            if (l3 != null) {
                System.out.print("->");
            }
        }
        

    }

    public static ListNode addTwoNumbers(ListNode l1, ListNode l2) {
        // 得到两个链表的深度 假设长度相同
        int length1 = getListNodeLength(l1);
        int length2 = getListNodeLength(l2);
        ListNode l3;
        //结果是否需要进位 标识  如 9->9 2->2 结果为 1->2->1 增加一位
        boolean needAdd = false;
        // 两个最大的个位数相加永远不能出现三位数 所以计算一号位是否大于10可以得出实际长度 +-误差1
        if (l1.val + l2.val + 1 >= 10) {
            l3 = creatListNodeValue(length1 + 1);
            l3.val = 1;
            needAdd = true;
        } else {
            l3 = creatListNodeValue(length1);
        }

        for (int i = length1; i > 0; i--) {
            int sum = getListNodeValue(l1, i) + getListNodeValue(l2, i);
            if (sum >= 10) {
                if (needAdd) {
                    setListNodeValue(l3, i + 1, sum - 10);
                    // 如果大于10 重新赋值l1的前一位的值在原基础加一
                    setListNodeValue(l1, i -1 , getListNodeValue(l1, i - 1) + 1);

                } else {
                    setListNodeValue(l3, i, sum - 10);
                    // 如果大于10 重新赋值l1的前一位的值在原基础加一
                    setListNodeValue(l1, i - 1, getListNodeValue(l1, i - 1) + 1);

                }
            } else {
                if (needAdd) {
                    setListNodeValue(l3, i + 1, sum);
                } else {
                    setListNodeValue(l3, i, sum);
                }

            }
        }

        return l3;
    }

    // 定义链表
    static class ListNode {
        int val;
        ListNode next;

        ListNode(int x) {
            val = x;
        }
    }

    // 得到最大深度
    public static int getListNodeLength(ListNode l) {
        int length = 0;
        while (l != null) {
            length++;
            l = l.next;
        }

        return length;
    }

    // 得到对应深度的值
    public static int getListNodeValue(ListNode l, int depth) {

        for (int i = 1; i < depth; i++) {
            l = l.next;
        }
        return l.val;
    }

    // 对应深度的赋值
    public static void setListNodeValue(ListNode l, int depth, int value) {
        // 索引是从1开始
        for (int i = 1; i < depth; i++) {
            l = l.next;
        }
        l.val = value;
    }

    // 创建一个空的数组
    public static ListNode creatListNodeValue(int depth) {
        ListNode l2 = new ListNode(0);

        ListNode l3 = l2;
        // 因为外层定义了一个所以长度减一
        for (int i = 0; i < depth - 1; i++) {

            l2.next = new ListNode(0);
            l2 = l2.next;
        }
        return l3;
    }
}

然后发现审错题了,有点尴尬

ListNode链表加和题目_第1张图片
原题目是往深处添加
也就是 个 十 百 千 万
有点违背常识啊…
ListNode链表加和题目_第2张图片
稍微修改 将最深处位赋值为1
从1开始赋值 每层循环加加就可以了

ListNode链表加和题目_第3张图片
出错果然没有考虑位数不同是不行的 给出的是[1,8] 和[0]
重新修改代码吧

public static ListNode addTwoNumbers(ListNode l1, ListNode l2) {
        // 得到两个链表的深度 假设长度相同
        int length1 = getListNodeLength(l1);
        int length2 = getListNodeLength(l2);
        int maxLength;
        int minLength;
        if(length1 - length2 >= 0 ){
            maxLength = length1;
            minLength = length2;
        }else{
            maxLength = length2;
            minLength = length1;
        }
        ListNode l3;
        //结果是否需要进位 标识  如 9->9 2->2 结果为 1->2->1 增加一位
        boolean needAdd = false;
        // 两个最大的个位数相加永远不能出现三位数 所以计算一号位是否大于10可以得出实际长度 +-误差1
       
        if (l1.val + l2.val + 1 >= 10 && length1 == length2) {
            l3 = creatListNodeValue(length1 + 1);
            setListNodeValue(l3, maxLength + 1, 1);
            needAdd = true;
        } else {
            l3 = creatListNodeValue(maxLength);
        }
        int l3Length = 1;
        for (int i = maxLength; i > 0; i--) {
            int sum;
            if(i > minLength){
                if(length1 - length2 > 0 ){
                     sum = getListNodeValue(l1, i);
                   
                }else{
                    maxLength = length2;
                     sum = getListNodeValue(l2, i);
                }
            }else{
                 sum = getListNodeValue(l1, i) + getListNodeValue(l2, i);
            }
            if (sum >= 10) {
                if (needAdd) {
                    setListNodeValue(l3, l3Length, sum - 10);
                    // 如果大于10 重新赋值l1的前一位的值在原基础加一
                    setListNodeValue(l1, i -1 , getListNodeValue(l1, i - 1) + 1);

                } else {
                    setListNodeValue(l3, l3Length, sum - 10);
                    // 如果大于10 重新赋值l1的前一位的值在原基础加一
                    setListNodeValue(l1, i - 1, getListNodeValue(l1, i - 1) + 1);

                }
            } else {
                if (needAdd) {
                    setListNodeValue(l3, l3Length, sum);
                } else {
                    setListNodeValue(l3, l3Length, sum);
                }

            }
            l3Length++;
        }

        return l3;
    }

ListNode链表加和题目_第4张图片
ListNode链表加和题目_第5张图片因为考虑误差导致错误
我决定忽视误差看能否骗过考官
ListNode链表加和题目_第6张图片
又双叒叕审错题
添加一个方法 关于最后一位问题

 // 得到对应深度的值
     public static ListNode getListNode(ListNode l, int depth) {

        for (int i = 1; i < depth; i++) {
            l = l.next;
        }
        return l;
    }
public static ListNode addTwoNumbers(ListNode l1, ListNode l2) {
        // 得到两个链表的深度 假设长度相同
        int length1 = getListNodeLength(l1);
        int length2 = getListNodeLength(l2);
        int maxLength;
        int minLength;
        if(length1 - length2 >= 0 ){
            maxLength = length1;
            minLength = length2;
        }else{
            maxLength = length2;
            minLength = length1;
        }
        ListNode  l3 = creatListNodeValue(maxLength);
       
        int l3Length = 1;
        for (int i = 1; i <= maxLength; i++) {
            int sum;
            if(i > minLength){
                if(length1 - length2 > 0 ){
                     sum = getListNodeValue(l1, i);
                   
                }else{
                     sum = getListNodeValue(l2, i);
                }
            }else{
                 sum = getListNodeValue(l1, i) + getListNodeValue(l2, i);
            }
            System.out.println(sum);
            boolean needAdd = i == maxLength;
            if (sum >= 10) {
                if (!needAdd) {
                    setListNodeValue(l3, l3Length, sum - 10);
                    // 如果大于10 重新赋值l1的前一位的值在原基础加一
                    if(length1 - length2 > 0 ){
                        setListNodeValue(l1,  i + 1, getListNodeValue(l1,  i + 1) + 1);
                      
                   }else{
                    setListNodeValue(l2,  i + 1, getListNodeValue(l2,  i + 1) + 1);
                   }

                } else {
                    setListNodeValue(l3, l3Length, sum - 10);
                     //结果是否需要进位 标识  如 9->9 2->2 结果为 1->2->1 增加
                    getListNode(l3,  i).next = new ListNode(1);
                }
            } else {
                if (needAdd) {
                    setListNodeValue(l3, l3Length, sum);
                } else {
                    setListNodeValue(l3, l3Length, sum);
                }

            }
            l3Length++;
        }

        return l3;
    }

ListNode链表加和题目_第7张图片
已然不想优化
ListNode链表加和题目_第8张图片
去掉打印速度勉强可以接受吧

你可能感兴趣的:(算法)