基于C#实现三元组

我们知道矩阵是一个非常强大的数据结构,在动态规划以及各种图论算法上都有广泛的应用,当然矩阵有着不足的地方就是空间和时间复杂度都维持在 N2 上,比如 1w 个数字建立一个矩阵,在内存中会占用 1w*1w=1 亿的类型空间,这时就会遇到 outofmemory。。。那么面临的一个问题就是如何来压缩矩阵,当然压缩的方式有很多种,这里就介绍一个顺序表的压缩方式:三元组。

一、三元组

有时候我们的矩阵中只有零星的一些非零元素,其余的都是零元素,那么我们称之为稀疏矩阵,当然没有绝对的说有多少个零元素才算稀疏。
image.png
针对上面的这个无规律的存放非零元素,三元组提出了一种方法,就是仅仅记录矩阵中的非零元素以及它的行,列以及值 N(x,y,v)构成的一个三元组,标识一个稀疏矩阵的话,还要记录该矩阵的阶数,这样我们就将一个二维的变成了一个一维,极大的压缩的存储空间,这里要注意的就是,三元组的构建采用“行“是从上到下,“列”也是从左到右的方式构建的顺序表。
image.png

 /// <summary>/// 三元组/// </summary>public class Unit{public int x;public int y;public int element;}/// <summary>/// 标识矩阵/// </summary>public class SPNode{//矩阵总行数public int rows;//矩阵总列数public int cols;//非零元素的个数public int count;//矩阵中非零元素public List<Unit> nodes = new List<Unit>();}

其实说到这里也就差不多了,我们只要知道三元组是用来做矩阵压缩的一个顺序存储方式即可,然后知道怎么用三元组表来做一些常规的矩阵运算,好了,既然说已经做成线性存储了,那就做个“行列置换”玩玩。

二、行列置换

做行列置换很容易,也就是交换"非零元素"的(x,y)坐标,要注意的就是,原先我们的三元组采用的是”行优先“,所以在做转置的时候需要遵循"列优先“。
image.png

 /// <summary>/// 行转列运算/// </summary>/// <param name="spNode"></param>/// <returns></returns>public SPNode ConvertSpNode(SPNode spNode){//矩阵元素的x和y坐标进行交换SPNode spNodeLast = new SPNode();//行列互换spNodeLast.rows = spNode.cols;spNodeLast.cols = spNode.rows;spNodeLast.count = spNode.count;//循环原矩阵的列数 (行列转换)for (int col = 0; col < spNode.cols; col++){//循环三元组行的个数for (int sp = 0; sp < spNode.count; sp++){var single = spNode.nodes[sp];//找到三元组中存在的相同编号if (col == single.y){spNodeLast.nodes.Add(new Unit(){x = single.y,y = single.x,element = single.element});}}}return spNodeLast;}

最后是总的代码:

 using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Diagnostics;using System.Threading;using System.IO;namespace ConsoleApplication2{public class Program{public static void Main(){Martix martix = new Martix();//构建三元组var node = martix.Build();foreach (var item in node.nodes){Console.WriteLine(item.x + "\t" + item.y + "\t" + item.element);}Console.WriteLine("******************************************************");var mynode = martix.ConvertSpNode(node);foreach (var item in mynode.nodes){Console.WriteLine(item.x + "\t" + item.y + "\t" + item.element);}Console.Read();}}public class Martix{/// <summary>/// 三元组/// </summary>public class Unit{public int x;public int y;public int element;}/// <summary>/// 标识矩阵/// </summary>public class SPNode{//矩阵总行数public int rows;//矩阵总列数public int cols;//非零元素的个数public int count;//矩阵中非零元素public List<Unit> nodes = new List<Unit>();}/// <summary>/// 构建一个三元组/// </summary>/// <returns></returns>public SPNode Build(){SPNode spNode = new SPNode();//遵循行优先的原则spNode.nodes.Add(new Unit() { x = 0, y = 0, element = 8 });spNode.nodes.Add(new Unit() { x = 1, y = 2, element = 1 });spNode.nodes.Add(new Unit() { x = 2, y = 3, element = 6 });spNode.nodes.Add(new Unit() { x = 3, y = 1, element = 4 });//4阶矩阵spNode.rows = spNode.cols = 4;//非零元素的个数spNode.count = spNode.nodes.Count;return spNode;}/// <summary>/// 行转列运算/// </summary>/// <param name="spNode"></param>/// <returns></returns>public SPNode ConvertSpNode(SPNode spNode){//矩阵元素的x和y坐标进行交换SPNode spNodeLast = new SPNode();//行列互换spNodeLast.rows = spNode.cols;spNodeLast.cols = spNode.rows;spNodeLast.count = spNode.count;//循环原矩阵的列数 (行列转换)for (int col = 0; col < spNode.cols; col++){//循环三元组行的个数for (int sp = 0; sp < spNode.count; sp++){var single = spNode.nodes[sp];//找到三元组中存在的相同编号if (col == single.y){spNodeLast.nodes.Add(new Unit(){x = single.y,y = single.x,element = single.element});}}}return spNodeLast;}}}

image.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/216758.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Leetcode—1457.二叉树中的伪回文路径【中等】

2023每日刷题&#xff08;四十&#xff09; Leetcode—1457.二叉树中的伪回文路径 实现代码 /*** Definition for a binary tree node.* struct TreeNode {* int val;* struct TreeNode *left;* struct TreeNode *right;* };*/ int record[10] {0};int accumula…

【Spring集成MyBatis】MyBatis的多表查询

文章目录 1. 一对一什么是一对一User、Order类及Mapper&#xff0c;User、Order表一对一操作的实现一对一操作实现的第二种方式 2. 一对多什么是一对多一对多操作实现 3. 多对多什么是多对多多对多的实现 4. 小结 1. 一对一 什么是一对一 一对一指的是表与表之间通过外键进行…

PIL.UnidentifiedImageError: cannot identify image file ...

按照网上搜的重新安装pillow库&#xff0c;对我这个不适用。我的解决方法是把有问题的图片删掉。

bugkuctf--Crypto--抄错的字符

抄错的字符 描  述: 老师让小明抄写一段话&#xff0c;结果粗心的小明把部分数字抄成了字母&#xff0c;还因为强迫症把所有字母都换成大写。你能帮小明恢复并解开答案吗&#xff1a;QWIHBLGZZXJSXZNVBZW 这里其实是base64加密只是更换了字母大写&#xff0c;还有数字 QW…

公众号留言功能还有可能开放吗?

为什么公众号没有留言功能&#xff1f;2018年2月12日&#xff0c;TX新规出台&#xff1a;根据相关规定和平台规则要求&#xff0c;我们暂时调整留言功能开放规则&#xff0c;后续新注册帐号无留言功能。这就意味着2018年2月12日号之后注册的公众号不论个人主体还是组织主体&…

【网络】传输层 --- 详解TCP协议

目录 一、协议段格式及其策略确认应答(ACK)机制6个标志位超时重传流量控制滑动窗口1、先谈滑动窗口一般情况2、再谈特殊窗口 拥塞控制拥塞窗口 延迟应答&&捎带应答面向字节流粘包问题 二、三次握手和四次挥手三次握手为什么是3次&#xff1f;不是2、4、5、6次呢 四次挥…

医学检验科LIS系统源码 样本采集、检验、分析

LIS把检验、检疫、放免、细菌微生物及科研使用的各类分析仪器&#xff0c;通过计算机联网&#xff0c;实现各类仪器数据结果的实时自动接收、自动控制及综合分析&#xff1b;系统可与条码设备配套使用&#xff0c;自动生成条码&#xff0c;减少实验室信息传递中人为因素导致的误…

FreeRTOS学习之路,以STM32F103C8T6为实验MCU(2-5:队列)

学习之路主要为FreeRTOS操作系统在STM32F103&#xff08;STM32F103C8T6&#xff09;上的运用&#xff0c;采用的是标准库编程的方式&#xff0c;使用的IDE为KEIL5。 注意&#xff01;&#xff01;&#xff01;本学习之路可以通过购买STM32最小系统板以及部分配件的方式进行学习…

HTML4总结

一、前序知识 1. 认识两位先驱 2. 计算机基础知识 1. 计算机俗称电脑&#xff0c;是现代一种用于高速计算的电子计算机器&#xff0c;可以进行数值计算、逻辑计算&#xff0c;还 具有存储记忆功能。 2. 计算机由 硬件 软件 成&#xff1a; 硬件&#xff1a;看得见摸得着…

蓝牙运动耳机哪个好?蓝牙运动耳机排行榜前十名

​在运动中&#xff0c;音乐可以激发你的热情和动力&#xff0c;而一款好的运动耳机则可以让你更好地享受音乐。然而&#xff0c;市面上的运动耳机品牌和型号众多&#xff0c;质量参差不齐。所以&#xff0c;今天精选了5款市面上比较优秀的运动耳机给大家参考&#xff0c;是你运…

论文阅读——DDeP(cvpr2023)

分割标签耗时且贵&#xff0c;所以常常使用预训练提高分割模型标签有效性&#xff0c;反正就是&#xff0c;需要一个预训练分割模型。典型的分割模型encoder部分通过分类任务预训练&#xff0c;decoder部分参数随机初始化。作者认为这个方法次优&#xff0c;尤其标签比较少的情…

C语言每日一题(37)两数相加

力扣网 2 两数相加 题目描述 给你两个 非空 的链表&#xff0c;表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的&#xff0c;并且每个节点只能存储 一位 数字。 请你将两个数相加&#xff0c;并以相同形式返回一个表示和的链表。 你可以假设除了数字 0 之外&a…