当前位置: 首页 > news >正文

【PTA刷题】求链式线性表的倒数第K项(代码+详解)

文章目录

    • 题目
    • 代码
    • 详解

题目

给定一系列正整数,请设计一个尽可能高效的算法,查找倒数第K个位置上的数字。

输入格式:

输入首先给出一个正整数K,随后是若干非负整数,最后以一个负整数表示结尾(该负数不算在序列内,不要处理)。

输出格式:

输出倒数第K个位置上的数据。如果这个位置不存在,输出错误信息NULL

输入样例:

4 1 2 3 4 5 6 7 8 9 0 -1

输出样例:

7

代码

#include <stdio.h>
#include <stdlib.h>// 定义链表节点结构
struct ListNode {int val;struct ListNode* next;
};// 查找倒数第K个位置上的数字
int findKthFromEnd(struct ListNode* head, int k) {if (!head || k <= 0) return -1; // 如果链表为空或k小于等于0,返回-1表示错误struct ListNode* slow = head;struct ListNode* fast = head;// 快指针先移动k步for (int i = 0; i < k; ++i) {if (!fast) return -1; // 如果链表长度小于k,返回-1表示错误fast = fast->next;}// 同时移动慢指针和快指针,直到快指针到达链表尾部while (fast) {slow = slow->next;fast = fast->next;}return slow->val;
}int main() {int k;scanf("%d", &k);int num;struct ListNode* head = NULL;struct ListNode* tail = NULL;// 构建链表while (scanf("%d", &num) && num >= 0) {struct ListNode* newNode = (struct ListNode*)malloc(sizeof(struct ListNode));newNode->val = num;newNode->next = NULL;if (!head) {head = newNode;tail = newNode;} else {tail->next = newNode;tail = newNode;}}int result = findKthFromEnd(head, k);if (result != -1) {printf("%d\n", result);} else {printf("NULL\n");}// 释放链表内存while (head) {struct ListNode* temp = head;head = head->next;free(temp);}return 0;
}

详解

这个问题要求找出一个正整数序列中倒数第K个元素的值。为了解决这个问题,代码使用了一个快慢指针的方法,并且用链表来存储输入的序列。下面是对这个算法和代码的详细解释:

算法逻辑

  1. 使用快慢指针:

    • 快指针 (fast) 和慢指针 (slow) 都从链表的头结点开始。
    • 先让快指针向前移动K步。这样,快慢指针之间就保持了K个节点的距离。
    • 然后,同时移动快指针和慢指针,直到快指针到达链表的末尾。此时,慢指针所在的位置就是倒数第K个节点。
  2. 边界条件处理:

    • 如果链表为空(head == NULL),或者K值不合理(k <= 0),函数直接返回-1,表示错误。
    • 如果链表长度小于K,也就是快指针在移动K步之前已经到达了链表末尾,函数同样返回-1。

代码解释

  1. 链表节点定义:

    • struct ListNode 定义了链表的节点结构,包括节点值 val 和指向下一个节点的指针 next
  2. 主函数 main:

    • 读取K值。
    • 通过循环读取输入的正整数,并构建链表。遇到负数时停止读取。
    • 调用 findKthFromEnd 函数来查找倒数第K个元素的值。
  3. 查找函数 findKthFromEnd:

    • 初始化快慢指针。
    • 让快指针先移动K步。
    • 同时移动快慢指针,直到快指针到达末尾。
    • 返回慢指针所指向的节点的值。
  4. 输出结果:

    • 如果返回值不是-1,则输出该值。
    • 如果返回值是-1,输出"NULL"。
  5. 释放内存:

    • 循环释放链表的每个节点,避免内存泄露。

这个算法的时间复杂度是O(n),因为它最多遍历链表两次:一次用于构建链表,一次用于找到倒数第K个元素。

http://www.lryc.cn/news/259431.html

相关文章:

  • VSCode 创建工作区,多文件夹终端切换
  • 高阶函数(js的问题)
  • android-android源码目录
  • Json格式化
  • 数据可视化设计:让数据故事更有说服力
  • java面试题-Spring事务以及@Transactional注解详解
  • ARM流水灯
  • docker-compose单机容器编排
  • matlab信号分选系统算法-完整算法结构
  • 十八)Stable Diffusion使用教程:艺术二维码案例
  • 【LeetCode每日一题】53. 最大子数组和
  • 机器学习笔记 什么是协方差矩阵?
  • 使用Python监控服务器在线状态
  • 【JAVA】黑马MybatisPlus 学习笔记【二】【核心功能】
  • 区块链实验室(30) - 区块链期刊:Distributed Ledger Technologies: Research and Practice
  • Nginx【通俗易懂】《中篇》
  • 组件的二次封装
  • curl+postman 在java开发中的使用(提高效率)
  • 【电子取证:FTK IMAGER 篇】DD、E01系统镜像动态仿真
  • netcat瑞士军刀
  • 【征稿倒计时十天】第三届高性能计算与通信工程国际学术会议(HPCCE 2023)
  • 编程应用实际场景:台球厅怎么样用电脑给客人计时,台球计时收费系统操作教程
  • 云计算大屏,可视化云计算分析平台(云实时数据大屏PSD源文件)
  • 高频js-----js执行机制 Event Loop
  • 恢复出厂设置后在 Android 上恢复照片的 6 种常用方法
  • 人工智能_机器学习065_SVM支持向量机KKT条件_深度理解KKT条件下的损失函数求解过程_公式详细推导_---人工智能工作笔记0105
  • 网线市场现状与发展趋势预测
  • 力扣二叉树--第四十一天
  • 计算机视觉(P2)-计算机视觉任务和应用
  • redis-学习笔记(Jedis zset 简单命令)