首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

循环双向链表在OCaml中的实现

循环双向链表是一种数据结构,它由多个节点组成,每个节点包含一个数据元素和两个指针,分别指向前一个节点和后一个节点。在OCaml中,可以使用记录类型和递归定义来实现循环双向链表。

以下是一个简单的循环双向链表的实现示例:

代码语言:txt
复制
type 'a node = {
  value: 'a;
  mutable prev: 'a node option;
  mutable next: 'a node option;
}

type 'a circular_doubly_linked_list = {
  mutable head: 'a node option;
  mutable tail: 'a node option;
}

let create () =
  { head = None; tail = None }

let is_empty list =
  list.head = None

let add_first list value =
  let new_node = { value; prev = None; next = None } in
  match list.head with
  | None ->
    new_node.prev <- Some new_node;
    new_node.next <- Some new_node;
    list.head <- Some new_node;
    list.tail <- Some new_node;
  | Some head ->
    new_node.prev <- Some list.tail;
    new_node.next <- Some head;
    head.prev <- Some new_node;
    list.tail <- Some new_node;
    list.head <- Some new_node

let add_last list value =
  let new_node = { value; prev = None; next = None } in
  match list.tail with
  | None -> add_first list value
  | Some tail ->
    new_node.prev <- Some tail;
    new_node.next <- Some list.head;
    tail.next <- Some new_node;
    list.head <- Some new_node;
    list.tail <- Some new_node

let remove_first list =
  match list.head with
  | None -> ()
  | Some head ->
    match head.next with
    | None ->
      list.head <- None;
      list.tail <- None;
    | Some next ->
      let tail = Option.get head.prev in
      tail.next <- Some next;
      next.prev <- Some tail;
      list.head <- Some next

let remove_last list =
  match list.tail with
  | None -> ()
  | Some tail ->
    match tail.prev with
    | None ->
      list.head <- None;
      list.tail <- None;
    | Some prev ->
      let head = Option.get tail.next in
      prev.next <- Some head;
      head.prev <- Some prev;
      list.tail <- Some prev

let rec to_list list =
  match list.head with
  | None -> []
  | Some head ->
    let rec loop node acc =
      if node == list.tail then
        node.value :: acc
      else
        loop (Option.get node.next) (node.value :: acc)
    in
    loop head []

let example_list = create ()
add_first example_list 1
add_last example_list 2
add_last example_list 3
remove_first example_list
remove_last example_list
let example_list_values = to_list example_list

在这个示例中,我们定义了一个记录类型'a node来表示链表的节点,其中包含一个值value和两个可变指针prevnext。我们还定义了一个记录类型'a circular_doubly_linked_list来表示循环双向链表,其中包含一个可变指针headtail,分别指向链表的头部和尾部。

我们提供了一些基本的操作函数,如create用于创建一个空的循环双向链表,is_empty用于判断链表是否为空,add_first用于在链表的头部添加一个节点,add_last用于在链表的尾部添加一个节点,remove_first用于移除链表的头部节点,remove_last用于移除链表的尾部节点,to_list用于将链表转换为列表。

在示例中,我们创建了一个循环双向链表example_list,并添加了一些节点,然后移除了头部和尾部的节点,并将链表转换为列表example_list_values

循环双向链表在OCaml中的实现可以用于各种场景,例如实现LRU缓存、实现循环队列等。腾讯云提供了丰富的云计算产品,例如云服务器、云数据库、云存储等,可以根据具体需求选择适合的产品。

页面内容是否对你有帮助?
有帮助
没帮助

相关·内容

循环链表的实现_建立双向循环链表

循环链表   循环链表是一个收尾相接的链表,将单链表的最后一个指针域改由NULL改为指向表头结点这就是单链式的循环链表,并称为循环单链表   带头结点的循环单链表的各种操作的算法实现与带头结点单链表的算法实现类似...单链表中的判别条件为p!=NULL或p->next!=NULL,而单循环链表判别条件是p!=L或p->next!=L   在循环单链表中附设尾指针有时候比附设头指针更简单。...如:在用头指针的循环单链表中找a1的时间复杂度是O(1),找an需要从头找到尾,时间复杂度是O(n),如果用为指针rear,找开始结点和终端结点的存储位置分别是rear->next->next和rear...    方法一:先找到两个链表LA,LB的表尾,分别用p,q指向它,然后将第一个链表的表尾与第二个链表的第一个结点连起来,修改第二个表的尾q,使它的链域指向第一个表头 //头指针合并循环链表 #include...;//返回新的链表的尾指针 }   循环链表求长度 #include #define len sizeof(Node) #include typedef struct

75320

单循环链表-带头双向循环链表的实现

带头双向循环链表   前言   对于链表来说,不只有单链表这一个品种;   链表有很多种形态   按方向分:单向、双向   按带不带头:带头、不带头   按循环:循环、不循环   1、单向或则双向:...今天我们就来学习一下结构最复杂的带头双向循环链表!!!...;   虽然名字听上去比较复杂单循环链表,但是实现起来比单链表(全名:不带头、不循环、单向链表)更加简单,也不需要过多考虑特殊情况;   两种链表的比较:(上面是单链表,下面是带头双向循环链表)   结构分析...、这两个接口就能快速实现出带头双向循环链表了;   总代码及头文件   头文件的包含:    #pragma once #include #include #include...// 双向链表在pos的前面进行插入 void ListInsert(ListNode* pos, LTDataType x); // 双向链表删除pos位置的节点 void

61130
  • 如何实现双向循环链表

    引言 双向带头循环链表是一种常见的数据结构,它具有双向遍历的特性,并且在表头和表尾之间形成一个循环。本文将深入探讨双向带头循环链表的结构、操作和应用场景,帮助读者更好地理解和运用这一数据结构。...本篇博客将以图表和代码相结合的方式手撕双向带头循环链表,代码使用C语言进行实现。 1....我们要实现的是一个双向带头循环链表,所以在初始化的时候使哨兵节点的next指向自己,prev指向自己,这样的结构对后面对链表的操作会方便很多,提供了很大的便利。...,所以在循环带头双向链表中哨兵节点的前驱节点就是最后一个节点的后继节点。...双向带头循环链表作为一种重要的数据结构,在实际开发中有着广泛的应用,希望本文能够帮助读者更好地理解和应用这一数据结构。

    12910

    DS:带头双向循环链表的实现

    博主的上篇文章介绍了链表,以及单链表的实现。 单链表的实现(超详细!!) 其实单链表的全称叫做不带头单向不循环链表,本文会重点介绍链表的分类以及双链表的实现!...虽然有这么多的链表的结构,但是我们实际中最常用还是两种结构: 单链表(不带头单向不循环链表)和 双向链表(带头双向循环链表) 1. 无头单向非循环链表:结构简单,⼀般不会单独⽤来存数据。...实际中更多是作为其他数据结 构的⼦结构,如哈希桶、图的邻接表等等。另外这种结构在笔试⾯试中出现很多。 2. 带头双向循环链表:结构最复杂,⼀般⽤在单独存储数据。...实际中使⽤的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使⽤代码实现以后会发现结构会带 来很多优势,实现反⽽简单了,后⾯我们代码实现了就知道了。...struct ListNode* prev;//指针保存前一个结点的地址 struct ListNode* next;//指针保存后一个结点的地址 }LTNode; 四、带头双向循环链表的实现 4.1

    12310

    循环双向链表的

    需要删除节点p3时就很麻烦,我们需要从头去遍历,找到next指针为p3时将next指针指向p3的next;   为此方便起见,我们可以使用双向链表进行实现。...内核中是这样处理的,   创建一个双向循环链表   =>headp1p2p3p4=   向链表中指定位置插入节点   原有链prenext   这也是最基本的插入节点的方法...add_data的基础上实现就直观多了;   add_head(struct data *input,struct data* head){     _add_data(input,head,head-...,head->pre,head);//头节点的前一个节点就是尾,在尾和头结点之间插入就是插入到尾了。   ...}   没有做释放的代码,创建链的时候需要用malloc去创建,内核中的双向链表正是这么实现的,   特别容易书写,不太会产生副作用。二级指向是在太难理解了

    29010

    TencentOS-tiny中双向循环链表的实现及使用

    什么是双向循环链表 双向链表也是链表的一种,区别在于每个节点除了后继指针外,还有一个前驱指针,双向链表的节点长下面这样: [c7p68g2ngv.png] 由这种节点构成的双向链表有两种分类:按照是否有头结点可以分为两种...本文讨论的是不带头节点的双向循环链表,如下图: [qowp0vrk7c.png] 2. 双向循环链表的实现 TencentOS-tiny中的双向链表实现在tos_list.h中。 2.1....双向链表初始化 链表初始化的实现如下: void tos_list_init(k_list_t *list) { list->next = list; list->prev = list...插入前的双向循环链表如下: [12x9hk0jf4.png] 插入后的双向循环链表如下: [g8b3e5w8ks.png] 图中的四个插入过程分别对应代码中的四行代码。...TencentOS-tiny中依然提供了两个宏定义来解决这一问题,在tos_klib.h中。

    1.1K1313

    【数据结构】—带头双向循环链表的实现(完美链表)

    目录 前言 链表的实现 新节点的创建 链表初始化 尾插与尾删 头插与头删 查找数据 在任意位置的插入与删除 链表的销毁 总结 前言 链表结构一共有八种形式,在前面的文章里已经讲完了不带头单向非循环链表的实现...,但是我们发现该链表实现尾插与尾删时比较麻烦,要先从头节点进行遍历,找到尾节点,时间复杂度为O(N),而本次所讲的带头双向循环单链表,则可以直接找到尾节点。...空表状态下应该是如下图这样的,因为它是带头的循环链表,所以第一个节点不用来存储有效数据。...// 双向链表在pos的前面进行插入 void ListInsert(ListNode* pos, LTDataType x) { assert(pos); //pos前面的节点 ListNode...真的是链表中的完美存在,不过在进行删除操作时,一定要考虑空表情况下不可进行删除。因此要加个assert进行断言。

    61820

    数据结构:双向链表实现队列与循环链表

    一、双向链表(double linked list)如图26.5,是在单链表的每个结点中,再设置一个指向其前驱结点的指针域。...要实现双向链表只需在《图示单链表的插入和删除操作》中代码的基础上改动两个地方。...在linkedlist.h中修改链表节点的结构体定义: struct node  {  unsigned char item;  link prev, next; }; 在linkedlist.c...在《队列的链式存储结构》中我们使用单链表实现队列的尾进头出,下面我们演示使用双向链表实现队列的头进尾出。...我们在《队列的顺序存储结构(循环队列)》中使用数组实现了环形队列,我们还要“假想”它是首尾相接的,而如果基于链表实现环形队列,我们本来就可以用指针串成首尾相接的。

    2K80

    数据结构Java实现:循环链表和双向链表

    单链表是最基础的一种链表结构,在实际开发中的使用有一些局限性,比如只能单向往后查找节点,如果需要找到某元素的前驱节点,单链表是无法实现的,今天来给大家分享另外两个复杂一些的链表结构:循环链表和双向链表。...循环链表 循环链表本质上就是一种单链表,两者的不同之处在于链表中最后一个节点的指针指向哪里,在单链表中,末尾节点的指针指向空,如下图所示。 ?...而在循环链表中,末尾节点的指针指向首节点,形成一个闭环,所以它叫循环链表,应该很好理解,如下图所示。 ?...接下来用 Java 实现一个循环链表的结构,只需要在原有单链表的基础上稍作修改即可,如下所示。...如果是双向链表的结构,每一个节点都会记录其前驱节点,可直接获取,所以此时的时间复杂度为 O(1)。 ? 搞清楚了双向链表的概念,接下来我们用 Java 来实现双向链表的结构。

    3.5K20

    双向带头循环链表的(增删查改)的实现

    一、双向带头循环链表 构成 二、双向带头循环链表的实现 1.函数的定义和结构体的创建——list.h #include #include #include双向带头循环链表与单链表的传递参数区别 1.单链表: 单链表因为没有头节点的存在,导致在尾插时会改变链表的头节点 所以需要传递二级指针的地址即二级指针。...2.双向带头循环链表: 初始化头指针时,是需要传递二级指针,只不过用函数传回结构体指针的方式代替了, 而在后续接口则不需要传递二级指针,因为后来都是在头指针的基础上进行的,而头节点本身不会存储有效数据,...4.双向带头循环链表的接口 1.初始化 struct listNode* stackinit()//初始化头节点 { struct listNode* phead = (struct listNode...(2)在动态开辟空间时,会造成一定的浪费。 2.链表: 逻辑上是来连续的,物理上的不连续。

    40540

    链表和双向链表的实现

    前言 ---- 链表中的数据通过指针连接,添加、插入或删除节点只需要修改指针指向 实现思路 实现一个链表需要具备以下方法 在链表尾部添加节点 获取链表所有节点的数据 链表指定位置插入元素 获取链表指定位置的节点数据...获取节点在链表中的位置 更新链表指定位置的数据 移除链表指定位置的节点 移除链表中的指定节点 判断链表是否为空 获取链表长度 链表内部需要定义head指针和链表长度 实现代码 定义head指针和length...(linkedList.size()) 双向链表 双向链表的指针是双向的,前指针指向上一个节点,后指针指向下一个节点 head指向第一个节点,tail指向最后一个节点 双向链表实现思路 需要具备以下方法...尾部插入元素 任意位置插入元素 获取所有节点数据 正向遍历链表获取节点数据 反向遍历链表获取节点数据 获取指定位置的节点数据 获取指定数据在链表中的位置 更新指定位置的节点数据 移除指定位置的节点 移除指定数据的节点...判断链表是否为空 获取链表长度 定义head和tail分别指向第一个节点和最后一个节点 代码实现 /** * 双向链表 */ function DoublyLinkedList() { //指向第一个节点

    71040

    数据结构 | TencentOS-tiny中的双向循环链表的实现及使用

    什么是双向循环链表 双向链表也是链表的一种,区别在于每个节点除了后继指针外,还有一个前驱指针,双向链表的节点长下面这样: ?...由这种节点构成的双向链表有两种分类:按照是否有头结点可以分为两种,按照是否循环可以分为两种。 本文讨论的是不带头节点的双向循环链表,如下图: ?...相较于其他形式的链表,双向循环链表的添加节点,删除节点,遍历节点都非常的简单。 2. 双向循环链表的实现 TencentOS-tiny中的双向链表实现在tos_list.h中。 2.1....插入前的双向循环链表如下: ? 插入后的双向循环链表如下: ? 图中的四个插入过程分别对应代码中的四行代码。...TencentOS-tiny中依然提供了两个宏定义来解决这一问题,在tos_klib.h中。

    91020

    Linux内核中双向链表的经典实现

    概要 本文对双向链表进行探讨,介绍的内容是Linux内核中双向链表的经典实现和用法。其中,也会涉及到Linux内核中非常常用的两个经典宏定义offsetof和container_of。...内容包括: 1.Linux中的两个经典宏定义 2.Linux中双向链表的经典实现 Linux中的两个经典宏定义 倘若你查看过Linux Kernel的源码,那么你对 offsetof 和 container_of...Linux中双向链表的经典实现 1.Linux中双向链表介绍 Linux双向链表的定义主要涉及到两个文件: include/linux/types.h include/linux/list.h Linux...中双向链表的使用思想 它是将双向链表节点嵌套在其它的结构体中;在遍历链表的时候,根据双链表节点的指针获取"它所在结构体的指针",从而再获取数据。...通过双向链表将人进行关联的模型图如下: ? person代表人,它有name和age属性。为了通过双向链表对person进行链接,我们在person中添加了list_head属性。

    2.7K30

    带头双向循环链表增删查改实现(C语言)

    带头双向循环链表 结点结构与头结点的创建 头插尾插 打印链表 头删与尾删 链表的查找 在pos的前面进行插入与删除pos位置的结点 销毁链表 完整代码 结点结构与头结点的创建 创建两个源文件和一个头文件...test.c linked_list.c linked_list.h 带头双向循环链表,那么,结点的结构就要有两个指针域,分别指向前一个结点和后一个结点。...这里尾插就很方便了,不像之前需要遍历找尾,因为是循环链表,尾的next就是头结点。 当然这里要先写一个创建新结点的函数。...打印链表 这里就要遍历链表了,因为是循环结构,所以结尾就不用空指针进行判断了。...printf("\n"); } 头删与尾删 删除的话,我们需要写一个函数判断链表中是否还有数据,如果只剩一个头结点就不能继续删除了。

    57300

    双向链表的优雅实现

    文中涉及的代码可访问 GitHub:https://github.com/UniqueDong/algorithms.git 上次我们说了「单向链表」的代码实现,今天带大家一起玩下双向链表,双向链表的节点比单项多了一个指针引用...双向链表就像渣男,跟「前女友」和「现女友」,还有一个「备胎』都保持联系。前女友就像是前驱节点,现女友就是 「当前 data」,而「next」指针就像是他套住的备胎。...使用这样的数据结构就能实现「进可攻退可守」灵活状态。 接下来让我们一起实现『渣男双向链表』。...定义好渣男节点后,就开始实现我们的双向链表。...一种是在指定节点的前面插入新节点。 在后面添加前面尾巴添加已经说过,对于在指定节点的前面插入需要我们先找到指定位置节点,然后改变他们的 prev next 指向。

    82130

    002 Linux内核中双向链表的经典实现

    概要 本文对双向链表进行探讨,介绍的内容是Linux内核中双向链表的经典实现和用法。其中,也会涉及到Linux内核中非常常用的两个经典宏定义offsetof和container_of。...内容包括: 1.Linux中的两个经典宏定义 2.Linux中双向链表的经典实现 Linux中的两个经典宏定义 倘若你查看过Linux Kernel的源码,那么你对 offsetof 和 container_of...Linux中双向链表的经典实现 1.Linux中双向链表介绍 Linux双向链表的定义主要涉及到两个文件: include/linux/types.h include/linux/list.h Linux...中双向链表的使用思想 它是将双向链表节点嵌套在其它的结构体中;在遍历链表的时候,根据双链表节点的指针获取"它所在结构体的指针",从而再获取数据。...通过双向链表将人进行关联的模型图如下: ? person代表人,它有name和age属性。为了通过双向链表对person进行链接,我们在person中添加了list_head属性。

    1.8K20

    java双向链表解析实现双向链表的创建含代码

    一.双向链表 单向链表从头部开始我们的每一个节点指向后驱的节点。...此处为单向链表 单向链表 双向链表是相互指向前驱以及后驱的链表 前驱链表我们需要在我们的MyListCode内部类中在定义一个previous来接收每一个前驱的地址 想要删除任意节点可以直接通过访问下一个节点使其...prev获取想要删除的上一个节点,然后将想要删除的上一个节点.next获取到被删除对象下一个节点的指向 这里我们可以模拟实现MyListCode类中的一些方法,入头插法、尾叉法、任意位置插入节点...、指定元素删除含有该元素的第一个节点、指定元素删除含有该元素的所有节点等… 二.创建MyListCode类实现双向链表创建 public class MyListNode implements IList...或者在尾巴,则头插法和尾叉法 将给的坐标作为循环条件节点开始走,跳出循环后改节点位置就是要添加的位置 首先要把改节点的坐标向后移动一位,插入其中间 单链表的话将cur先指向后一个节点在指向前一个节点

    9010
    领券