本文基于VS2022,将介绍一系列的C语言常见概念,让读者对C语言有一个初步的了解,并对后续的学习做下铺垫。
具体代码可以下载资源
人和人交流使用的是自然语言,如:汉语、英语、日语,那人和计算机是怎么交流的呢?使用计算机语言。
目前已知已经有上千种计算机语言,人们是通过计算机语言书写的程序,给计算机下达指令,让计算机工作的。
C语言就是众多计算机语言中的一种,当然C++/Java/Go/Python都是计算机语言。
C语言是一门编译型计算机语言,C语言源代码都是文本文件,文本文件本身无法执行,必须通过编译器翻译和链接器的链接,生成二进制的可执行文件,可执行文件才能执行。
C语言代码是放在 .c 为后缀的文件中的,要得到最终运行的可执⾏程序,中间要经过编译和链接2个过程。
c语言代码展示:
一个工程一般都会有多个源文件组成,如下图所示,演示了源程序经过编译器和链接器处理的过程。
- 每个源文件(.c)单独经过编译器处理生成对应的目标文件(.obj为后缀的文件)
- 多个目标文件和库文件经过链接器处理生成对应的可执行程序(.exe文件)
这就是,在Windows电脑上C语言程序生成的exe可执行文件
C语言是一门编译型的计算机语言,需要依赖编译器将计算机语言转换成机器能够执行的机器指令。
那我们常见的C语言编译器都有哪些呢?
比如:msvc、clang、gcc就是⼀些常见的编译器,当然也有⼀些集成开发环境如:VS2022、XCode、CodeBlocks、DevC++、Clion等。
集成开发环境(IDE)用于提供程序开发环境的应用程序,一般包括代码编辑器、编译器、调试器和图形⽤⼾界⾯等⼯具。集成了代码编写功能、分析功能、编译功能、调试功能等⼀体化的开发软件服务套。
VS2022安装方法,可以点开这个链接,了解一下VS2022的安装方法
• 功能丰富,安装包大,占用空间多。
在VS上写代码,我们是需要创建项目的,直接新建项目就可以了。
在项目中就可以添加源文件和头文件。
C语⾔把 .c 为后缀的文件称为源文件,把 .h 为后缀的文件称为头文件。
#include
#include
int main()
{
printf("Hello C\n");
system("pasue");
return 0;
}
每个C语言程序不管有多少行代码,都是从 main 函数开始执行的, main 函数是程序的入口,main 函数也被叫做:主函数。 main 前面的 int 表示 main 函数执行结束的时候返回一个整型类型的值。所以在 main 函数的最后写 return 0; 正好前后呼应。
main函数的位置可以在任意位置,但是如果在主函数之中调用了哪些函数,必须在main函数前对其所调用函数进行声明或包含其被调用函数的头文件。 main函数不一定非要在头文件的后面
VS2022 注释选中行:ctrl+k+c 取消注释:ctrl+k+u
argc:参数计数器,整型变量 ,表示参数的个数.
argv:参数数组本身,指向字符串的指针数组,表示存放参数的具体内容.
【注意】:argv[0]保存自身运行的目录路径和程序名,从argv[1]开始才是指向对应的参数
故main函数一般写为:
int main (int argc,char *argv[]){…}
int main (int argc,char **argv){…}
envp 指的是环境变量
在上面的代码中有一句代码如下:
printf("Hello C\n");
代码中使用了 printf 函数,实现了在屏幕上的信息的打印。
这里简单的介绍⼀下 printf ,printf 是一个库函数,它的功能是在标准输出设备(一般指屏幕)上进行信息的打印。上面的代码是使用 printf 函数打印字符串。只要把想要打印的一串字符放在双引号中并传递给printf函数就可以打印。
printf函数也可以用来打印其他类型的数据,比如:
int n = 100;
printf("%d\n", n); //printf打印整型
printf("%c\n", 'q'); //printf打印字符
printf("%lf\n", 3.14); //printf打印双精度浮点型
这⾥的 %d , %c 等是占位符,指的是在输出的时候会被后边的值替换。
库函数的时候,是需要包含头文件的,比如: printf 函数需要包含的就是 stdio.h 这个头⽂件,
具体的方法就是:
#include
当然我们也可以使用
#include "stdio.h"
具体区别在后续文章
那什么是库函数呢?
为了不再重复实现常见的代码,让程序员提升开发效率,C语言标准规定了一组函数,这些函数再由不同的编译器厂商根据标准进行实现,提供给程序员使用。这些函数组成了⼀个函数库,被称为标准库,这些函数也被称为库函数。在这个基础上一些编译器厂商可能会额外扩展提供部分函数(这些函数其他编译器不⼀定支持)。
⼀个系列的库函数一般会声明在同一个头文件中,所以库函数的使用,要包含对应的头文件。
C语⾔中有⼀批保留的名字的符号,比如: int 、 if 、 return ,这些符号被称为保留字或者关键字。
在C99标准中加⼊了 inline 、 restrict 、 _Bool 、 _Comploex 、 _Imaginary 等关键字。
在键盘上可以敲出各种字符,如:a,q,@,#等,这些符号都被称为字符,C语言中字符是用单引号括起来的,如:‘a’,‘b’,‘@’。
我们知道在计算机中所有的数据都是以二进制的形式存储的,那这些字符在内存中分别以什么样的二进制存储的呢?如果我们每个⼈自己给这些字符中的每个字符编一个二进制序列,这个叫做编码,为了方便大家相互通信,不造成混乱,后来美国国家标准学会(ANSI)出台了⼀个标准ASCII编码,C语言中的字符就遵循了ASCII编码的⽅式。
我们不需要记住所有的ASCII码表中的数字,在有需要的时候使用时查看就可以,不过我们最好能掌握几组特殊的数据:
- 字符A ~ Z的ASCII码值从65 ~ 90
- 字符a ~ z的ASCII码值从97 ~ 122
- 对应的大小写字符(a和A)的ASCII码值的差值是32
- 数字字符0 ~ 9的ASCII码值从48 ~ 57
- 换行 \n 的ASCII值是:10
- 在这些字符中ASCII码值从0 ~ 31这32个字符是不可打印字符,无法打印在屏幕上观察
小写字母的ASCII码值-32就能得到对应的大写字母的ASCII码值
小写字母的ASCII码值比对应的大写字母的ASCII码值更大的。
单个字符的打印可以使⽤%c来指定格式:
#include
int main()
{
printf("%c\n", 'Q');
printf("%c\n", 81); //这⾥的81是字符Q的ASCII码值,也是可以正常打印的
return 0;
}
可打印字符展⽰:
#include
int main()
{
int i = 0;
for (i = 32; i <= 127; i++)
{
printf("%c ", i);
if (i % 16 == 15)
printf("\n");
}
return 0;
}
C语言中如何表示字符串呢?使用双引号括起来的一串字符就被称为字符串,如:“abcdef”,就是⼀个字符串。
字符串的打印格式可以使用 %s 来指定,也可以直接打印如下:
#include
int main()
{
printf("%s\n", "hello C");
printf("hello c");
return 0;
}
C语言字符串中⼀个特殊的知识,就是在字符串的末尾隐藏放着⼀个 \0 字符,这个 \0 字符是字符串的结束标志。
VS2022的监视窗口观察字符串
对于字符串"abcdef",我们实际上看到了6个字符:a,b,c,d,e,f,但是实际上在末尾还隐藏⼀个 \0 的
转义字符, \0 是字符串的结束标志。所以我们在使用库函数 printf() 打印字符串或者strlen() 计算字符串长度的时候,遇到 \0 的时候就自动停止了。
C语言中也可以把⼀个字符串放在⼀个字符数组中,我们在这里利用下面的代码验证⼀下 \0 的功能。
#include
int main()
{
char arr1[] = {'a', 'b', 'c'}; //arr1数组中存放3个字符
char arr2[] = "abc"; //arr2数组中存放字符串
printf("%s\n", arr1);
printf("%s\n", arr2);
return 0;
}
这样的代码,我调试的时候,观察⼀下 arr1 和 arr2 的内容:
运行结果:
我们可以看到, arr1 字符数组在打印的时候,打印了 a 、 b 、 c 后还打印了⼀些随机值,这就是
因为 arr1 在末尾的地方没有 \0 字符作为结束标志,在打印的时候没有停止。
但是 arr2 的打印就是完全正常的,就是因为 arr2 数组是使用字符串常量初始化的,数组中有 \0
作为技术标志,打印可以正常停止。
如果我们在arr1数组中单独放⼀个’\0’字符会怎么样呢?
#include
int main()
{
char arr1[] = {'a', 'b', 'c', '\0'};
char arr2[] = "abc";
printf("%s\n", arr1);
printf("%s\n", arr2);
printf("%s\n", "abc\0def");
return 0;
}
看到三次打印的结果是一样的了,都是打印到 \0 的时候就停止了,那从上述的例子我们确实能够观察到 \0 的作用和重要性的。
也许在前面的代码中你看到 \n , \0 很纳闷时啥。其实在字符中有⼀组特殊的字符是转义字符,转义字符顾名思义:转变原来的意思的字符。
比如:我们有字符 n ,在字符串中打印的时候自然能打印出这个字符,如下:
#include
int main()
{
printf("abcndef");
return 0;
}
输出的结果:
如果我们修改⼀下代码,在 n 的前面加上 \ ,变成如下代码:
#include
int main()
{
printf("abc\ndef");
return 0;
}
输出的结果:
我们可以看到修改的前后代码输出的结果,截然不同的,那这是为什么呢?
这就是转义字符的问题, \n 是⼀个转义字符表示换行的意思,我们可以简单的理解为 \ 让 n 的意思发生了转变, n 本来是⼀个普通的字符,被 \ 转义为换行的意思。
C语言中像这样的转义字符还有⼀些,具体如下:
下面2种转义字符可以理解为:字符的8进制或者16进制表示形式
#include
int main()
{
printf("%c\n", '\'');//引号会默认与第一个引号匹配,如果是'''则会报错,这时候就需要转义字符了
printf("%s\n", "\"");
printf("c:\\test\\code\\test.c\n");
printf("\a");
printf("%c\n", '\130'); //130是8进制,转换成10进制是88,以88作为ASCII码值的字符是
printf("%c\n", '\x30'); //x30中的30是16进制,转换成10进制是48,以48作为ASCII码值的
return 0;
}
这些ASCII码值是可以自己写代码验证的,大家也可以自己验证。
关于转义字符我们首先要了解,然后要能在字符串中识别出来。
例题:
int main()
{
//strlen是求字符串长度的函数- 统计的是字符串中\0之前的字符的个数,不包含\0
//string.h
printf("%zd\n", strlen("abc"));//abc\0 3
printf("%zd\n", strlen("c:\test\130\test.c"));//? 13
return 0;
}
\t \130 \t 都是一个字符,不能按照正常的来数个数 %zd 是打印 size_t 类型的数字 size_t 就是无符号数,具体可看第二篇C语言数据类型和变量 1.5.1 sizeof操作符 的关于size_t的解释
C语言的代码是由⼀条⼀条的语句构成的,C语言中的语句可为以下五类:
空语句是最简单的,⼀个分号就是⼀条语句,是空语句。
#include
int main()
{
; //空语句
return 0;
}
空语句,⼀般出现的地方是:这里需要⼀条语句,但是这个语句不需要做任何事,就可以写⼀个空语句。
表达式语句就是在表达式的后边加上分号。如下所⽰:
#include
int main()
{
;//空语句无意义
int a = 20;
int b = 0;
b = a + 5; //表达式语句
return 0;
}
函数调用的时候,也会加上分号,就是函数调用语句。
#include
int Add(int x, int y)
{
return x+y;
}
int main()
{
printf("hehe\n"); //函数调⽤语句
int ret = Add(2, 3); //函数调⽤语句
return 0;
}
复合语句其实就是前面讲过的代码块,成对括号中的代码就构成⼀个代码块,也被称为复合语句。
#include
void print(int arr[], int sz) //函数的⼤括号中的代码也构成复合语句
{
int i = 0;
for(i=0; i<sz; i++)
{
printf("%d ", arr[i]);
}
}
int main()
{
int i = 0;
int arr[10] = {0};
for(i=0; i<10; i++) //for循环的循环体的⼤括号中的就是复合语句
{
arr[i] = 10-i;
printf("%d\n", arr[i]);
}
return 0;
}
控制语句用于控制程序的执行流程,以实现程序的各种结构方式(C语⾔支持三种结构:顺序结构、选择结构、循环结构),它们由特定的语句定义符组成,C语言有九种控制语句。
可分成以下三类:
注释是对代码的说明,编译器会忽略注释,也就是说,注释对实际代码没有影响。
注释是给程序员自己,或者其他程序员看的。
好的注释可以帮我们更好的理解代码,但是也不要过度注释,不要写没必要的注释。
当然不写注释可能会让后期阅读代码的⼈抓狂。
写注释⼀定程度上反应了程序作者的素质,建议⼤家写必要的注释。
C语言的注释有两种表示方法。
第⼀种方法是将注释放在 /* … */ 之间,内部可以分行。
/* 注释 */
/*
这是⼀⾏注释
*/
这种注释可以插在行内。
int fopen(char* s /* file name */ , int mode);
上面示例中, /* file name */ 用来对函数参数进行说明,跟在它后面的代码依然会有效执行。
这种注释⼀定不能忘记写结束符号 */ ,否则很容易导致错误。
printf("a "); /* 注释⼀
printf("b ");
printf("c "); /* 注释⼆ */
printf("d ");
上面示例的原意是,第⼀行和第三行代码的尾部,有两个注释。
但是,第⼀行注释忘记写结束符号,导致注释⼀延续到第三行结束。
/* */
的这个注释也不支持嵌套注释, /* 开始注释后,遇到第⼀个 */ 就认为注释结束了。
/*
printf("a ");
printf("b ");
printf("c "); /* 注释⼆ */
printf("d ");
*/
一般把/* * / 称为C语言风格注释,但是有缺点,不支持嵌套注释
第⼆种写法是将注释放在双斜杠 // 后面,从双斜杠到⾏尾都属于注释。这种注释只能是单行,可以放在行首,也可以放在一行语句的结尾。这是C99标准新增的语法。
// 这是⼀⾏注释
int x = 1; // 这也是注释
不管是哪⼀种注释,都不能放在双引号里面。
双引号里面的注释符号,会成为字符串的⼀部分,解释为普通符号,失去注释作⽤。
printf("// hello /* world */ ");
上面示例中,双引号里面的注释符号,都会被视为普通字符,没有注释作用。
编译时,注释会被替换成⼀个空格,所以 min/* 这里是注释*/Value 会变成 min Value ,而不是 minValue 。
int main()
{
int a/* hehe*/ b;//像这样就会报错,注释会被替换成空格
return 0;
}