C语言补习课

2024-08-31 19:12
文章标签 语言 补习

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

来源:黑马程序员

C语言常用占位符

1.%d 或 %i:用于输出有符号整数(int)。

2.%u:用于输出无符号整数(unsigned int)。

3.%f:用于输出浮点数(floatdouble)。默认情况下,输出六位小数。

4.%lf:专门用于输出double类型的浮点数(尽管%f也可以,但%lf更明确)。

5.%c:用于输出单个字符(char)。

6.%s:用于输出字符串(char数组)。

7.%x 或 %X:用于输出无符号整数(unsigned int)的十六进制形式,%x使用小写字母,%X使用大写字母。

8.%o:用于输出无符号整数(unsigned int)的八进制形式。

9.%%:输出一个%字符。

10.%p:用于输出指针的值,以十六进制形式表示。

11.%zu:专门用于输出size_t类型的无符号整数值。

第146讲 共同体

引入共同体的原因是一种数据可能有多种类型。

语法:

union [共同体名称]{};

示例:

union MoneyType {int moneyi;double moneyd;char  moneystr[100];
};

注意每个类型的共同体成员不能重名,并且每次只能赋一个值!

C语言给字符串型变量进行赋值建议用strcpy函数,函数原型如下:

char *strcpy(char *dest, const char *src);

变量介绍:

dest 是指向用于存储复制内容的目标数组(字符串)的指针。

src 是指向要复制的源字符串的指针。

函数返回目标字符串的指针。

我这里用的visual studio推荐用strcpy_s,中间需要加一个参数代表原字符串型变量的大小。

参考代码:

#include <stdio.h>
#include <string.h>typedef union{int moneyi;double moneyd;char  moneystr[100];
} MoneyType;int main()
{MoneyType money;//money.moneyi = 100;//money.moneyd = 99.5;strcpy_s(money.moneystr,sizeof(money.moneystr),"100万");//printf("%d\n",money.moneyi);//printf("%lf\n", money.moneyd);printf("%s\n", money.moneystr);return 0;
}

第147讲 共用体的特点

共用体的特点:

● 共用体,也叫联合体,共同体

● 所有的变量都使用同一个内存空间

● 所占的内存大小=最大成员的长度(也受内存对齐影响,总大小也一定是最大成员的整数倍)

● 每次只能给一个变量进行赋值,因为第二次赋值时会覆盖原有的数据

在上一讲的基础上运行下面的代码

	printf("%p\n",&(money.moneyi));printf("%p\n",&(money.moneyd));printf("%p\n",&(money.moneystr));

运行结果:

得证所有的变量都使用同一个内存空间。

那么上一讲的money到底占用多少内存呢?

	printf("%zu",sizeof(money));

输出结果:

原因是char  moneystr[100]并不能视为最大长度的成员,它是100个char类型的变量组成的数组。最大长度成员是double,也就是说这个联合体的大小不仅得大于100,而且得是8的倍数。100不起字节,加4刚好是8的倍数。

第148讲 结构体和共同体的区别

使用场景:

              结构体:一个事物包含多种属性

               联合体:一个属性有多种类型

存储方式:

                结构体:各存各的

                共用体:存一起,多次存会覆盖

内存占用:

               结构体:各个变量的总和(受内存对齐影响)

              共用体:最大类型(受内存对齐影响)

第149讲 动态内存分配常用函数

总览

注意:这四个函数需要#include <stdlib.h>才能使用。

malloc

malloc函数适合用来申请连续的空间,函数原型如下所示:

void* malloc(size_t size);

通过 malloc,程序可以在堆区(heap)动态地请求分配指定大小的内存块,并在分配成功后返回一个指向该内存块的指针。如果分配失败(例如,由于内存不足),malloc 会返回 NULL 指针。

参数size 指定了要分配的内存大小,单位是字节(byte)。

由于数组是占用一段连续内存空间的数据结构,所以我们可以直接用数组来操作这一段内存空间。

举例:

int *p=malloc(10 * sizeof(int));

指针p指向的连续内存可以存储10个int类型数据,这时候可以直接给p[0]到p[9]赋值。

p[0]=100;

总体示例:

#include <stdio.h>
#include <stdlib.h>int main()
{//开辟一片能存储100个int类型整数的连续内存空间int *p=malloc(10 * sizeof(int));printf("%p\n",p);for (int i = 0; i < 10; i++){//赋值方法1 指针步进//*(p + 1) = (i + 1) * 10;//赋值方法2 遍历数组p[i]= (i + 1) * 10;}for (int i = 0; i < 10; i++){printf("%d ",p[i]);}return 0;
}

calloc

函数原型:

void *calloc(size_t num, size_t size);

参数:①num 要分配的元素数量

           ②size  每个元素的大小 

返回值:成功时,返回一个指向分配的内存的指针。这块内存的大小至少是 num * size 字节,并且所有位都被初始化为零。如果分配失败(例如,因为内存不足),则返回 NULL 指针。

重点掌握malloc函数,效率更高;calloc的初始化就是默认赋值为0作用不大。

realloc 

函数原型:

void* realloc( void*  _Block, size_t _Size);

参数:①通用指针_Block 指向想要修改的内存块的指针

           ②size 想要扩充的字节大小

realloc可以扩容内存并且保存原来的数据,但是新开辟的空间里面存储的是乱码

#include <stdio.h>
#include <stdlib.h>int main()
{int* p = calloc(10,sizeof(int));for (int j = 0; j < 10; j++) {p[j] = j * 10+1;printf("%d ",p[j]);}printf("\n");int *pp=realloc(p,20*sizeof(int));for (int i = 0; i < 20; i++) {printf("%d ", p[i]);}return 0;
}

运行结果:

另外最好不要用这个函数,影响代码安全性。上次windows蓝屏不就是因为微软新来了一个小孩,他的野指针在天上飞么?(bushi

free

函数原型:

void free(void* _Block);

不用了的内存空间最好随手释放,填入你想要释放的内存块的地址。

第150讲 malloc函数的细节点

动态内存分配的小细节:

1,malloc创建空间的单位是字节

因此malloc(100)之后能存储25个int类型数据,50个short型数据。

2,malloc返回的是void类型的指针,没有步长的概念,也无法获取空间中的数据,需要强转

malloc返回类型是void*,void *p无法进行*(p+1)这样的操作,正确做法:

	int *p=(int *)malloc(100);*(p + 1) = 10;

当然也可以强转成其他返回类型

3,malloc返回的仅仅是首地址,没有总大小,最好定义一个变量记录总大小

上文的int*规定了指针步长,p指针指向了内存的首地址,所以开辟一个100字节的内存可以存储25个int类型。

#include <stdio.h>
#include <stdlib.h>#define length 25int main()
{int* p = (int*)malloc(length * sizeof(int));for (int i = 0; i < length; i++) {p[i] = i * 4 + 7;}printPoint(p,length);return 0;
}int printPoint(int* point, int len)
{for (int i = 0; i < len; i++) {printf("%d ",point[i]);}printf("\n");
}

4,malloc申请的空间不会自动消失,如果不能正确释放,会导致内存泄露

5,malloc申请的空间过多,会产生虚拟内存
当申请的空间过多,因为每一个内存空间不会在刚申请的时候就立马使用,所以c语言并不会立马就在内存中去开辟空间,而是什么时候存储数据了,才会真正的分配空间
目的:为了提高内存的使用效率

6,malloc申请的空间没有初始化值,需要先赋值才能使用

7,free释放完空间之后,空间中数据叫做脏数据,可能被清空,可能被修改为其他值

8,calloc就是在malloc的基础上多一个初始化的动作

9,realloc修改之后的空间,地址值有可能发生变化,也有可能不会改变,但是原本的数据不会丢失

10,realloc修改之后,无需释放原来的空间,函数底层会进行处理

这篇关于C语言补习课的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

基于Go语言实现Base62编码的三种方式以及对比分析

《基于Go语言实现Base62编码的三种方式以及对比分析》Base62编码是一种在字符编码中使用62个字符的编码方式,在计算机科学中,,Go语言是一种静态类型、编译型语言,它由Google开发并开源,... 目录一、标准库现状与解决方案1. 标准库对比表2. 解决方案完整实现代码(含边界处理)二、关键实现细

如何合理管控Java语言的异常

《如何合理管控Java语言的异常》:本文主要介绍如何合理管控Java语言的异常问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、介绍2、Thorwable类3、Error4、Exception类4.1、检查异常4.2、运行时异常5、处理方式5.1. 捕获异常

C语言中的常见进制转换详解(从二进制到十六进制)

《C语言中的常见进制转换详解(从二进制到十六进制)》进制转换是计算机编程中的一个常见任务,特别是在处理低级别的数据操作时,C语言作为一门底层编程语言,在进制转换方面提供了灵活的操作方式,今天,我们将深... 目录1、进制基础2、C语言中的进制转换2.1 从十进制转换为其他进制十进制转二进制十进制转八进制十进

$在R语言中的作用示例小结

《$在R语言中的作用示例小结》在R语言中,$是一个非常重要的操作符,主要用于访问对象的成员或组件,它的用途非常广泛,不仅限于数据框(dataframe),还可以用于列表(list)、环境(enviro... 目录1. 访问数据框(data frame)中的列2. 访问列表(list)中的元素3. 访问jav

C语言中位操作的实际应用举例

《C语言中位操作的实际应用举例》:本文主要介绍C语言中位操作的实际应用,总结了位操作的使用场景,并指出了需要注意的问题,如可读性、平台依赖性和溢出风险,文中通过代码介绍的非常详细,需要的朋友可以参... 目录1. 嵌入式系统与硬件寄存器操作2. 网络协议解析3. 图像处理与颜色编码4. 高效处理布尔标志集合

Go语言开发实现查询IP信息的MCP服务器

《Go语言开发实现查询IP信息的MCP服务器》随着MCP的快速普及和广泛应用,MCP服务器也层出不穷,本文将详细介绍如何在Go语言中使用go-mcp库来开发一个查询IP信息的MCP... 目录前言mcp-ip-geo 服务器目录结构说明查询 IP 信息功能实现工具实现工具管理查询单个 IP 信息工具的实现服

C 语言中enum枚举的定义和使用小结

《C语言中enum枚举的定义和使用小结》在C语言里,enum(枚举)是一种用户自定义的数据类型,它能够让你创建一组具名的整数常量,下面我会从定义、使用、特性等方面详细介绍enum,感兴趣的朋友一起看... 目录1、引言2、基本定义3、定义枚举变量4、自定义枚举常量的值5、枚举与switch语句结合使用6、枚

Go 语言中的select语句详解及工作原理

《Go语言中的select语句详解及工作原理》在Go语言中,select语句是用于处理多个通道(channel)操作的一种控制结构,它类似于switch语句,本文给大家介绍Go语言中的select语... 目录Go 语言中的 select 是做什么的基本功能语法工作原理示例示例 1:监听多个通道示例 2:带

C语言函数递归实际应用举例详解

《C语言函数递归实际应用举例详解》程序调用自身的编程技巧称为递归,递归做为一种算法在程序设计语言中广泛应用,:本文主要介绍C语言函数递归实际应用举例的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录前言一、递归的概念与思想二、递归的限制条件 三、递归的实际应用举例(一)求 n 的阶乘(二)顺序打印

C语言中的数据类型强制转换

《C语言中的数据类型强制转换》:本文主要介绍C语言中的数据类型强制转换方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C语言数据类型强制转换自动转换强制转换类型总结C语言数据类型强制转换强制类型转换:是通过类型转换运算来实现的,主要的数据类型转换分为自动转换