python反转单链表算法题

现在算法是大厂面试的必考题,而且越来越难,已经不是简单的列表,字符串操作了,会涉及到各种数据结结构。单链表的反转也是经常考的一道题,里面故在此记录一下。

1.链表的特点:

顺序存储元素,但是元素在空间上是不连续的,所以在链表每个元素中除了存储元素的值,还会存储下一个元素的地址,单链表的话就只有指向下一个元素的指针,双向链表的话还会有指向前一个元素的指针。正是由于链表以上的存储特点,在做插入和删除操作时只需要断开指针的连接,不需要移动后面的数据,所以对链表修改的效率会很高,但是查找的效率会很低,这也正是链表和数组的区别。链表的存储示意图:

完成这道题至少有3种方式:

1.先对原链表做头部删操作,再对新链表做头部插入操作;
2.通过3个指针实现,p0指向前一个节点的指针,p1表示当前节点的指针,p2表示下一个节点的指针
3.用递归实现;

2.方式一:

1)创建一个新的空列表;

2)取出旧链表中头结点的元素,将其next指针设置为新链表的头指针,同时将旧链表的头结点执行下一个元素

3)依次重复第2)步的操作,直到取出就链表中所有的节点。

4)最后形成的新链表如下,实现了反转的功能:

5)代码实现:

# encoding=utf-8
import copy
 
 
class Node:
    """节点类,包含值和下一个节点的指针"""
    def __init__(self, value, next=None):
        self.value = value
        self.next = next
 
    def __str__(self):
        return "Node value:%s" % self.value
 
 
class LinkedList:
    def __init__(self, head=None, tail=None):
        self.head = head
        self.tail = tail
 
    def get_all(self):
        """获取链表中所有的节点"""
        nodes = []
        temp = self.head
        while temp:
            nodes.append(temp.value)
            temp = temp.next
        return nodes
 
    def add_node(self, value):
        """在列表中添加节点"""
        node = Node(value)
        # 空链表,收尾指针都指向新增加的节点
        if self.head is None:
            self.head = node
            self.tail = node
        else:
            self.tail.next = node
            self.tail = node
 
    def reverse_list(self):
        """反转单向列表
        思路一:先对原链表做头删操作,再对新链表做头插
        """
        # 定义一个新的链表
        new_list_node = LinkedList()
        temp = copy.deepcopy(self.head)
        while temp:
            # 1.对之前的链表做头删除操作
            node = temp
            temp = temp.next
 
            # 2.对新的链表做头插入操作
            if new_list_node.head is None:
                new_list_node.tail = node
            node.next = new_list_node.head
            new_list_node.head = node
        return new_list_node
 
 
if __name__ == "__main__":
    l = LinkedList()
    for i in range(5):
        l.add_node(i)
    new_list_node = l.reverse_list()
    print(new_list_node.get_all())
    print(new_list_node.tail)

3.方式二

借助3个指针来实现反转,p0之前前一个节点,p1指向当前操作的节点,p2指向下一个节点。操作过程如下:将p1的next指针之前p0,之后将p0指向p1节点,p1指向p2节点,如果p1不为空,重复以上操作,最后,p0即为新列表的头节点。

1)开始时p0为空;将p1指向链表的头节点,p1节点的next指针指向p0;p2指向下一个节点:

2)调整3个指针的指向:将p0指向p1;p1指向p2,p1的next指向p0;p2指向下一个节点

3)循环执行以上步骤,直到p1指向的节点不为空,最后得到的链表为:

4)代码实现:

# encoding=utf-8
import copy
 
 
class Node:
    """节点类,包含值和下一个节点的指针"""
    def __init__(self, value, next=None):
        self.value = value
        self.next = next
 
    def __str__(self):
        return "Node value:%s" % self.value
 
 
class LinkedList:
    def __init__(self, head=None, tail=None):
        self.head = head
        self.tail = tail
 
    def get_all(self):
        """获取链表中所有的节点"""
        nodes = []
        temp = self.head
        while temp:
            nodes.append(temp.value)
            temp = temp.next
        return nodes
 
    def add_node(self, value):
        """在列表中添加节点"""
        node = Node(value)
        # 空链表,收尾指针都指向新增加的节点
        if self.head is None:
            self.head = node
            self.tail = node
        else:
            self.tail.next = node
            self.tail = node
 
    def reverse_list(self):
        """
        反转单向列表
        思路二:通过3个指针实现,p0指向前一个节点的指针,p1表示当前节点的指针,p2表示下一个节点的指针
        :return: LinkedList 对象
        """
        p1 = copy.deepcopy(self.head)
        p2 = p1.next
        # 定义一个新的链表对象
        new_list_node = LinkedList()
        while p1:
            if new_list_node.head is None:
                new_list_node.tail = p1
            # 将p1的next指向新链表的头结点
            p1.next = new_list_node.head
            # 将新链表的头结点指向p1
            new_list_node.head = p1
            # 将p1指向p2
            p1 = p2
            # 判断p2是否指向了链表的末尾
            if p2:
                p2 = p2.next
        return new_list_node
 
 
if __name__ == "__main__":
    l = LinkedList()
    for i in range(5):
        l.add_node(i)
    new_list_node = l.reverse_list()
    print(new_list_node.get_all())
    print(new_list_node.tail)

4.方式三:

递归就是将问题分解为处理过程一致的子问题进行处理,但是一定要要结束条件。链表的反转也可以采用递归的方式实现,每次传入的节点不是最后一个的话,就将下一个节点作为参数传递进去,递归调用;直到传入的是最后一个节点时开始逐级返回。

1)将链表中每个节点作为参数,依次传入到reverse_list函数中;

2)当传入的是最后一个节点时,以此节点为头结点创建一个新的链表,并将此链表返回;

3)上一级的调用者接受到返回的链表后,将传入的节点作为链表的尾结点放到链表中;

4)逐级返回,直到回到最开始执行reverse_list函数中,将生成的新链表返回即可

5)代码实现:

# encoding=utf-8
import copy
 
 
class Node:
    """节点类,包含值和下一个节点的指针"""
    def __init__(self, value, next=None):
        self.value = value
        self.next = next
 
    def __str__(self):
        return "Node value:%s" % self.value
 
 
class LinkedList:
    def __init__(self, head=None, tail=None):
        self.head = head
        self.tail = tail
 
    def get_all(self):
        """获取链表中所有的节点"""
        nodes = []
        temp = self.head
        while temp:
            nodes.append(temp.value)
            temp = temp.next
        return nodes
 
    def add_node(self, value):
        """在列表中添加节点"""
        node = Node(value)
        # 空链表,收尾指针都指向新增加的节点
        if self.head is None:
            self.head = node
            self.tail = node
        else:
            self.tail.next = node
            self.tail = node
 
 
    def reverse_list(self, node):
        """
        反转单链表
        思路三:用递归实现
        :return:LinkedList 对象
        """
        if node.next is None:
            return LinkedList(node, node)
        temp = copy.deepcopy(node)
        # 断开当前节点的连接
        temp.next = None
        # 将当前节点放到内层递归返回的链表结尾
        l = self.reverse_list(node.next)
        l.tail.next = temp
        l.tail = temp
        return l
 
 
if __name__ == "__main__":
    l = LinkedList()
    for i in range(5):
        l.add_node(i)
    new_list_node = l.reverse_list1()
    print(new_list_node.get_all())
    print(new_list_node.tail)

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • python如何实现单链表的反转

    这篇文章主要介绍了python如何实现单链表的反转,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 代码如下 # coding=utf-8 class Node: def __init__(self, data=None, next=None): self.data = data self.next = next def Reserver(link): pre = link cur = link.next pre.next = None whil

  • 单链表反转python实现代码示例

    单链表的反转可以使用循环,也可以使用递归的方式 1.循环反转单链表 循环的方法中,使用pre指向前一个结点,cur指向当前结点,每次把cur->next指向pre即可. 代码: class ListNode: def __init__(self,x): self.val=x; self.next=None; def nonrecurse(head): #循环的方法反转链表 if head is None or head.next is None: return head; pre=None; c

  • Python3实现的反转单链表算法示例

    本文实例讲述了Python3实现的反转单链表算法.分享给大家供大家参考,具体如下: 反转一个单链表. 方案一:迭代 # Definition for singly-linked list. # class ListNode: # def __init__(self, x): # self.val = x # self.next = None class Solution: def reverseList(self, head): """ :type head: ListNod

  • python反转单链表算法题

    现在算法是大厂面试的必考题,而且越来越难,已经不是简单的列表,字符串操作了,会涉及到各种数据结结构.单链表的反转也是经常考的一道题,里面故在此记录一下. 1.链表的特点: 顺序存储元素,但是元素在空间上是不连续的,所以在链表每个元素中除了存储元素的值,还会存储下一个元素的地址,单链表的话就只有指向下一个元素的指针,双向链表的话还会有指向前一个元素的指针.正是由于链表以上的存储特点,在做插入和删除操作时只需要断开指针的连接,不需要移动后面的数据,所以对链表修改的效率会很高,但是查找的效率会很低,这

  • C++和python实现单链表及其原理

    目录 一.链表的基本概念 二.单链表 1.python实现 (1)节点设计 (2)链表类:Single_Linked_List (3)判断链表是否为空:is_empty()函数 (4)头插法:add()函数 (5)尾插法:append()函数 (6)在任意位置插入:insert()函数 (7)计算链表长度:get_length()函数 (8)遍历所有节点:traversal()函数 (9)搜索:search()函数 (10)删除:delete()函数 2.C++实现 (1)节点设计 (2)链表类

  • python实现单链表中删除倒数第K个节点的方法

    本文实例为大家分享了python实现单链表中删除倒数第K个节点的具体代码,供大家参考,具体内容如下 题目: 给定一个链表,删除其中倒数第k个节点. 代码: class LinkedListAlgorithms(object): def __init__(self): pass def rm_last_kth_node(self, k, linked_list): # 删除倒数第 K 个节点,针对单链表的 if linked_list.is_empty(): print 'The given li

  • python版单链表反转

    本文实例为大家分享了vue + element ui实现锚点定位的具体代码,供大家参考,具体内容如下 代码如下: class Node(object):     def __init__(self, elem, next_=None):         self.elem = elem         self.next = next_   def reverseList(head):     if head == None or head.next==None:  # 若链表为空或者仅一个数就

  • Python实现单链表中元素的反转

    给定一个单链表,将其反转.其实很容易想到,只需要修改每个结点的指针指向:即令后一个结点指向前一个结点,并且将表头指针指向最后一个结点即可. 这个过程可以用循环实现,也可以用递归来实现. 1.用循环来实现: class LNode:     def __init__(self, elem):         self.elem = elem         self.pnext = None   def reverse(head):     if head is None or head.pnex

  • python实现单链表的方法示例

    前言 首先说下线性表,线性表是一种最基本,最简单的数据结构,通俗点讲就是一维的存储数据的结构. 线性表分为顺序表和链接表: 顺序表示指的是用一组地址连续的存储单元依次存储线性表的数据元素,称为线性表的顺序存储结构或顺序映像: 链式表示指的是用一组任意的存储单元存储线性表中的数据元素,称为线性表的链式存储结构.而他既可以是连续的也可以不连续,是通过一个与后继结点的连接信息构建起来的. *顺序表(这个不是本次重点,简单介绍一下) 顺序表是用一段连续的存储单元依次存储数据元素,查找元素是很方便的,但是

  • python环形单链表的约瑟夫问题详解

    题目: 一个环形单链表,从头结点开始向后,指针每移动一个结点,就计数加1,当数到第m个节点时,就把该结点删除,然后继续从下一个节点开始从1计数,循环往复,直到环形单链表中只剩下了一个结点,返回该结点. 这个问题就是著名的约瑟夫问题. 代码: 首先给出环形单链表的数据结构: class Node(object): def __init__(self, value, next=0): self.value = value self.next = next # 指针 class RingLinkedL

  • c语言实现单链表算法示例分享

    复制代码 代码如下: #include <stdio.h>#include <stdlib.h>typedef char DataType;typedef struct Node{    DataType data;    struct Node * Next;}ListNode,* LinkList;void Judement(LinkList head){ //判断分配内存    if (!head){        printf("Overflow.");

随机推荐