为了账号安全,请及时绑定邮箱和手机立即绑定

经典算法题每日演练——第二十一题 十字链表

标签:
算法

 

     上一篇我们看了矩阵的顺序存储,这篇我们再看看一种链式存储方法“十字链表”,当然目的都是一样,压缩空间。

一:概念

    既然要用链表节点来模拟矩阵中的非零元素,肯定需要如下5个元素(row,col,val,down,right),其中:

row:矩阵中的行。

col:矩阵中的列。

val:矩阵中的值。

right:指向右侧的一个非零元素。

down:指向下侧的一个非零元素。

现在我们知道单个节点该如何表示了,那么矩阵中同行的非零元素的表示不就是一个单链表吗?比如如下:

那么进一步来说一个多行的非零元素的表示不就是多个单链表吗,是的,这里我把单链表做成循环链表,我们来看看如何用十字链表

来表示稀疏矩阵。

从上面的十字链表中要注意两个问题:

第一:这里有一个填充色的节点,是十字链表中的总结点,它是记录该矩阵中的(row,col,value)和一个指向下一个头节点的next指针。

第二:每个链表都有一个头指针,总结点用next指针将它们贯穿起来。

 

二:操作

1:数据结构

   刚才也说了,十字链表的总结点有一个next指针,而其他非零节点没有,所以为了方便,我们用一个Unit类包装起来。

复制代码

 1         #region 单一节点 2         /// <summary> 3         /// 单一节点 4         /// </summary> 5         public class Node 6         { 7             //行号 8             public int rows; 9 10             //列号11             public int cols;12 13             //向下的指针域14             public Node down;15 16             //向右的指针域17             public Node right;18 19             //单元值(头指针的next和val)20             public Unit unit;21         }22         #endregion23 24         #region 统一“表头节点”和“非零节点”25         /// <summary>26         /// 统一“表头节点”和“非零节点”27         /// </summary>28         public class Unit29         {30             //表头节点的next域31             public Node next;32 33             //非零元素的值34             public int value;35         }36         #endregion

复制代码

2:初始化

   这一步,我们初始化总结点,并且用next指针将每个单链表的头节点链接成单链表(也就是上图中十字链表的第一行)

复制代码

 1 #region 十字链表中的“行数,列数,非零元素个数” 2         /// <summary> 3         /// 十字链表中的“行数,列数,非零元素个数” 4         /// </summary> 5         /// <param name="rows"></param> 6         /// <param name="cols"></param> 7         /// <param name="count"></param> 8         public void Init(int rows, int cols, int count) 9         {10             var len = Math.Max(rows, cols) + 1;11 12             //从下标1开始算起13             nodes = new Node[len];14 15             //十字链表的总头节点16             nodes[0] = new Node();17 18             nodes[0].rows = rows;19             nodes[0].cols = cols;20             nodes[0].unit = new Unit()21             {22                 value = count,23                 next = null,24             };25 26             //down和right都指向自身27             nodes[0].right = nodes[0];28             nodes[0].down = nodes[0];29 30             var temp = nodes[0];31 32             //初始化多条链表的头结点33             for (int i = 1; i < len; i++)34             {35                 nodes[i] = new Node();36 37                 nodes[i].rows = 0;38                 nodes[i].cols = 0;39                 nodes[i].unit = new Unit()40                 {41                     value = 0,42                     next = temp.unit.next43                 };44 45                 //给上一个节点的next域赋值46                 temp.unit.next = nodes[i];47 48                 //将当前节点作为下一次循环的上一个节点49                 temp = nodes[i];50 51                 nodes[i].right = nodes[i];52                 nodes[i].down = nodes[i];53             }54         }55         #endregion

复制代码

3:插入节点

     根据插入节点的row和col将节点插入到十字链表中指定的位置即可。

复制代码

 1 #region 插入十字链表中 2         /// <summary> 3         /// 插入十字链表中 4         /// </summary> 5         /// <param name="nums">矩阵</param> 6         /// <param name="rows">矩阵的行数</param> 7         /// <param name="cols">矩阵的列数</param> 8         /// <param name="count">非0元素个数</param> 9         /// <returns></returns>10         public Node[] Insert(int[,] nums, int rows, int cols, int count)11         {12             //初始化操作13             Init(rows, cols, count);14 15             //插入操作16             for (int i = 0; i < rows; i++)17             {18                 for (int j = 0; j < cols; j++)19                 {20                     //直插入"非0元素"21                     if (nums[i, j] != 0)22                     {23                         var node = new Node();24 25                         node.rows = i + 1;26                         node.cols = j + 1;27                         node.unit = new Unit()28                         {29                             value = nums[i, j]30                         };31                         node.right = node;32                         node.down = node;33 34                         InsertNode(node);35                     }36                 }37             }38 39             return nodes;40         }41         #endregion

复制代码

4:打印链表

   我们只要遍历每行链表的right指针即可。

复制代码

 1 #region 打印十字链表 2         /// <summary> 3         /// 打印十字链表 4         /// </summary> 5         /// <param name="nodes"></param> 6         public void Print(Node[] nodes) 7         { 8             var head = nodes[0]; 9 10             //遍历每一行的right11             for (int i = 1; i < head.rows + 1; i++)12             {13                 var p = nodes[i];14 15                 while (p.right != nodes[i])16                 {17                     Console.WriteLine("({0},{1})\tval => {2}",18                         p.right.rows,19                         p.right.cols,20                         p.right.unit.value);21 22                     //指向下一个节点23                     p = p.right;24                 }25             }26         }27         #endregion

复制代码

总的代码:

View Code

  1 using System;  2 using System.Collections.Generic;  3 using System.Linq;  4 using System.Text;  5 using System.Diagnostics;  6 using System.Threading;  7 using System.IO;  8   9 namespace ConsoleApplication2 10 { 11     public class Program 12     { 13         public static void Main() 14         { 15             Crosslist crosslist = new Crosslist(); 16  17             int[,] nums = { 18             {2,0,4 }, 19             {0,3,0 }, 20             {0,0,9 } 21            }; 22  23             var nodes = crosslist.Insert(nums, 3, 3, 4); 24  25             crosslist.Print(nodes); 26  27             Console.Read(); 28         } 29     } 30  31     /// <summary> 32     /// 十字链表 33     /// </summary> 34     public class Crosslist 35     { 36         #region 单一节点 37         /// <summary> 38         /// 单一节点 39         /// </summary> 40         public class Node 41         { 42             //行号 43             public int rows; 44  45             //列号 46             public int cols; 47  48             //向下的指针域 49             public Node down; 50  51             //向右的指针域 52             public Node right; 53  54             //单元值(头指针的next和val) 55             public Unit unit; 56         } 57         #endregion 58  59         #region 统一“表头节点”和“非零节点” 60         /// <summary> 61         /// 统一“表头节点”和“非零节点” 62         /// </summary> 63         public class Unit 64         { 65             //表头节点的next域 66             public Node next; 67  68             //非零元素的值 69             public int value; 70         } 71         #endregion 72  73         Node[] nodes; 74  75         #region 十字链表中的“行数,列数,非零元素个数” 76         /// <summary> 77         /// 十字链表中的“行数,列数,非零元素个数” 78         /// </summary> 79         /// <param name="rows"></param> 80         /// <param name="cols"></param> 81         /// <param name="count"></param> 82         public void Init(int rows, int cols, int count) 83         { 84             var len = Math.Max(rows, cols) + 1; 85  86             //从下标1开始算起 87             nodes = new Node[len]; 88  89             //十字链表的总头节点 90             nodes[0] = new Node(); 91  92             nodes[0].rows = rows; 93             nodes[0].cols = cols; 94             nodes[0].unit = new Unit() 95             { 96                 value = count, 97                 next = null, 98             }; 99 100             //down和right都指向自身101             nodes[0].right = nodes[0];102             nodes[0].down = nodes[0];103 104             var temp = nodes[0];105 106             //初始化多条链表的头结点107             for (int i = 1; i < len; i++)108             {109                 nodes[i] = new Node();110 111                 nodes[i].rows = 0;112                 nodes[i].cols = 0;113                 nodes[i].unit = new Unit()114                 {115                     value = 0,116                     next = temp.unit.next117                 };118 119                 //给上一个节点的next域赋值120                 temp.unit.next = nodes[i];121 122                 //将当前节点作为下一次循环的上一个节点123                 temp = nodes[i];124 125                 nodes[i].right = nodes[i];126                 nodes[i].down = nodes[i];127             }128         }129         #endregion130 131         #region 在指定的“行,列”上插入节点132         /// <summary>133         /// 在指定的“行,列”上插入节点134         /// </summary>135         /// <param name="node"></param>136         /// <returns></returns>137         public void InsertNode(Node node)138         {139             //先定位行140             Node pnode = nodes[node.rows];141 142             //在指定的“行”中找,一直找到该行最后一个节点(right指针指向自己的为止)143             while (pnode.right != nodes[node.rows] && pnode.right.cols < node.cols)144                 pnode = pnode.right;145 146             //将最后一个节点的right指向插入节点的right,以此达到是循环链表147             node.right = pnode.right;148 149             //将插入节点给最后一个节点的right指针上150             pnode.right = node;151 152             //再定位列153             pnode = nodes[node.cols];154 155             //同理156             while (pnode.down != nodes[node.cols] && pnode.down.rows < node.rows)157             {158                 pnode = pnode.down;159             }160 161             node.down = pnode.down;162             pnode.down = node;163         }164         #endregion165 166         #region 插入十字链表中167         /// <summary>168         /// 插入十字链表中169         /// </summary>170         /// <param name="nums">矩阵</param>171         /// <param name="rows">矩阵的行数</param>172         /// <param name="cols">矩阵的列数</param>173         /// <param name="count">非0元素个数</param>174         /// <returns></returns>175         public Node[] Insert(int[,] nums, int rows, int cols, int count)176         {177             //初始化操作178             Init(rows, cols, count);179 180             //插入操作181             for (int i = 0; i < rows; i++)182             {183                 for (int j = 0; j < cols; j++)184                 {185                     //直插入"非0元素"186                     if (nums[i, j] != 0)187                     {188                         var node = new Node();189 190                         node.rows = i + 1;191                         node.cols = j + 1;192                         node.unit = new Unit()193                         {194                             value = nums[i, j]195                         };196                         node.right = node;197                         node.down = node;198 199                         InsertNode(node);200                     }201                 }202             }203 204             return nodes;205         }206         #endregion207 208         #region 打印十字链表209         /// <summary>210         /// 打印十字链表211         /// </summary>212         /// <param name="nodes"></param>213         public void Print(Node[] nodes)214         {215             var head = nodes[0];216 217             //遍历每一行的right218             for (int i = 1; i < head.rows + 1; i++)219             {220                 var p = nodes[i];221 222                 while (p.right != nodes[i])223                 {224                     Console.WriteLine("({0},{1})\tval => {2}",225                         p.right.rows,226                         p.right.cols,227                         p.right.unit.value);228 229                     //指向下一个节点230                     p = p.right;231                 }232             }233         }234         #endregion235     }236 }

 

点击查看更多内容
TA 点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
移动开发工程师
手记
粉丝
18
获赞与收藏
134

关注作者,订阅最新文章

阅读免费教程

  • 推荐
  • 评论
  • 收藏
  • 共同学习,写下你的评论
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消