链表求和-c语言

链表求和

给定两个用链表表示的整数,每个节点包含一个数位。

这些数位是反向存放的,也就是个位排在链表首部。

编写函数对这两个整数求和,并用链表形式返回结果。

示例:

输入:(7 -> 1 -> 6) + (5 -> 9 -> 2),即617 + 295
输出:2 -> 1 -> 9,即912

进阶:思考一下,假设这些数位是正向存放的,又该如何解决呢?

示例:

输入:(6 -> 1 -> 7) + (2 -> 9 -> 5),即617 + 295
输出:9 -> 1 -> 2,即912

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     struct ListNode *next;
 * };
 */

int f(struct ListNode* l1){
 
    int t=1;
    while(l1){
  //      printf("dfa");
    printf(" --%d",l1->val);
      
        sum=sum+l1->val*t;
        t=t*10;
        l1=l1->next;
    }
    //    printf("dfa");
    return sum;

}
struct ListNode*  f2(int val){
    struct ListNode *p=(struct ListNode*)malloc(sizeof(struct ListNode));
     struct ListNode *L;

    p->next=NULL;
    if(val==0){
          L=(struct ListNode*)malloc(sizeof(struct ListNode));
        L->val=0;
      
        L->next=p->next;
        p->next=L;

    }
    while(val){
        L=(struct ListNode*)malloc(sizeof(struct ListNode));
        L->val=val%10;
        val=val/10;
        L->next=p->next;
        p->next=L;

    }
   f_verser(p);
    return p;

}
void f_verser(struct ListNode*p){
    struct ListNode *L,*s;
    L=p->next;
    p->next=NULL;
    while(L){
        s=L;
        L=L->next;
        s->next=p->next;
        p->next=s;
    }
}

struct ListNode* addTwoNumbers(struct ListNode* l1, struct ListNode* l2){
    int s1=f(l1);
    int s2=f(l2);
    struct ListNode *p;
    printf(" l1 %d",s1);

   // printf("%d ",s1+s2);
    p=f2(s1+s2);
    return p->next;
   
 

}

下面有一个更好的解法:
思路与算法

由于输入的两个链表都是反向存放数字的位数的,因此两个链表中同一位置的数字可以直接相加。

我们同时遍历两个链表,逐位计算它们的和,并与当前位置的进位值相加。具体而言,如果当前两个链表处相应位置的数字为 n1,n2n_1,n_2n1​,n2​,进位值为 carry\textit{carry}carry,则它们的和为 n1+n2+carryn_1+n_2+\textit{carry}n1​+n2​+carry;其中,答案链表处相应位置的数字为 (n1+n2+carry) mod 10(n_1+n_2+\textit{carry}) \bmod 10(n1​+n2​+carry)mod10,而新的进位值为 ⌊n1+n2+carry10⌋\Big\lfloor\dfrac{n_1+n_2+\textit{carry}}{10}\Big\rfloor⌊10n1​+n2​+carry​⌋。

如果两个链表的长度不同,则可以认为长度短的链表的后面有若干个 000 。

此外,如果链表遍历结束后,有 carry>0\textit{carry} > 0carry>0,还需要在答案链表的后面附加一个节点,节点的值为 carry\textit{carry}carry。

struct ListNode* addTwoNumbers(struct ListNode* l1, struct ListNode* l2) {
    struct ListNode *head = NULL, *tail = NULL;
    int carry = 0;
    while (l1 || l2) {
        int n1 = l1 ? l1->val : 0;
        int n2 = l2 ? l2->val : 0;
        int sum = n1 + n2 + carry;
        if (!head) {
            head = tail = malloc(sizeof(struct ListNode));
            tail->val = sum % 10;
            tail->next = NULL;
        } else {
            tail->next = malloc(sizeof(struct ListNode));
            tail->next->val = sum % 10;
            tail = tail->next;
            tail->next = NULL;
        }
        carry = sum / 10;
        if (l1) {
            l1 = l1->next;
        }
        if (l2) {
            l2 = l2->next;
        }
    }
    if (carry > 0) {
        tail->next = malloc(sizeof(struct ListNode));
        tail->next->val = carry;
        tail->next->next = NULL;
    }
    return head;
}


  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我可以帮你回答这个问题。以下是用 C 语言实现单链表求和的代码: ```c #include <stdio.h> #include <stdlib.h> typedef struct ListNode { int val; struct ListNode *next; } ListNode; ListNode* addTwoNumbers(ListNode* l1, ListNode* l2) { ListNode* res = NULL; // 储存结果的链表 ListNode* tail = res; // 链表的尾部 int carry = 0; // 进位数初始化为0 while (l1 || l2 || carry) { int sum = (l1 ? l1->val : 0) + (l2 ? l2->val : 0) + carry; // 求和 carry = sum / 10; // 计算进位数 sum %= 10; // 取余数 ListNode* node = (ListNode*) malloc(sizeof(ListNode)); // 创建新节点 node->val = sum; // 赋值 node->next = NULL; if (!res) { res = node; tail = res; } else { tail->next = node; tail = tail->next; } // 更新 l1 和 l2,向后遍历 if (l1) l1 = l1->next; if (l2) l2 = l2->next; } return res; } void printList(ListNode* head) { while (head) { printf("%d ", head->val); head = head->next; } printf("\n"); } int main() { ListNode* l1 = (ListNode*) malloc(sizeof(ListNode)); ListNode* l2 = (ListNode*) malloc(sizeof(ListNode)); l1->val = 2; l1->next = (ListNode*) malloc(sizeof(ListNode)); l1->next->val = 4; l1->next->next = (ListNode*) malloc(sizeof(ListNode)); l1->next->next->val = 3; l1->next->next->next = NULL; l2->val = 5; l2->next = (ListNode*) malloc(sizeof(ListNode)); l2->next->val = 6; l2->next->next = (ListNode*) malloc(sizeof(ListNode)); l2->next->next->val = 4; l2->next->next->next = NULL; ListNode* res = addTwoNumbers(l1, l2); printList(res); return 0; } ``` 希望这个代码能帮到你!

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值