参考地址:
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;
}
}
然后发现审错题了,有点尴尬
原题目是往深处添加
也就是 个 十 百 千 万
有点违背常识啊…
稍微修改 将最深处位赋值为1
从1开始赋值 每层循环加加就可以了
出错果然没有考虑位数不同是不行的 给出的是[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;
}
因为考虑误差导致错误
我决定忽视误差看能否骗过考官
又双叒叕审错题
添加一个方法 关于最后一位问题
// 得到对应深度的值
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;
}