当前位置: 首页 > news >正文

全国计算机二级C语言二级考试通关笔记

C语言二级考试通关笔记

前言:值不值得考?

在所有二级证书中,C语言是我最推荐的一个,因为它会逼迫你去试着理解编程思维内存管理。准确来说,只要你真的学习c语言,就无法避开这些,哪怕是二级考试这种简单的考试,因而它一定程度上会逼迫你去理解和学习,虽然二级证书整体专业性一般,但是这已经是大学生除了蓝桥杯等io竞赛外最容易拿到的专业证书,而考虑到蓝桥杯这些比赛,如果希望拿奖,准备时间不会低于两个月,只需要14天就能稳定通过的二级考试性价比可以说极高。

在大厂实习期间我发现,面试官特别喜欢问C语言相关的底层问题。即使你应聘的是Java开发岗位,他们也会问指针、内存泄漏、段错误等概念,因为这些反映了你对计算机系统的理解深度。

备考用到的题库:c.code2ji.cn


第一章:C语言基础语法

1.1 C语言的特点

C语言有几个显著特点:

  • 高效简洁:代码简短,执行效率高
  • 贴近硬件:可以直接操作内存和硬件
  • 可移植性强:在不同平台上都能运行
  • 功能强大:系统编程的首选语言

1.2 数据类型 - 精确控制内存

C语言数据类型分类:

// 基本数据类型
char         // 字符类型,1字节
int          // 整数类型,通常4字节
float        // 单精度浮点数,4字节
double       // 双精度浮点数,8字节// 修饰符
short        // 短整型,2字节
long         // 长整型,4或8字节
signed       // 有符号(默认)
unsigned     // 无符号// 空类型
void         // 空类型

数据类型详细说明:

#include <stdio.h>
#include <limits.h>   // 整数限制
#include <float.h>    // 浮点数限制int main() {// 字符类型char c1 = 'A';              // 字符常量char c2 = 65;               // ASCII值signed char sc = -128;      // 有符号字符:-128到127unsigned char uc = 255;     // 无符号字符:0到255// 整数类型short s = 32767;            // 短整型:-32768到32767int i = 2147483647;         // 整数:-2^31到2^31-1long l = 2147483647L;       // 长整型,常数后加Lunsigned int ui = 4294967295U; // 无符号整数:0到2^32-1// 浮点数类型float f = 3.14159f;         // 单精度,常数后加fdouble d = 3.141592653589793; // 双精度long double ld = 3.141592653589793L; // 扩展精度// 输出信息printf("字符: c1='%c'(%d), c2='%c'(%d)\n", c1, c1, c2, c2);printf("整数: short=%d, int=%d, long=%ld\n", s, i, l);printf("浮点: float=%.6f, double=%.15f\n", f, d);// 数据类型大小printf("\n数据类型大小:\n");printf("char: %zu 字节\n", sizeof(char));printf("short: %zu 字节\n", sizeof(short));printf("int: %zu 字节\n", sizeof(int));printf("long: %zu 字节\n", sizeof(long));printf("float: %zu 字节\n", sizeof(float));printf("double: %zu 字节\n", sizeof(double));// 数据类型范围printf("\n数据类型范围:\n");printf("int: %d 到 %d\n", INT_MIN, INT_MAX);printf("unsigned int: 0 到 %u\n", UINT_MAX);printf("float: %e 到 %e\n", FLT_MIN, FLT_MAX);return 0;
}

变量声明和初始化:

// 变量声明语法
数据类型 变量名;
数据类型 变量名 = 初始值;// 实例
int age;                    // 只声明
age = 20;                   // 赋值
int score = 85;             // 声明同时初始化// 多个变量声明
int a, b, c;                // 声明多个变量
int x = 1, y = 2, z = 3;    // 声明并初始化// 常量声明
const int MAX_SIZE = 100;   // const关键字声明常量
#define PI 3.14159          // 宏定义常量// 存储类型
auto int local_var = 10;    // 自动变量(局部)
static int count = 0;       // 静态变量
register int fast_var = 5;  // 寄存器变量
extern int global_var;      // 外部变量声明

类型转换:

#include <stdio.h>int main() {// 自动类型转换(隐式转换)int i = 10;float f = i;        // int 自动转换为 floatdouble d = f;       // float 自动转换为 doublechar c = 'A';int ascii = c;      // char 自动转换为 intprintf("自动转换: i=%d, f=%.1f, d=%.1f, ascii=%d\n", i, f, d, ascii);// 强制类型转换(显式转换)double d1 = 3.14159;int i1 = (int)d1;           // 强制转换,丢失小数部分float f1 = (float)d1;       // double 转 floatchar c1 = (char)65;         // int 转 charprintf("强制转换: d1=%.5f, i1=%d, f1=%.5f, c1='%c'\n", d1, i1, f1, c1);// 运算中的类型提升char a = 10, b = 20;int result = a + b;         // char 自动提升为 intfloat x = 3.5f;double y = 2.1;double sum = x + y;         // float 自动提升为 doubleprintf("类型提升: result=%d, sum=%.2f\n", result, sum);return 0;
}

🔥 真题实战:去除偶数生成新数

在这里插入图片描述

这道题考查数字的位操作和指针传递:

题目要求: 从输入的正整数中提取所有奇数位,组成新整数

#include <stdio.h>void fun(unsigned long *n) {unsigned long x = 0, i;    int t;i = 1;while(*n) {t = *n % 10;           // 取个位数字if(t % 2 != 0) {       // 判断是否为奇数x = x + t * i;     // 构建新数字i = i * 10;        // 位权递增}*n = *n / 10;          // 去掉个位}*n = x;                    // 将结果存回原变量
}int main(void) {unsigned long n = -1;// 输入验证循环while(n > 99999999 || n < 0) {printf("Please input(0<n<100000000): "); scanf("%ld", &n); }fun(&n);                   // 传递地址printf("\nThe result is: %ld\n", n);return 0;
}

知识点解析:

  1. 指针传递fun(&n) 传递变量地址,*n 解引用访问值
  2. 数字分解n % 10 取个位,n / 10 去掉个位
  3. 奇偶判断t % 2 != 0 判断是否为奇数
  4. 数字构建:用位权i重新组合数字
  5. 输入验证:while循环确保输入在有效范围内

记忆技巧:

  • 指针:&取地址,*取值
  • 数字操作:%10取个位,/10去个位
  • 位权概念:个位权重1,十位权重10,百位权重100…

指针和数字处理常见考点:

  • 数字的位操作和分解
  • 指针传递和解引用
  • 条件判断和循环控制
  • 数字构造和组合

第二章:函数与文件操作 - 模块化编程

2.1 函数的定义与调用

C函数语法结构:

// 函数定义的完整语法
返回类型 函数名(参数类型1 参数1, 参数类型2 参数2, ...) {函数体;return 返回值;  // 可选
}// 函数声明(原型)
返回类型 函数名(参数类型1, 参数类型2, ...);

函数定义示例:

#include <stdio.h>// 函数声明(放在main前面)
int add(int a, int b);               // 有参数有返回值
void printMessage(void);             // 无参数无返回值
float calculateAverage(int arr[], int size); // 数组参数
int max(int x, int y);               // 比较函数
void swap(int *a, int *b);           // 指针参数int main() {// 函数调用示例int result = add(5, 3);printf("加法结果: %d\n", result);printMessage();int numbers[] = {85, 92, 78, 96, 88};float avg = calculateAverage(numbers, 5);printf("平均分: %.2f\n", avg);int maxVal = max(15, 23);printf("最大值: %d\n", maxVal);int x = 10, y = 20;printf("交换前: x=%d, y=%d\n", x, y);swap(&x, &y);printf("交换后: x=%d, y=%d\n", x, y);return 0;
}// 函数定义实现// 1. 有参数有返回值的函数
int add(int a, int b) {int result = a + b;return result;
}// 2. 无参数无返回值的函数
void printMessage(void) {printf("这是一个无参数无返回值的函数\n");
}// 3. 数组参数函数
float calculateAverage(int arr[], int size) {int sum = 0;for (int i = 0; i < size; i++) {sum += arr[i];}return (float)sum / size;
}// 4. 条件返回函数
int max(int x, int y) {if (x > y) {return x;} else {return y;}// 可以简化为: return (x > y) ? x : y;
}// 5. 指针参数函数(实现参数交换)
void swap(int *a, int *b) {int temp = *a;*a = *b;*b = temp;
}

函数参数传递方式:

#include <stdio.h>// 值传递(传入参数的副本)
void changeValue(int x) {x = 100;  // 只改变副本,不影响原变量printf("函数内x = %d\n", x);
}// 地址传递(传入参数的地址)
void changePointer(int *x) {*x = 100;  // 通过指针修改原变量的值printf("函数内*x = %d\n", *x);
}// 数组传递(传入数组名,实际上是传入数组首元素的地址)
void modifyArray(int arr[], int size) {for (int i = 0; i < size; i++) {arr[i] *= 2;  // 直接修改原数组元素}
}int main() {// 测试值传递int a = 10;printf("调用前 a = %d\n", a);changeValue(a);printf("调用后 a = %d\n\n", a);  // a仍为10// 测试地址传递int b = 20;printf("调用前 b = %d\n", b);changePointer(&b);printf("调用后 b = %d\n\n", b);  // b变为100// 测试数组传递int numbers[] = {1, 2, 3, 4, 5};printf("修改前数组: ");for (int i = 0; i < 5; i++) {printf("%d ", numbers[i]);}printf("\n");modifyArray(numbers, 5);printf("修改后数组: ");for (int i = 0; i < 5; i++) {printf("%d ", numbers[i]);}printf("\n");return 0;
}

函数的递归调用:

#include <stdio.h>// 递归计算阶乘
long factorial(int n) {if (n <= 1) {return 1;  // 递归的终止条件}return n * factorial(n - 1);  // 递归调用
}// 递归计算斐波那契数列
long fibonacci(int n) {if (n <= 1) {return n;}return fibonacci(n - 1) + fibonacci(n - 2);
}// 递归计算最大公约数
int gcd(int a, int b) {if (b == 0) {return a;}return gcd(b, a % b);
}int main() {// 测试阶乘int n = 5;printf("%d! = %ld\n", n, factorial(n));// 测试斐波那契数列printf("斐波那契数列前10项: ");for (int i = 0; i < 10; i++) {printf("%ld ", fibonacci(i));}printf("\n");// 测试最大公约数int a = 48, b = 18;printf("gcd(%d, %d) = %d\n", a, b, gcd(a, b));return 0;
}

2.2 文件操作 - 数据的持久化

文件操作就像是操作现实中的文件柜,需要打开、读写、关闭三个步骤:

#include <stdio.h>int main() {FILE *fp;// 打开文件进行写入fp = fopen("data.txt", "w");if (fp == NULL) {printf("文件打开失败!\n");return 1;}// 写入数据fprintf(fp, "Hello, C语言!\n");fprintf(fp, "数字: %d\n", 42);// 关闭文件fclose(fp);return 0;
}

🔥 真题实战:实型数四舍五入

在这里插入图片描述

这道题综合考查函数定义和文件操作:

题目要求: 实现浮点数四舍五入并进行文件批处理

#include <stdio.h>// 四舍五入函数实现
float fun(float h) {// 核心算法:乘100,加0.5,取整,再除100return (int)(h * 100 + 0.5) / 100.0;
}void NONO(void);   // 函数声明int main(void) {float a;printf("Enter  a:  ");if (scanf("%f", &a) != 1) return 0;  // 输入检查printf("The original data is :   %f\n\n", a);printf("The result :  %f\n", fun(a));NONO();  // 调用文件处理函数return 0;
}// 文件批处理函数
void NONO(void) {int i;float a;FILE *rf = fopen("in.txt", "r");     // 打开输入文件FILE *wf = fopen("out.txt", "w");    // 打开输出文件if (!rf || !wf) {                    // 检查文件是否成功打开printf("文件打开失败!\n");return;}// 读取并处理20个数字for (i = 0; i < 20 && fscanf(rf, "%f", &a) == 1; i++) {fprintf(wf, "%f\n", fun(a));     // 处理后写入文件}fclose(rf);  // 关闭文件fclose(wf);
}

知识点解析:

  1. 四舍五入算法(int)(h * 100 + 0.5) / 100.0
    • 乘100:将小数点后两位移到整数部分
    • 加0.5:实现四舍五入
    • 取整:去掉多余小数
    • 除100.0:恢复原来的小数位置
  2. 文件操作fopen() 打开,fscanf()/fprintf() 读写,fclose() 关闭
  3. 错误检查:检查文件是否成功打开,检查输入是否成功
  4. 函数分离:主函数处理交互,NONO函数处理文件批量操作

记忆技巧:

  • 四舍五入:×100 → +0.5 → 取整 → ÷100
  • 文件操作:打开 → 读写 → 关闭
  • 错误检查:每个操作都要检查返回值

函数和文件常见考点:

  • 函数的定义、声明和调用
  • 文件的打开、读写、关闭操作
  • 数学算法的实现(四舍五入等)
  • 批量数据处理和错误检查

第三章:数组与字符串 - 数据的批量处理

3.1 一维数组

数组就像是一排编了号的储物柜,每个柜子存放一个数据:

#include <stdio.h>int main() {int scores[5] = {85, 92, 78, 96, 88};  // 定义并初始化int sum = 0;// 计算总分for (int i = 0; i < 5; i++) {sum += scores[i];}// 计算平均分double average = (double)sum / 5;printf("平均分: %.2f\n", average);return 0;
}

数组声明和初始化语法:

// 一维数组声明语法
数据类型 数组名[数组大小];
数据类型 数组名[数组大小] = {初始值列表};// 实例
int numbers[5];                    // 声明5个整数的数组
int scores[5] = {85, 92, 78, 96, 88};  // 声明并初始化
int data[] = {1, 2, 3, 4};         // 自动推导大小// 二维数组声明
int matrix[3][4];                  // 3行4列的二维数组
int grid[2][3] = {{1,2,3}, {4,5,6}}; // 声明并初始化// 字符数组(字符串)
char str[20];                      // 声明字符数组
char name[20] = "Hello";           // 声明并初始化
char message[] = "Welcome";        // 自动推导大小

数组操作示例:

#include <stdio.h>int main() {// 数组的基本操作int numbers[5] = {10, 20, 30, 40, 50};int sum = 0;// 遍历数组printf("数组元素: ");for (int i = 0; i < 5; i++) {printf("%d ", numbers[i]);sum += numbers[i];}printf("\n总和: %d\n", sum);// 修改数组元素numbers[0] = 100;printf("修改后第一个元素: %d\n", numbers[0]);// 二维数组操作int matrix[3][3] = {{1, 2, 3},{4, 5, 6},{7, 8, 9}};printf("二维数组:\n");for (int i = 0; i < 3; i++) {for (int j = 0; j < 3; j++) {printf("%d ", matrix[i][j]);}printf("\n");}return 0;
}

3.2 字符串操作 - 字符数组的特殊应用

字符串基本语法:

#include <stdio.h>
#include <string.h>// 字符串声明和初始化
char str1[20];                    // 声明字符数组
char str2[20] = "Hello";          // 初始化字符串
char str3[] = "World";            // 自动推导大小
char str4[20] = {'H','e','l','l','o','\0'}; // 字符数组形式// 字符串常量
char *ptr = "Hello World";        // 指向字符串常量的指针

字符串函数详解:

#include <stdio.h>
#include <string.h>int main() {char name[50] = "张三";char greeting[100];char copy_str[50];char search_str[] = "Hello World Programming";// strlen() - 计算字符串长度int len = strlen(name);printf("字符串长度: %d\n", len);// strcpy() - 字符串复制strcpy(greeting, "你好, ");strcpy(copy_str, name);printf("复制结果: %s\n", copy_str);// strcat() - 字符串连接strcat(greeting, name);printf("连接结果: %s\n", greeting);  // 输出:你好, 张三// strcmp() - 字符串比较int result = strcmp("apple", "banana");if (result < 0) {printf("apple < banana\n");} else if (result > 0) {printf("apple > banana\n");} else {printf("apple == banana\n");}// strchr() - 查找字符char *pos = strchr(search_str, 'W');if (pos != NULL) {printf("找到字符W,位置: %ld\n", pos - search_str);}// strstr() - 查找子字符串char *sub_pos = strstr(search_str, "World");if (sub_pos != NULL) {printf("找到子字符串World,位置: %ld\n", sub_pos - search_str);}// strncpy() - 安全的字符串复制(指定长度)char safe_copy[10];strncpy(safe_copy, "Very Long String", 9);safe_copy[9] = '\0';  // 手动添加结束符printf("安全复制: %s\n", safe_copy);return 0;
}

字符串输入输出:

#include <stdio.h>int main() {char input[100];// 使用scanf读取字符串(遇到空格停止)printf("请输入一个单词: ");scanf("%s", input);  // 注意:数组名就是地址,不需要&printf("输入的单词: %s\n", input);// 清空输入缓冲区while (getchar() != '\n');// 使用fgets读取整行(包括空格)printf("请输入一行文本: ");fgets(input, sizeof(input), stdin);printf("输入的文本: %s", input);  // fgets保留换行符// 使用gets读取整行(不安全,不推荐)// gets(input);  // 已废弃,容易缓冲区溢出return 0;
}

字符串常见考点:

  • 字符串的声明和初始化
  • strlen、strcpy、strcat、strcmp等函数
  • 字符串的遍历和处理
  • 字符数组和指针的关系

第四章:指针 - C语言的精髓

4.1 指针的基本概念和语法

指针声明语法:

// 指针声明的基本语法
数据类型 *指针名;
数据类型 *指针名 = &变量名;// 实例
int *ptr;              // 声明一个指向int的指针
float *fptr;           // 声明一个指向float的指针
char *cptr;            // 声明一个指向char的指针int num = 42;
int *p = &num;         // 声明并初始化指针// 多级指针
int **pp = &p;         // 指向指针的指针
int ***ppp = &pp;      // 指向指针的指针的指针

指针基本操作:

#include <stdio.h>int main() {int num = 42;      // 定义变量int *ptr = &num;   // 定义指针,指向num的地址printf("=== 指针基本信息 ===\n");printf("num的值: %d\n", num);printf("num的地址: %p\n", &num);printf("ptr的值(存储的地址): %p\n", ptr);printf("ptr的地址: %p\n", &ptr);printf("ptr指向的值: %d\n", *ptr);printf("\n=== 通过指针修改值 ===\n");*ptr = 100;        // 通过指针修改num的值printf("修改后num的值: %d\n", num);printf("\n=== 指针运算 ===\n");int arr[5] = {10, 20, 30, 40, 50};int *arr_ptr = arr;  // 数组名就是指向第一个元素的指针printf("数组首地址: %p\n", arr);printf("指针指向: %p\n", arr_ptr);// 指针算术运算for (int i = 0; i < 5; i++) {printf("arr[%d] = %d, 地址: %p\n", i, *(arr_ptr + i), arr_ptr + i);}printf("\n=== 多级指针 ===\n");int **pp = &ptr;   // 指向指针的指针printf("pp的值(ptr的地址): %p\n", pp);printf("*pp的值(ptr的值): %p\n", *pp);printf("**pp的值(num的值): %d\n", **pp);return 0;
}

指针与不同数据类型:

#include <stdio.h>int main() {// 整型指针int x = 10;int *int_ptr = &x;printf("整型: *int_ptr = %d\n", *int_ptr);// 浮点型指针float y = 3.14f;float *float_ptr = &y;printf("浮点型: *float_ptr = %.2f\n", *float_ptr);// 字符指针char c = 'A';char *char_ptr = &c;printf("字符型: *char_ptr = %c\n", *char_ptr);// 字符串指针char *str_ptr = "Hello World";printf("字符串: %s\n", str_ptr);// void指针(通用指针)void *void_ptr;void_ptr = &x;printf("void指针指向整数: %d\n", *(int*)void_ptr);void_ptr = &y;printf("void指针指向浮点数: %.2f\n", *(float*)void_ptr);return 0;
}

空指针和野指针:

#include <stdio.h>
#include <stdlib.h>int main() {// 空指针int *null_ptr = NULL;  // 初始化为NULLif (null_ptr == NULL) {printf("指针为空,不能解引用\n");}// 检查指针是否为空(安全编程)if (null_ptr != NULL) {printf("指针值: %d\n", *null_ptr);} else {printf("指针为空,无法访问\n");}// 动态分配内存int *dynamic_ptr = (int*)malloc(sizeof(int));if (dynamic_ptr != NULL) {*dynamic_ptr = 100;printf("动态分配的值: %d\n", *dynamic_ptr);free(dynamic_ptr);     // 释放内存dynamic_ptr = NULL;    // 避免野指针}return 0;
}

4.2 指针与数组的关系

数组和指针的等价性:

#include <stdio.h>int main() {int numbers[] = {1, 2, 3, 4, 5};int *ptr = numbers;  // 数组名就是指向第一个元素的指针printf("=== 数组访问方式比较 ===\n");for (int i = 0; i < 5; i++) {printf("numbers[%d] = %d\n", i, numbers[i]);    // 数组下标方式printf("*(numbers+%d) = %d\n", i, *(numbers+i)); // 指针运算方式printf("ptr[%d] = %d\n", i, ptr[i]);           // 指针下标方式printf("*(ptr+%d) = %d\n", i, *(ptr+i));       // 指针运算方式printf("\n");}return 0;
}

指针数组和数组指针:

#include <stdio.h>int main() {// 指针数组:存放指针的数组char *names[3] = {"张三", "李四", "王五"};printf("=== 指针数组 ===\n");for (int i = 0; i < 3; i++) {printf("names[%d] = %s\n", i, names[i]);}// 数组指针:指向数组的指针int matrix[2][3] = {{1,2,3}, {4,5,6}};int (*array_ptr)[3] = matrix;  // 指向包含3个int的数组printf("\n=== 数组指针 ===\n");for (int i = 0; i < 2; i++) {for (int j = 0; j < 3; j++) {printf("matrix[%d][%d] = %d\n", i, j, array_ptr[i][j]);}}return 0;
}

函数参数中的数组和指针:

#include <stdio.h>// 数组作为参数的三种等价写法
void printArray1(int arr[], int size) {printf("方式1 - arr[]: ");for (int i = 0; i < size; i++) {printf("%d ", arr[i]);}printf("\n");
}void printArray2(int *arr, int size) {printf("方式2 - *arr: ");for (int i = 0; i < size; i++) {printf("%d ", *(arr + i));}printf("\n");
}void printArray3(int arr[5], int size) {  // 数字5会被忽略printf("方式3 - arr[5]: ");for (int i = 0; i < size; i++) {printf("%d ", arr[i]);}printf("\n");
}// 修改数组元素
void modifyArray(int *arr, int size) {for (int i = 0; i < size; i++) {arr[i] *= 2;  // 将每个元素翻倍}
}int main() {int numbers[] = {1, 2, 3, 4, 5};int size = sizeof(numbers) / sizeof(numbers[0]);printf("原始数组: ");for (int i = 0; i < size; i++) {printf("%d ", numbers[i]);}printf("\n\n");printArray1(numbers, size);printArray2(numbers, size);printArray3(numbers, size);printf("\n修改数组后: ");modifyArray(numbers, size);for (int i = 0; i < size; i++) {printf("%d ", numbers[i]);}printf("\n");return 0;
}

第五章:结构体 - 自定义数据类型

5.1 结构体的定义和语法

结构体定义语法:

// 结构体定义的基本语法
struct 结构体名 {数据类型1 成员名1;数据类型2 成员名2;// ...
};// 实例:学生结构体
struct Student {int id;           // 学号char name[20];    // 姓名float score;      // 成绩char grade;       // 年级
};// 结构体变量声明方式
struct Student stu1;                    // 声明一个变量
struct Student stu2, stu3;              // 声明多个变量
struct Student students[10];            // 声明结构体数组// 结构体类型定义(typedef)
typedef struct {int x;int y;
} Point;                                // 直接使用Point作为类型名Point p1, p2;                           // 不需要struct关键字

结构体初始化方式:

#include <stdio.h>
#include <string.h>struct Student {int id;char name[20];float score;
};int main() {// 方式1:分别赋值struct Student stu1;stu1.id = 1001;strcpy(stu1.name, "张三");stu1.score = 85.5;// 方式2:初始化列表struct Student stu2 = {1002, "李四", 92.0};// 方式3:指定初始化(C99标准)struct Student stu3 = {.id = 1003,.name = "王五",.score = 78.5};// 方式4:部分初始化struct Student stu4 = {1004};  // 只初始化第一个成员strcpy(stu4.name, "赵六");stu4.score = 88.0;printf("学生信息:\n");printf("ID: %d, Name: %s, Score: %.1f\n", stu1.id, stu1.name, stu1.score);printf("ID: %d, Name: %s, Score: %.1f\n", stu2.id, stu2.name, stu2.score);printf("ID: %d, Name: %s, Score: %.1f\n", stu3.id, stu3.name, stu3.score);printf("ID: %d, Name: %s, Score: %.1f\n", stu4.id, stu4.name, stu4.score);return 0;
}

结构体数组和指针:

#include <stdio.h>
#include <string.h>struct Student {int id;char name[20];float score;
};// 处理结构体数组的函数
void printStudents(struct Student students[], int count) {for (int i = 0; i < count; i++) {printf("学号: %d, 姓名: %s, 成绩: %.1f\n", students[i].id, students[i].name, students[i].score);}
}// 使用指针处理结构体
void updateScore(struct Student *stu, float newScore) {stu->score = newScore;  // 指针访问结构体成员的语法
}int main() {// 结构体数组初始化struct Student students[3] = {{1001, "张三", 85.5},{1002, "李四", 92.0},{1003, "王五", 78.5}};printf("原始成绩:\n");printStudents(students, 3);// 使用指针修改成绩updateScore(&students[1], 95.0);printf("\n修改后成绩:\n");printStudents(students, 3);// 结构体指针数组struct Student *ptr_array[3];for (int i = 0; i < 3; i++) {ptr_array[i] = &students[i];}printf("\n通过指针数组访问:\n");for (int i = 0; i < 3; i++) {printf("学号: %d, 姓名: %s, 成绩: %.1f\n", ptr_array[i]->id, ptr_array[i]->name, ptr_array[i]->score);}return 0;
}

嵌套结构体和联合:

#include <stdio.h>
#include <string.h>// 嵌套结构体
struct Date {int year;int month;int day;
};struct Person {char name[20];int age;struct Date birthday;  // 嵌套结构体
};// 联合(union)
union Data {int i;float f;char c;
};  // 所有成员共享同一块内存int main() {// 嵌套结构体使用struct Person p1;strcpy(p1.name, "张三");p1.age = 25;p1.birthday.year = 1998;p1.birthday.month = 5;p1.birthday.day = 15;printf("个人信息:\n");printf("姓名: %s\n", p1.name);printf("年龄: %d\n", p1.age);printf("生日: %d-%d-%d\n", p1.birthday.year, p1.birthday.month, p1.birthday.day);// 联合使用union Data data;data.i = 10;printf("联合中的整数: %d\n", data.i);data.f = 3.14f;printf("联合中的浮点数: %.2f\n", data.f);printf("此时整数值变为: %d(因为共享内存)\n", data.i);printf("联合大小: %zu 字节\n", sizeof(union Data));printf("结构体大小: %zu 字节\n", sizeof(struct Person));return 0;
}

第六章:动态内存分配 - 灵活管理内存

6.1 malloc和free

动态内存分配就like是向银行借钱,用完要记得还:

#include <stdio.h>
#include <stdlib.h>int main() {int n;printf("请输入数组大小: ");scanf("%d", &n);// 动态分配内存int *arr = (int*)malloc(n * sizeof(int));if (arr == NULL) {printf("内存分配失败!\n");return 1;}// 初始化数组for (int i = 0; i < n; i++) {arr[i] = i + 1;}// 使用数组for (int i = 0; i < n; i++) {printf("%d ", arr[i]);}printf("\n");// 释放内存free(arr);arr = NULL;  // 避免野指针return 0;
}

第七章:预处理与编译 - 程序的构建过程

7.1 预处理指令

#include <stdio.h>     // 包含标准库
#define PI 3.14159     // 定义常量
#define MAX(a,b) ((a)>(b)?(a):(b))  // 定义宏#ifdef DEBUG#define PRINT(x) printf(x)
#else#define PRINT(x)
#endifint main() {printf("圆周率: %f\n", PI);printf("最大值: %d\n", MAX(5, 3));return 0;
}

第八章:考试技巧与注意事项

7.2 文件操作语法

文件操作函数:

#include <stdio.h>// 文件操作基本函数
FILE* fopen(const char* filename, const char* mode);  // 打开文件
int fclose(FILE* stream);                            // 关闭文件
int fgetc(FILE* stream);                             // 读取字符
int fputc(int c, FILE* stream);                      // 写入字符
char* fgets(char* str, int n, FILE* stream);         // 读取字符串
int fputs(const char* str, FILE* stream);            // 写入字符串
int fscanf(FILE* stream, const char* format, ...);   // 格式化读取
int fprintf(FILE* stream, const char* format, ...);  // 格式化写入// 文件位置操作
long ftell(FILE* stream);                            // 获取当前位置
int fseek(FILE* stream, long offset, int whence);     // 设置文件位置
void rewind(FILE* stream);                           // 重置到文件开头// 文件状态检查
int feof(FILE* stream);                              // 检查是否到达文件末尾
int ferror(FILE* stream);                            // 检查文件错误
void clearerr(FILE* stream);                         // 清除错误标志

文件打开模式:

// 文件打开模式说明
"r"    // 只读模式,文件必须存在
"w"    // 只写模式,创建新文件或覆盖现有文件
"a"    // 附加模式,在文件末尾写入
"r+"   // 读写模式,文件必须存在
"w+"   // 读写模式,创建新文件或覆盖现有文件
"a+"   // 读写模式,在文件末尾附加
"rb"   // 二进制只读模式
"wb"   // 二进制只写模式
"ab"   // 二进制附加模式

文件操作完整示例:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>// 写入文件示例
void write_file_example() {printf("=== 文件写入示例 ===\n");FILE *fp = fopen("student_data.txt", "w");if (fp == NULL) {printf("无法打开文件进行写入!\n");return;}// 写入学生信息fprintf(fp, "学号,姓名,性别,成绩\n");fprintf(fp, "1001,张三,男,85.5\n");fprintf(fp, "1002,李四,女,92.0\n");fprintf(fp, "1003,王五,男,78.5\n");fclose(fp);printf("学生数据已写入文件\n\n");
}// 读取文件示例
void read_file_example() {printf("=== 文件读取示例 ===\n");FILE *fp = fopen("student_data.txt", "r");if (fp == NULL) {printf("无法打开文件进行读取!\n");return;}char line[100];int line_num = 1;// 逐行读取文件while (fgets(line, sizeof(line), fp) != NULL) {printf("第%d行: %s", line_num, line);line_num++;}fclose(fp);printf("\n");
}// 格式化读取示例
void formatted_read_example() {printf("=== 格式化读取示例 ===\n");FILE *fp = fopen("numbers.txt", "w");if (fp != NULL) {fprintf(fp, "10 20.5 30\n");fprintf(fp, "40 50.5 60\n");fprintf(fp, "70 80.5 90\n");fclose(fp);}fp = fopen("numbers.txt", "r");if (fp == NULL) {printf("无法打开文件!\n");return;}int a, c;float b;// 格式化读取while (fscanf(fp, "%d %f %d", &a, &b, &c) == 3) {printf("读取到: a=%d, b=%.1f, c=%d\n", a, b, c);}fclose(fp);printf("\n");
}// 文件位置操作示例
void file_position_example() {printf("=== 文件位置操作示例 ===\n");FILE *fp = fopen("position_test.txt", "w+");if (fp == NULL) {printf("无法打开文件!\n");return;}// 写入一些数据fputs("Hello World!", fp);// 获取当前位置long pos = ftell(fp);printf("当前文件位置: %ld\n", pos);// 重置到文件开头rewind(fp);// 读取数据char buffer[20];fgets(buffer, sizeof(buffer), fp);printf("从文件开头读取: %s\n", buffer);// 移动到文件中间fseek(fp, 6, SEEK_SET);  // 从开头第6个字节fgets(buffer, 6, fp);printf("从位置6开始读取: %s\n", buffer);fclose(fp);printf("\n");
}// 错误处理示例
void error_handling_example() {printf("=== 文件错误处理示例 ===\n");FILE *fp = fopen("nonexistent.txt", "r");if (fp == NULL) {perror("打开文件失败");  // perror会打印系统错误信息return;}// 尝试读取数据int num;if (fscanf(fp, "%d", &num) != 1) {if (feof(fp)) {printf("到达文件末尾\n");}if (ferror(fp)) {printf("文件读取发生错误\n");clearerr(fp);  // 清除错误标志}}fclose(fp);
}int main() {write_file_example();read_file_example();formatted_read_example();file_position_example();error_handling_example();return 0;
}

8.1 C程序的标准结构

#include <stdio.h>        // 头文件包含#define MAXSIZE 100       // 宏定义// 函数声明
int add(int a, int b);// 全局变量
int global_var = 0;// 主函数
int main() {// 局部变量int local_var = 10;// 程序逻辑printf("Hello, C!\n");return 0;
}// 函数定义
int add(int a, int b) {return a + b;
}

8.2 常见错误避免

语法错误:

  • 忘记分号
  • 括号不匹配
  • 变量未声明就使用
  • 数组下标越界

逻辑错误:

  • 指针未初始化就使用
  • 内存泄漏(malloc后忘记free)
  • 字符串操作越界
  • 文件操作后忘记关闭

8.3 调试技巧

#include <stdio.h>int main() {int a = 5, b = 3;// 使用printf调试printf("DEBUG: a=%d, b=%d\n", a, b);int result = a + b;printf("DEBUG: result=%d\n", result);return 0;
}

结语:C语言学习的进阶之路

恭喜你掌握了C语言二级考试的核心知识!🎉

C语言的学习不应止步于考试,掌握了这些基础后,你可以:

  • 深入学习数据结构与算法
  • 探索操作系统原理
  • 学习嵌入式开发
  • 为学习C++打下坚实基础

最后推荐一下我使用的题库,三端可用【支持苹果,安卓,pc】在手机上写操作题,特别是指针可视化和内存管理功能,让抽象的概念变得具体可见!和考试环境一致的评分系统也能让你提前适应考试节奏。

在这里插入图片描述

祝愿每位同学都能在C语言的世界里找到编程的乐趣! 💪


http://www.dtcms.com/a/315334.html

相关文章:

  • 风光储并网协同运行simulink仿真模型实现
  • [找出字符串中第一个匹配项的下标]
  • MiDSS复现
  • Codeforces Round 1010 (Div. 2, Unrated)
  • 8.4IO进程线程——进程
  • MySQL 基本操作入门指南
  • 代码随想录day55图论5
  • 通往L4之路:构建自我进化的智能驾驶决策大脑
  • Dubbo 3.x源码(32)—Dubbo Provider处理服务调用请求源码
  • CSS 安卓应用卸载碎片化动画效果
  • pyqt5-tools/pyqt6-tools 安装失败,解决办法
  • 【秋招笔试】2025.08.03虾皮秋招笔试-第三题
  • 7.2 I/O接口 (答案见原书 P305)
  • 大模型部署、nvidia-smi、token数
  • Java项目:基于SSM框架实现的商铺租赁管理系统【ssm+B/S架构+源码+数据库+毕业论文+开题报告+任务书+远程部署】
  • pytorch 学习笔记3-利用框架内网络训练糖尿病数据集
  • Linux 使用 firewalld :开放端口与常用操作指南
  • Endpoint(端点)详解
  • ROS2机器人编程新书推荐-2025-精通ROS 2机器人编程:使用ROS 2进行复杂机器人的设计、构建、仿真与原型开发(第四版)
  • 16_OpenCV_漫水填充(floodFill)
  • 【web应用】若依框架:基础篇18-二次开发-菜品管理
  • VGMP(VRRP Group Management Protocol)VRRP组管理协议
  • 知识蒸馏 - 基于KL散度的知识蒸馏 HelloWorld 示例 KL散度公式变化
  • Demo-LangGraph构建Agent
  • Assistant API——构建基于大语言模型的智能体应用
  • 通义万相国际版wan2.2开源第6天:主题运动
  • 二值图针对内部轮廓腐蚀膨胀
  • 李宏毅深度学习教程 第10-11章 自监督学习self-supervised learning+自编码器
  • FFmpeg02:常用命令实战
  • 【LeetCode 热题 100】215. 数组中的第K个最大元素——(解法一)快速选择