C语言学习笔记之程序结构

2023-12-04 13:18

本文主要是介绍C语言学习笔记之程序结构,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

程序结构

全局变量

全局变量

1.定义在函数外部的变量就叫全局变量
2.全局变量具有全局的生存期和作用域
3.它们与任何函数都无关
4.在任何函数内部都可以使用它们

全局变量初始化

1.没有做初始化的全局变量会得到0值
2.指针会得到NULL值
3.只能用编译时刻已知的值来初始化全局变量
不能实现:

int a=10;
int b=a;
int main(){...

但可以:

const int a=10;
int b=a;
int main(){...

4.它们的初始化发生在main函数之前
5.如果在子函数内定义了一个与全局变量相同的变量,全局变量将被隐藏。

静态本地变量

1.在本地变量定义时加上static修饰符就成为静态本地变量
2.当函数离开时,静态本地变量会继续存在并保持其值
3.静态本地变量的初始化只会在第一次进入函数时做,以后进入函数时会保持上次离开时的值。

补充:
1.静态本地变量实际上是特殊的全局变量
2.它们位于相同的内存区域
3.静态本地变量具有全局的生存期,函数内的局部作用
——static在这里的意思是局部作用域(本地可访问)
测试代码:

#include<stdio.h>
int a=2;
int main(){int k=0;static int b=1;printf("&a=%p\n",&a);printf("&b=%p\n",&b);printf("&k=%p\n",&k);return 0;
} 

结果:

&a=0000000000403010
&b=0000000000403014
&k=000000000062FE1C

返回指针的函数,使用全局变量的贴士

*返回指针的函数

1.返回本地变量的地址是危险的
2.返回全局变量或静态本地变量的地址是安全的
3.返回在函数内malloc的内存是安全的,但是很容易造成问题,
4.最好的做法是返回传入的指针

#include<stdio.h>
int* f(void);
void g(void);
int main(){int *p=f();printf("*p=%d\n",*p);g();printf("*p=%d\n",*p);return 0;
} 
int* f(void){int a=1;printf("&a=%d\n",&a);return &a;
}
void g(void){int b=2;printf("b=%d\n",b);printf("&b=%d\n",&b);
}

结果:

&a=6487516
*p=1
b=2
&b=6487516
*p=2

因为在函数结束后本地变量会消亡!!

tips

1.不要使用全局变量在函数间传递参数和结果
2.尽量避免使用全局变量
3.使用全局变量和静态本地变量的函数是线程不安全的

编译预处理和宏

宏定义

编译预处理指令

1.#开头的是编译预处理指令
2.它们不是C语言的成分,但是C语言程序离不开它们
3.#define用来定义一个宏
{
a.#define<名字><值>
b.注意没有结尾的分号,因为不是C的语句
c.名字必须是一个单词,但值可以是各种东西
d.在C语言的编译器开始编译之前,编译预处理程序(cpp)会把程序中的名字换成值
即完全的文本替换
e.gcc ——save-temps
}

没有值的宏

1.#define_DEBUG
2.这类宏是用于条件编译的,后面有其他的编译预处理指令来检查这个宏是否已经被定义过了

预定义的宏
_LINE_//行号
_FILE_//文件名
_DATE_//定义的日期
_TIME_//定义的时间
_STDC_

带参数的宏

类似于函数

#include<stdio.h>
#define cube(x) ((x)*(x)*(x))
int main()
{
printf("%d\n",cube(5));
return 0;
}

1宏可以带参数
2.可以带多个参数
#define MIN(a,b) ((a)>(b)?(b):(a))
3.也可以组合(嵌套)使用其他宏

错误定义的宏
#include<stdio.h>
#define RADTODEG1(x) (x*57.29578)
#define RADTODEG2(x) (x)*57.29578
int main()
{printf("%f\n",RADTODEG1(5+2));printf("%f\n",180/RADTODEG2(1));return 0;
}	

结果:

119.591560
10313.240400
带参数的宏的原则

1.一切都要括号
a.整个值要括号
b.参数出现的每个地方都要括号
2.#define RADTODEG(x) ((x)*57.29578)
3.宏定义的最后不要加分号!

总结:

1.在大型程序的代码中使用非常普遍
2.可以非常复杂,如“产生”函数
在#和##两个运算符帮助下
3.存在中西方文化差异
4.部分宏会被inline函数代替

其他编译预处理指令

1.条件编译
2.error
3.。。。

大程序结构

多个源代码文件

思路:
{
1.main()里的代码太长了适合分成几个函数
2.一个源代码文件太长了适合分成几个文件
3.两个独立的源代码文件不能编译形成可执行的程序
}

项目

1.在Dev C++中新建一个项目,然后把几个源代码文件加入进去
2.对于项目,Dev C++的编译会把一个项目中的所有的源代码文件都编译后,链接起来
3.有的IDE(VS)有分开的编译和构建两个按钮,前者是对单个源代码文件编译,后者是对整个项目做链接

头文件

把函数原型放到一个头文件(以.h结尾)中,在需要调用这个函数的源代码文件(.c文件)中#include这个头文件,就能让编译器在编译的时候知道函数的原型

include

1.#include是一个编译预处理指令,和宏一样,在编译前就处理了
2.它把那个文件的全部文本内容原封不动地插入到它所在的地方
3.所以也不是一定要在.C文件的最前面#include

选择“”还是<>
1.#include有两种形式来指出要插入的文件
“”要求编译器首先在当前目录(.c文件所在的目录)寻找这个文件,如果没 有,到编译器指定的目录去找
<>让编译器只在指定的目录去找
2.编译器自己知道自己的标准库的头文件在哪
3.环境变量和编译器命令行参数也可以指定寻找头文件的目录

include的误区

1.#include不是用来引入库的
2.stdio.h里只有printf的原型,printf的代码在另外的地方,某个.lib(Windows)或.a(Unix)中
3.现状的C语言编译器默认会引入所有的标准库
4.#include<stdio.h>只是为了让编译器知道printf函数的原型,保证你调用时间给出的参数值是正确的类型

不对外公开的函数

1.在函数前面加上static就使它成为只能在所在的编译单元中被使用的函数
2.在全局变量前面加上static就使得它成为只能在所在的编译单元中被使用的全局变量

声明

标准头文件结构:

#ifndef __LIST_HEAD__
#define __LIST_HEAD__
#include "node.h"
typedef struct _list{Node* head;Node* tail;} List;#endif 

1.运用条件编译和宏,保证这个头文件在一个编译单元中只会被#include一次
2.pragma once也能起到相同的作用,但不是所有的编译器都支持

这篇关于C语言学习笔记之程序结构的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验

【学习笔记】 陈强-机器学习-Python-Ch15 人工神经网络(1)sklearn

系列文章目录 监督学习:参数方法 【学习笔记】 陈强-机器学习-Python-Ch4 线性回归 【学习笔记】 陈强-机器学习-Python-Ch5 逻辑回归 【课后题练习】 陈强-机器学习-Python-Ch5 逻辑回归(SAheart.csv) 【学习笔记】 陈强-机器学习-Python-Ch6 多项逻辑回归 【学习笔记 及 课后题练习】 陈强-机器学习-Python-Ch7 判别分析 【学

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识