IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 浙江大学慕课《数据结构》学习笔记_1.1 什么是数据结构 -> 正文阅读

[数据结构与算法]浙江大学慕课《数据结构》学习笔记_1.1 什么是数据结构

第一讲 基本概念


1.1 什么是数据结构


官方统一定义—— 没有……

“数据结构是数据对象,以及存在于该对象的实例和组成实例的数据元素之间的各种联系。这些联系可以通过定义相关的函数来给出。”
——Sartaj Sahni,《数据结构、算法与应用》

“数据结构是ADT(抽象数据类型 Abstract Data Type)的物理实现。”
——Clifford A.Shaffer,《数据结构与算法分析》

“数据结构(data structure)是计算机中存储、组织数据的方式。通常情况下,精心选择的数据结构可以带来最优效率的算法。”
——中文维基百科

如何在书架上摆放图书

例:在小书架上,在家用书柜上,在图书馆书柜上。

图书的摆放要使得2个相关操作方便实现:

  • 操作1:新书怎么插入?

  • 操作2:怎么找到某本指定的书?

方法1:随便放

  • 操作1:新书怎么插入?
    哪里有空放哪里,一步到位!

  • 操作2:怎么找到某本指定的书?
    ……累死

方法2:按照书名的拼音字母顺序排放

  • 操作1:新书怎么插入?
    新进一本《阿Q正传》……

  • 操作2:怎么找到某本指定的书?
    二分查找!

方法3:把书架划分成几块区域,每块区域指定摆放某种类别的图书;在每种类别内,按照书名的拼音字母顺序排放

  • 操作1:新书怎么插入?
    先定类别,二分查找确定位置,移出空位

  • 操作2:怎么找到某本指定的书?
    先定类别,再二分查找

  • 问题:空间如何分配?类别应该分多细?
    空间不能分太小,小了新书就插不进去;空间不能分太大,大了就空间利用率就太少。类别不能分太粗略,不然查书时还是要查很多书;类别不能分太细致,不然查书时就要查很多类。

结论:解决问题方法的效率,跟数据的组织方式有关


实现函数传入参数N,打印1到N的正整数

循环实现:

void PrintN ( int N )
{ 
	int i;
	for ( i=1; i<=N; i++ )
		printf("%d\n", i );
	return;
} 

递归实现:

void PrintN ( int N )
{ 
	if ( N )
	{
		PrintN( N – 1 ); 
		printf("%d\n", N );
	}
	return;
} 

测试代码:

#include <stdio.h>
void PrintN ( int N );
int main ()
{
	int N;
	scanf("%d", &N);
	PrintN( N );
	return 0;
}

依次输入100,1000,10000,100000…

循环实现结果:
循环实现结果
递归实现结果:
递归实现结果
总结:解决问题方法的效率,跟空间的利用效率有关


计算给定多项式在定点x的值

给定多项式
根据所给式子直接实现:

double f( int n, double a[], double x )
{
	int i;
	double p = a[0];
	for ( i=1; i<=n; i++ )
		p += (a[i] * pow(x, i)); 
	return p;
} 

秦九韶算法实现:
秦九韶

double f( int n, double a[], double x )
{ 
	int i;
	double p = a[n];
	for ( i=n; i>0; i-- )
		p = a[i-1] + x*p;
	return p;
}

这两种函数的优劣可以通过运行时间判断:
clock():捕捉从程序开始运行到clock()被调用时所耗费的时间。这个时间单位是clock tick,即“时钟打点”。

常数CLK_TCK(或CLOCKS_PER_SEC):机器时钟每秒所走的时钟打点数,该常数在每台机器可能不一样。

#include <stdio.h>
#include <time.h>
clock_t start, stop;
/* clock_t是clock()函数返回的变量类型 */
double duration;
/* 记录被测函数运行时间,以秒为单位 */
int main ()
{ 
	/* 不在测试范围内的准备工作写在clock()调用之前 */
	start = clock(); /* 开始计时 */
	MyFunction(); /* 把被测函数加在这里 */
	stop = clock(); /* 停止计时 */
	duration = ((double)(stop - start))/CLK_TCK;
	/* 计算运行时间 */
	/* 其他不在测试范围的处理写在后面,例如输出duration的值 */
	return 0;
} 

写程序计算给定多项式在给定点 x = 1.1 处的值 f(1.1)

两种思路函数的实现:

double f1( int n, double a[], double x )
{ 
	int i;
	double p = a[0];
	for ( i=1; i<=n; i++ )
		p += (a[i] * pow(x, i)); 
	return p;
}
double f2( int n, double a[], double x )
{ 
	int i;
	double p = a[n];
	for ( i=n; i>0; i-- )
		p = a[i-1] + x*p;
	return p;
}

测试代码:

#include <stdio.h>
#include <time.h>
#include <math.h>
clock_t start, stop; 
double duration;
#define MAXN 10 /* 多项式最大项数,即多项式阶数+1 */
double f1( int n, double a[], double x );
double f2( int n, double a[], double x );
int main ()
{ 
	int i;
	double a[MAXN]; /* 存储多项式的系数 */
	for ( i=0; i<MAXN; i++ ) a[i] = (double)i; 
	start = clock();
	f1(MAXN-1, a, 1.1); 
	stop = clock();
	duration = ((double)(stop - start))/CLK_TCK; 
	printf("ticks1 = %f\n", (double)(stop - start));
	printf("duration1 = %6.2e\n", duration);
	start = clock();
	f2(MAXN-1, a, 1.1); 
	stop = clock();
	duration = ((double)(stop - start))/CLK_TCK; 
	printf("ticks2 = %f\n", (double)(stop - start));
	printf("duration2 = %6.2e\n", duration);
	return 0;
}

结果:
结果
原因:两种函数的运行时间都过短,无法精确捕捉到运行时间。

解决方案:让被测函数重复运行充分多次,使得测出的总的时钟打点间隔充分长,最后计算被测函数平均每次运行的时间即可!

测试代码:

#include <stdio.h>
#include <time.h>
#include <math.h>
……
#define MAXK 1e7 /* 被测函数最大重复调用次数 */
……
int main ()
{ ……
start = clock();
for ( i=0; i<MAXK; i++ ) /* 重复调用函数以获得充分多的时钟打点数*/
f1(MAXN-1, a, 1.1); 
stop = clock();
duration = ((double)(stop - start))/CLK_TCK/MAXK; /* 计算函数单次运行的时间 */
printf("ticks1 = %f\n", (double)(stop - start));
printf("duration1 = %6.2e\n", duration);
……
return 0;
}

结果:
结果
总结:解决问题方法的效率,跟算法的巧妙程度有关


所以到底什么是数据结构

数据对象在计算机中的组织方式

  • 逻辑结构
  • 物理存储结构

数据对象必定与一系列加在其上的操作相关联。

完成这些操作所用的方法就是算法


抽象数据类型(Abstract Data Type)

将这个名词拆为”数据类型“和”抽象“来解释。

数据类型

  • 数据对象集
  • 数据集合相关联的操作集

抽象:描述数据类型的方法不依赖于具体实现

  • 与存放数据的机器无关
  • 与数据存储的物理结构无关
  • 与实现操作的算法和编程语言均无关

只描述数据对象集和相关操作集“是什么”,并不涉及“如何做到”的问题。


“矩阵”的抽象数据类型定义

类型名称:矩阵(Matrix)
数据对象集:一个M×N的矩阵AM×N = (aij) (i=1, …, M; j=1, …, N )由M×N个三
元组< a, i, j >构成,其中a是矩阵元素的值,i是元素所在的行号,j是元素所在的列号。
操作集:对于任意矩阵A、B、C ∈ Matrix,以及整数i、j、M、N

  • Matrix Create( int M, int N ):返回一个M×N的空矩阵;
  • int GetMaxRow( Matrix A ):返回矩阵A的总行数;
  • int GetMaxCol( Matrix A ):返回矩阵A的总列数;
  • ElementType GetEntry( Matrix A, int i, int j ):返回矩阵A的第i行、第j列的元素;
  • Matrix Add( Matrix A, Matrix B ):如果A和B的行、列数一致,则返回矩阵C=A+B,否则返回错误标志;
  • Matrix Multiply( Matrix A, Matrix B ):如果A的列数等于B的行数,则返回矩阵C=AB,否则返回错误标志;
    ……

抽象在于
不知道矩阵内元素的数据类型,是用一维数组、二维数组还是十字链表等来实现矩阵,C=A+B是先加行,还是先加列。整个数据类型及其操作是用什么语言来实现的。

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-07-15 23:50:43  更:2021-07-15 23:50:54 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年12日历 -2024/12/27 10:22:24-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码
数据统计