LinuxC语言中的数组和rand函数和宏定义和全排列

2024-06-08 07:48

本文主要是介绍LinuxC语言中的数组和rand函数和宏定义和全排列,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 一、数组也是一种复合数据类型
    • 1.数组的定义
    • 2.数组中的元素通过下标(或者叫索引,Index) 来访问。
    • 3.C语言中后缀运算符的优先级高于前缀运算符
    • 4.数组下标也可以是表达式,但表达式的值必须是整型的
    • 5.数组的初始化
    • 6.定义和访问数组——遍历的使用
    • 7.数组与结构体的显著区别
  • 二、数组的应用例子:统计随机数
    • 1.rand函数介绍
      • (i)预处理器的作用:
      • (ii)那么用 #define 定义的常量和第 3 节 “数据类型标志enum”讲的枚举常量有什么区别呢?
      • (iii)硬编码的含义
    • 2.习题
  • 三、数组的应用例子:直方图
    • 1.直方图的含义
    • 2.用数组的效率低与效率高的方法比较
    • 3.习题
  • 四、字符串
    • 1.字符串的定义
    • 2.字符数组可用一个字符串字面值来初始化
    • 3. printf 函数的格式化字符串中%s 占位符的使用注意事项
  • 五、多维数组
    • 1.多维数组的定义
    • 2.多维数组的初始化
    • 3.最后,综合本章的知识,我们来写一个最简单的小游戏--剪刀石头布:

一、数组也是一种复合数据类型

1.数组的定义

(1)
在这里插入图片描述
说明:

  • 和结构体成员类似,数组 count 的4个元素的存储空间也是相邻的。
  • 结构体成员可以是基本数据类型,也可以是复合数据类型,数组中的元素也是如此。

(2)根据组合规则,我们可以定义一个由4个结构体元素组成的数组:
在这里插入图片描述
也可以定义一个包含数组成员的结构体:
在这里插入图片描述

2.数组中的元素通过下标(或者叫索引,Index) 来访问。

(1)
在这里插入图片描述
在这里插入图片描述

3.C语言中后缀运算符的优先级高于前缀运算符

在这里插入图片描述

4.数组下标也可以是表达式,但表达式的值必须是整型的

在这里插入图片描述

5.数组的初始化

在这里插入图片描述

6.定义和访问数组——遍历的使用

在这里插入图片描述

7.数组与结构体的显著区别

(1)显著不同在于:
在这里插入图片描述
(2)
在这里插入图片描述
说明:

  • 编译器也不会报错,但这样写并不是传一个数组类型参数的意思。对于数组类型有一条特殊规则:数组类型做右值使用时,自动转换成指向数组首元素的指针。 所以上面的函数调用其实是传一个指针类型的参数,而不是数组类型的参数。
  • 接下来的几章里有的函数需要访问数组,我们就把数组定义为全局变量给函数访问。 等以后讲了指针再使用传参的办法。
  • 这也解释了为什么数组类型不能相互赋值或初始化,例如上面提到的 a = b 这个表达式, a 和 b 都是数组类型的变量,但是 b 做右值使用,自动转换成指针类型,而左边仍然是数组类型,所以编译器报的错是 error: incompatible types in assignment 。

二、数组的应用例子:统计随机数

1.rand函数介绍

(1)
在这里插入图片描述
(2)完整的程序如下:
在这里插入图片描述
说明:

  • 这里介绍一种新的语法:用 #define 定义一个常量。
  • 实际上编译器的工作分为两个阶段,先是预处理(Preprocess) 阶段,然后才是编译阶段,
    用 gcc 的 -E 选项可以看到预处理之后、编译之前的程序,例如:
    在这里插入图片描述
    说明:

(i)预处理器的作用:

  • 一是把头文件 stdio.h 和 stdlib.h 在代码中展开;
  • 二是把 #define 定义的标识符 N 替换成它的定义20(在代码中做了三处替换,分别位于数组的定义中和两个函数中) 。

(ii)那么用 #define 定义的常量和第 3 节 “数据类型标志enum”讲的枚举常量有什么区别呢?

  • define 不仅用于定义常量,也可以定义更复杂的语法结构,称为宏(Macro) 定义。
  • define 定义是在预处理阶段处理的,而枚举是在编译阶段处理的。
    在这里插入图片描述
    (3)
    在这里插入图片描述
    毕竟我们的样本太少了,才20个数,如果样本足够多,比如说100000个数,统计一下其中每个数字出现的次数也许能说明问题。但总不能把100000个数都打印出来然后挨个去数吧?
    我们需要写一个函数统计每个数字出现的次数。
    完整的程序如下:
    在这里插入图片描述
    说明:

(iii)硬编码的含义

在这里插入图片描述

2.习题

在这里插入图片描述
在这里插入图片描述

三、数组的应用例子:直方图

1.直方图的含义

在这里插入图片描述

2.用数组的效率低与效率高的方法比较

(1)效率低的方法:
在这里插入图片描述
说明:
(a)
在这里插入图片描述
(b)尽管上面的方法可以准确地得到统计结果,但是效率很低,这100000个随机数需要从头到尾检查十遍,每一遍检查只统计一种数字的出现次数。

(2)效率高的方法
在这里插入图片描述
说明:
在这里插入图片描述
在这里插入图片描述

3.习题

在这里插入图片描述

int main(void)
{int i,histogram[10]={0};gen_random(10);for(i=0;i<N;i++)histogram[a[i]]++;int sum=0;for(int i=0;i<10;i++){sum+=histogram[i];printf("%d\t",i);}while(sum!=0){for(int i=0;i<10;i++){if(histogram[i]!=0){printf("*\t");histogram[i]--;sum--;}elseprintf(" \t")}printf("\n");}		return 0;
}

在这里插入图片描述
在这里插入图片描述
如果还不懂的话:
参考:https://blog.csdn.net/qq_33724710/article/details/50924097(有输出结果)
参考:https://blog.csdn.net/sunbingxi_/article/details/6125426(感觉是元老级答案)
参考:https://segmentfault.com/a/1190000000725176(答案都有,但不完全对)

第一个问题的答案如下:

void prem(int offset);
void print();
void pozition(int i,int offset);#define N 3
int a[N]={0};int main(void)
{int i;for(i=0;i<N;i++)a[i]=i;prem(0);return 0;
}void prem(int offset)
{if(offset==N-1){print();return;/* 每次输出以后,return到main函数,使得offset=0*/}else{for(int i=offset;i<N;i++){pozition(i,offset);prem(offset+1);pozition(i,offset);}}
}void print()
{int i;for(i=0;i<N;i++)printf("%d ",a[i]);printf("\n");
}void pozition(int i,int offset)
{int temp;temp=a[i];a[i]=a[offset];a[offset]=temp;
}解释如下:
(1)如果对0 1做全排列(N=2),那么全排列的结果是:
0 1
1 0
(2)现在来看程序,主要的全排列程序是perm()函数。
第一次else的结果是:i=offset=0;
positon函数交换的是0和0,offset=1,,由于N=2的原因,直接print,这个时候打印 0 1;
然后返回到main函数,从main函数回来之后(也就是prem(offset+1)之后),offset=0,接着再次交换,这时i=0,offset=0,也就没啥交换的。
接着,i++之后,i=1,然后交换i=1和offset=0的位置,也就是变成了1 0;
prem函数过后,offset=1,再次print出来,就是 1 0了;
接着,i=1,offset=0,交换后,又回到了0 1;但是i=2了,不满足for循环了,跳出。
(3)这个全排列程序基本就是在这个for循环内变化,所以当for循环和if语句不满足的时候,就是得到所有全排列的时候。
但是每次print的时候,都是满足if语句的时候,但是却把offset又刷成了0。

第二个问题的答案如下:

需要改动的地方有:
#define N 3
#define M 2void prem(int offset)
{if(offset==M-1){print();return;/* 每次输出以后,return到main函数,使得offset=0*/}else{for(int i=offset;i<M;i++){pozition(i,offset);prem(offset+1);pozition(i,offset);}}
}void print()
{int i;for(i=0;i<M;i++)printf("%d ",a[i]);printf("\n");
}

第三个问题的答案:
假设有一个两两元素互不相同的N长数组a,从数组尾端依次取M个数。
数组b[]得重新定义。

void comb(int n, int m)
{int i;if (m == 0) {print();return;} else {for (int i = n-1; i >= 0; --i){b[m-1] = a[i];comb(i, m-1);}}
}

四、字符串

1.字符串的定义

(1)字符串可以看作一个数组,它的每个元素是字符型的。
例如字符串 “Hello, world.\n” 图示如下:
在这里插入图片描述
注意每个字符末尾都有一个字符 ‘\0’ 做结束符,这里的 \0 是ASCII码的八进制表示,也就
是ASCII码为0的Null字符,在C语言中这种字符串也称为以零结尾的字符串(Null-terminated String)。

(2)数组元素可以通过数组名加下标的方式访问,而字符串字面值也可以像数组名一样
使用,可以加下标访问其中的字符:
在这里插入图片描述

2.字符数组可用一个字符串字面值来初始化

在这里插入图片描述
(1)str 的后四个元素没有指定,自动初始化为0,即Null字符.。
(2)在本书中只要是以Null字符结尾的一串字符都叫字符串,不管是像 str 这样的数组,还是像 “Hello” 这样的字符串字面值。
(3)如果用于初始化的字符串字面值比数组还长,eg:
在这里插入图片描述
则数组 str 只包含字符串的前10个字符,不包含Null字符,这种情况编译器会给出警告。

(4)如果要用一个字符串字面值准确地初始化一个字符数组,最好的办法是不指定数组的长度,让编译器自己计算:
在这里插入图片描述
(5)有一种情况需要特别注意,如果用于初始化的字符串字面值比数组刚好长出一个Null字符的长度(打印字符串可能会出错)
在这里插入图片描述

3. printf 函数的格式化字符串中%s 占位符的使用注意事项

在这里插入图片描述

五、多维数组

1.多维数组的定义

(1)一个数组的元素可以是另外一个数组,这样就构成了多维数组(Multi-dimensional Array) 。例如定义并初始化一个二维数组:
在这里插入图片描述
(2)从概念模型和物理模型上去理解多维数组 (从物理模型上去理解,是真的爽)
在这里插入图片描述

2.多维数组的初始化

(1)多维数组的初始化
在这里插入图片描述
(2)如果是多维字符数组,也可以嵌套使用字符串字面值做Initializer
在这里插入图片描述
多维字符数组的解释如下:
在这里插入图片描述

3.最后,综合本章的知识,我们来写一个最简单的小游戏--剪刀石头布:

(1)代码如下
在这里插入图片描述
解释如下:
在这里插入图片描述
(2)练习
在这里插入图片描述

                      胜 负 平 胜 负
man-computer:        -2 -1 0 1 2
man-computer+4:       4 2 3 4 5 6 
(man-computer+4)%:    3 2 0 1 2 0 
(man-computer+4)%3-1: 1 -1 0 1 -1

剪刀石头布相生相克,形成一个环,凡是具有环的特性的数学模型都可以考虑用取模运算, 首先确定了man-computer和%3,然后再调整其它常数得到normalized的结果。

这篇关于LinuxC语言中的数组和rand函数和宏定义和全排列的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1041584

相关文章

C语言中联合体union的使用

本文编辑整理自: http://bbs.chinaunix.net/forum.php?mod=viewthread&tid=179471 一、前言 “联合体”(union)与“结构体”(struct)有一些相似之处。但两者有本质上的不同。在结构体中,各成员有各自的内存空间, 一个结构变量的总长度是各成员长度之和。而在“联合”中,各成员共享一段内存空间, 一个联合变量

大语言模型(LLMs)能够进行推理和规划吗?

大语言模型(LLMs),基本上是经过强化训练的 n-gram 模型,它们在网络规模的语言语料库(实际上,可以说是我们文明的知识库)上进行了训练,展现出了一种超乎预期的语言行为,引发了我们的广泛关注。从训练和操作的角度来看,LLMs 可以被认为是一种巨大的、非真实的记忆库,相当于为我们所有人提供了一个外部的系统 1(见图 1)。然而,它们表面上的多功能性让许多研究者好奇,这些模型是否也能在通常需要系

【操作系统】信号Signal超详解|捕捉函数

🔥博客主页: 我要成为C++领域大神🎥系列专栏:【C++核心编程】 【计算机网络】 【Linux编程】 【操作系统】 ❤️感谢大家点赞👍收藏⭐评论✍️ 本博客致力于知识分享,与更多的人进行学习交流 ​ 如何触发信号 信号是Linux下的经典技术,一般操作系统利用信号杀死违规进程,典型进程干预手段,信号除了杀死进程外也可以挂起进程 kill -l 查看系统支持的信号

java中查看函数运行时间和cpu运行时间

android开发调查性能问题中有一个现象,函数的运行时间远低于cpu执行时间,因为函数运行期间线程可能包含等待操作。native层可以查看实际的cpu执行时间和函数执行时间。在java中如何实现? 借助AI得到了答案 import java.lang.management.ManagementFactory;import java.lang.management.Threa

通俗范畴论4 范畴的定义

注:由于CSDN无法显示本文章源文件的公式,因此部分下标、字母花体、箭头表示可能会不正常,请读者谅解 范畴的正式定义 上一节我们在没有引入范畴这个数学概念的情况下,直接体验了一个“苹果1”范畴,建立了一个对范畴的直观。本节我们正式学习范畴的定义和基本性质。 一个范畴(Category) C𝐶,由以下部分组成: 数据: 对象(Objects):包含若干个对象(Objects),这些

人工和AI大语言模型成本对比 ai语音模型

这里既有AI,又有生活大道理,无数渺小的思考填满了一生。 上一专题搭建了一套GMM-HMM系统,来识别连续0123456789的英文语音。 但若不是仅针对数字,而是所有普通词汇,可能达到十几万个词,解码过程将非常复杂,识别结果组合太多,识别结果不会理想。因此只有声学模型是完全不够的,需要引入语言模型来约束识别结果。让“今天天气很好”的概率高于“今天天汽很好”的概率,得到声学模型概率高,又符合表达

SQL Server中,isnull()函数以及null的用法

SQL Serve中的isnull()函数:          isnull(value1,value2)         1、value1与value2的数据类型必须一致。         2、如果value1的值不为null,结果返回value1。         3、如果value1为null,结果返回vaule2的值。vaule2是你设定的值。        如

C语言 将“China”译成密码

将“China”译成密码,密码规律是:用原来的字母后面的第4个字母代替原来的字母。例如,字母“A”后面的第4个字母是“E”,用“E”代替“A”。因此,“China”应译为“Glmre”。编译程序用付赋初值的方法使c1,c2,c3,c4,c5这五个变量的值分别为“C”,“h”,“i”,“n”,“a”,经过运算,使c1,c2,c3,c4,c5分别变成“G”,“l”,“m”,“r”,“e”。分别用put

tf.split()函数解析

API原型(TensorFlow 1.8.0): tf.split(     value,     num_or_size_splits,     axis=0,     num=None,     name='split' ) 这个函数是用来切割张量的。输入切割的张量和参数,返回切割的结果。  value传入的就是需要切割的张量。  这个函数有两种切割的方式: 以三个维度的张量为例,比如说一

C语言入门系列:探秘二级指针与多级指针的奇妙世界

文章目录 一,指针的回忆杀1,指针的概念2,指针的声明和赋值3,指针的使用3.1 直接给指针变量赋值3.2 通过*运算符读写指针指向的内存3.2.1 读3.2.2 写 二,二级指针详解1,定义2,示例说明3,二级指针与一级指针、普通变量的关系3.1,与一级指针的关系3.2,与普通变量的关系,示例说明 4,二级指针的常见用途5,二级指针扩展到多级指针 小结 C语言的学习之旅中,二级