1.1.1 C语言常用的一些函数(持续更新)

总框架见(0. 总框架-CSDN博客)
(1)socket

(a)分配fd;(b)分配tcp控制块(tcb)

int socket(int domain, int type, int protocol);

       AF_INET             IPv4 Internet protocols          ip(7)
       AF_INET6            IPv6 Internet protocols          ipv6(7)
       AF_IPX              IPX - Novell protocols
(2)bind

把fd和ip绑定到tcp控制块

int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
(3)listen

开始监听fd

int listen(int sockfd, int backlog);
(4)accept

(a)为客户端分配fd;(b)分配tcp控制块(tcb)

int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
(5)recv

从内核copy数据到用户空间

 ssize_t recv(int sockfd, void *buf, size_t len, int flags);
(6)send

从用户空间copy数据到内核

ssize_t send(int sockfd, const void *buf, size_t len, int flags);
(7)connect

connect客户端发送连接请求

 int connect(int sockfd, const struct sockaddr *addr,
                   socklen_t addrlen);
(8)epoll_create
int epoll_create(int size);
(9)epoll_ctl
int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event);

    EPOLL_CTL_ADD       EPOLLIN        EPOLLLT 
    EPOLL_CTL_MOD       EPOLLOUT       EPOLLET
    EPOLL_CTL_DEL
(10)epoll_wait
int epoll_wait(int epfd, struct epoll_event *events,
                      int maxevents, int timeout);
(11)poll
int poll(struct pollfd *fds, nfds_t nfds, int timeout);
(12)select
int select(int nfds, fd_set *readfds, fd_set *writefds,
                  fd_set *exceptfds, struct timeval *timeout);

       void FD_CLR(int fd, fd_set *set);
       int  FD_ISSET(int fd, fd_set *set);
       void FD_SET(int fd, fd_set *set);
       void FD_ZERO(fd_set *set);
(13)read
ssize_t read(int fd, void *buf, size_t count);

参数说明
int fd
文件描述符,表示要读取的文件、管道、套接字或设备。
通常由系统调用(如 open()、socket() 等)返回。
void *buf
指向接收读取数据的缓冲区指针。
系统调用会将读取的数据写入到此缓冲区。
size_t count
要读取的最大字节数,指定缓冲区的大小。

返回值
成功时:
返回读取的字节数(可能小于请求的字节数 count)。
返回 0 表示已到达文件末尾(EOF)。
失败时:
返回 -1,并设置全局变量 errno 来指示错误原因
(14)write
ssize_t write(int fd, const void *buf, size_t count);

参数说明
int fd
文件描述符,表示数据要写入的目标。
通常由 open() 或 socket() 等函数返回。
const void *buf
指向包含要写入数据的缓冲区的指针。
size_t count
指定要写入的字节数

返回值
成功时:
返回实际写入的字节数(可能小于请求的字节数 count)。
失败时:
返回 -1,并设置全局变量 errno 表示错误原因。
(15)fork

创建的新进程,它是父进程的一个副本,几乎所有的数据(变量、文件描述符等)都会被复制。

返回值
父进程中:fork() 返回子进程的 PID(正整数)。
子进程中:fork() 返回 0。
出错时:返回 -1,同时设置 errno
(16)perror

读取系统定义的全局变量 errno,然后通过查找对应的错误码输出一个与其相关联的错误消息

int main() {
    int fd = open("nonexistent_file.txt", O_RDONLY);
    if (fd == -1) {
        perror("Error opening file");
    }
    return 0;

输出:
Error opening file: No such file or directory
(17)strerror

如果需要获取错误描述并在代码中使用,而不是直接打印,可以使用 strerror(errno)。

#include <string.h>
printf("Error: %s\n", strerror(errno));
(18)atoi

是 C 标准库中用于将字符串转换为整数的函数。atoi 是 ASCII to Integer 的缩写。

int atoi(const char *str);

参数
str
指向需要转换的字符串(应以空字符 \0 结束)。

返回值
如果字符串可以成功解析为整数,则返回该整数值。
如果字符串不包含有效的数字,则返回值可能为 0(不安全,推荐使用 strtol 替代)。

int main() {
    char str1[] = "   -456";
    char str2[] = "+789";

    printf("Converted number 1: %d\n", atoi(str1)); 
    // Output: -456
    printf("Converted number 2: %d\n", atoi(str2)); 
    // Output: 789
    return 0;
}
(19)strcmp

是 C 标准库中的一个函数,用于比较两个字符串的大小(基于字典序)。

int strcmp(const char *str1, const char *str2);

参数解释
str1
指向第一个字符串的指针。
str2
指向第二个字符串的指针

返回值
小于 0
如果 str1 的字典序小于 str2。
等于 0
如果 str1 和 str2 内容完全相同。
大于 0
如果 str1 的字典序大于 str2

int main() {
    char str1[] = "apple";
    char str2[] = "banana";

    int result = strcmp(str1, str2);
    printf("Result: %d\n", result); 
    // Output: Result: negative value (e.g., -1)
    return 0;
}
(20)memset

是 C 标准库中的一个函数,主要用于对一块内存区域进行初始化或设置。

void *memset(void *s, int c, size_t n);

参数说明
void *s
指向要初始化的内存区域的起始地址。
int c
用于设置的值,会被解释为一个 unsigned char(0 到 255),然后复制到内存区域。
size_t n
要设置的字节数。

返回值
返回指向内存区域 s 的指针。
功能
将指定的值填充到连续的内存区域中。
常用于数组或结构体的初始化。

常见用途
功能1:初始化数组
使用 memset 将数组清零或设置为某个默认值。
int arr[10];
memset(arr, 0, sizeof(arr)); // 初始化为 0

功能2:清空结构体中的所有成员变量。
struct Data {
    int a;
    char b[20];
} data;
memset(&data, 0, sizeof(data)); // 将结构体清零
(21)memcpy

是 C 标准库中的一个函数,将内存块中的数据从一个位置复制到另一个位置。

void *memcpy(void *dest, const void *src, size_t n);

参数说明
void *dest
目标内存地址,表示复制数据的目的地。
const void *src
源内存地址,表示复制数据的来源。
size_t n
要复制的字节数。

返回值
返回目标内存地址 (dest) 的指针。

基本用法
int main() {
    char src[] = "Hello, World!";
    char dest[20];
    memcpy(dest, src, strlen(src) + 1);  
    // 复制 src 到 dest
    printf("Source: %s\n", src);
    printf("Destination: %s\n", dest);
    return 0;
}

输出:
Source: Hello, World!
Destination: Hello, World!

int main() {
    int src[5] = {1, 2, 3, 4, 5};
    int dest[5];
    memcpy(dest, src, 5 * sizeof(int));  // 复制 5 个整型值
    for (int i = 0; i < 5; i++) {
        printf("dest[%d] = %d\n", i, dest[i]);
    }
    return 0;
}

输出:
dest[0] = 1
dest[1] = 2
dest[2] = 3
dest[3] = 4
dest[4] = 5

应用场景
字符串操作:复制字符串的原始字节数据(与 strcpy 的主要区别是 memcpy 可以复制非空终止的二进制数据)。
数组复制:处理任意类型的数组,例如 int 或 float。
内存初始化:与 memset 配合使用,分配和初始化内存。
结构体复制:在需要字节级拷贝时,处理复杂结构体数据。
(22)strncpy

复制字符串的指定长度

char *strncpy(char *dest, const char *src, size_t n);

参数说明
char *dest
目标字符数组的指针,复制后的字符串将存储在这里。
const char *src
源字符串的指针,要被复制的字符串内容。
size_t n
要复制的最大字符数。如果 src 长度小于 n,则目标字符串会填充空字符(\0);如果 src 长度大于或等于 n,则目标字符串不会被空字符终止。

返回值
返回 dest 指针,即目标字符串的指针。

功能
将源字符串 src 复制到目标字符数组 dest 中,最多复制 n 个字符。
如果源字符串长度小于 n,strncpy 会用空字符(\0)填充目标字符串的剩余部分;如果源字符串长度大于或等于 n,目标字符串没有终止符(\0)。

与 strcpy 的对比
strcpy:直接复制整个字符串,直到遇到终止符 \0。无法防止溢出。
例如:strcpy(dest, src);
strncpy:复制指定数量的字符,并会根据情况使用终止符 \0 填充目标字符串。
更安全,因为可以限制复制的字符数,防止溢出

int main() {
    char src[] = "Hello, World!";
    char dest[20];
    // 将最多 10 个字符从 src 复制到 dest
    strncpy(dest, src, 10);
    dest[10] = '\0';  // 确保终止符
    printf("Copied string: %s\n", dest);  
    // 输出: Hello, Wo
    return 0;
}

代码解析
strncpy(dest, src, 10);:最多复制 10 个字符。
然后手动添加终止符 dest[10] = '\0';,这是因为源字符串可能没有包含终止符(在字符串更长的情况下)。
结果是 dest 中的内容为 "Hello, Wo",因为 src 前 10 个字符被复制。
(23)strlen
size_t strlen(const char *str);

功能
strlen 用于计算字符串的长度(不包括终止符 \0)。
它会遍历字符串直到遇到第一个空字符 \0 为止,并返回已遍历的字符数。
参数说明
const char *str
指向以空字符 \0 结尾的字符串。
返回值
返回字符串的长度(不包括终止符 \0)的大小,以 size_t 类型表示。

int main() {
    char str[] = "Hello, World!";
    size_t length = strlen(str);
    printf("The length of the string is: %zu\n", length); 
    // 输出: 13
    return 0;
}

代码解析
字符串 str[]:
包含 "Hello, World!",长度为 13(不包括终止符 \0)。
strlen(str):
遍历字符串,直到遇到终止符 \0。
计算字符数:'H', 'e', 'l', 'l', 'o', ',', ' ', 'W', 'o', 'r', 'l', 'd', '!'。
返回值:
返回 13,表示字符串的长度

与 sizeof 的对比
strlen运行时计算字符串长度。适用于动态字符串。不包括终止符 \0。
sizeof编译时计算数组大小(字节数)。适用于固定大小的字符数组。包括终止符 \0 的大小。
示例:
char str1[] = "Hello";
char *str2 = "Hello";
printf("strlen(str1): %zu\n", strlen(str1));  // 输出 5
printf("sizeof(str1): %zu\n", sizeof(str1));  // 输出 6
printf("strlen(str2): %zu\n", strlen(str2));  // 输出 5
printf("sizeof(str2): %zu\n", sizeof(str2));  // 输出 8 (指针大小)
(24)snprint

用于将格式化的数据写入字符串,并确保不会溢出字符串缓冲区。它是 sprintf 的更安全版本,避免了可能的缓冲区溢出问题。

int snprintf(char *str, size_t size, const char *format, ...);

参数说明
char *str
指向要写入格式化输出的字符数组。
size_t size
指定目标字符串的最大大小,包括结尾的 '\0'(空字符)。snprintf 确保总输出长度不超过这个值。
const char *format
格式化字符串,与 printf 的格式化字符串类似,例如 %d, %s, %f 等。
... (可变参数)
需要插入到格式化字符串中的数据

返回值
返回写入的字符数,不包括终止的 \0 字符:
如果 size 足够大,返回值是实际写入的字符数。
如果 size 不够大,返回值是本应该写入的字符数(截断的字符串长度)。

缓冲区不足的情况
int main() {
    char buffer[10];
    int num = 123456;
    int written = snprintf(buffer, sizeof(buffer), "Number: %d", num);
    printf("Formatted string: %s\n", buffer);  // 输出截断的内容
    printf("Characters written: %d\n", written);  // 返回未截断的实际字符数
    return 0;
}

输出:
Formatted string: Number: 1
Characters written: 12
说明:
缓冲区 buffer 长度为 10,无法容纳完整字符串 "Number: 123456"。
结果字符串被截断为 "Number: 1",但 snprintf 的返回值是实际所需的字符数 12
(25)malloc

堆上分配内存,并返回指向分配内存块的指针。

void *malloc(size_t size);

参数说明
size
请求分配的内存大小(以字节为单位)。size 必须大于 0。

返回值
返回分配的内存的起始地址,类型是 void*,即通用指针。
如果内存分配失败(比如系统没有足够的内存),malloc 返回 NULL。

特性
动态内存分配
malloc 函数从程序的堆内存区域(Heap)分配指定大小的内存。在堆上分配的内存空间不会随着函数调用结束而销毁,必须通过 free 显式释放。

基本用法
int main() {
    int *arr = (int *)malloc(5 * sizeof(int));  // 动态分配 5 个整数的空间
    if (arr == NULL) {
        printf("Memory allocation failed!\n");
        return 1;
    }
    for (int i = 0; i < 5; i++) {
        arr[i] = i + 1;
        printf("%d ", arr[i]);
    }
    printf("\n");
    free(arr);  // 释放内存
    return 0;
}

输出:
1 2 3 4 5
(26)free

用于释放由动态内存分配函数(如 malloc、calloc 或 realloc)分配的内存。将之前分配的动态内存归还给操作系统或内存管理器。释放后,该内存块可以被再次分配和使用。

void free(void *ptr);

参数说明
void *ptr
动态分配的内存块的指针,该指针通常是通过 malloc、calloc 或 realloc 返回的。
如果 ptr 是 NULL,free 不会进行任何操作。

基本用法
int main() {
    int *arr = (int *)malloc(5 * sizeof(int));  // 动态分配 5 个整数的空间
    if (arr == NULL) {
        perror("malloc failed");
        return 1;
    }
    for (int i = 0; i < 5; i++) {
        arr[i] = i + 1;
        printf("%d ", arr[i]);
    }
    printf("\n");
    free(arr);  // 释放内存
    arr = NULL; // 避免悬空指针
    return 0;
}

输出:
1 2 3 4 5

常见问题
1. 为什么释放未分配的内存是危险的?
如果尝试 free 一个未分配的指针,可能会导致程序崩溃或者未定义行为,因为内存管理器试图释放一个它没有跟踪的内存区域。
2. 为什么要将指针置为 NULL?
避免悬空指针(Dangling Pointer)。
悬空指针是指指向已释放内存区域的指针,如果不小心使用它(如读写操作),可能会导致不可预测的错误。
3. 始终配对 malloc 和 free,避免内存泄漏和悬空指针。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/954539.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

向量数据库Milvus详解

向量数据库Milvus详解 0. 什么是向量数据库? 在现实世界中,并非所有数据都可以整齐地放到行和列中。在处理图像、视频和自然语言等复杂的非结构化数据时尤其如此。这就是向量数据库的用武之地。 向量数据库是一种以高维向量的形式来存储数据的数据库,这些向量本质上是表示…

海豚调度DolphinScheduler-3.1.9配置windows本地开发环境

源代码下载地址https://dolphinscheduler.apache.org/zh-cn/docs/3.1.9 1.Zookeeper安装与使用 如图下载解压zookeeper安装包&#xff0c;并创建data和log目录 下载地址 https://archive.apache.org/dist/zookeeper/zookeeper-3.6.4/apache-zookeeper-3.6.4-bin.tar.gz 进入…

springCloudGateway+nacos自定义负载均衡-通过IP隔离开发环境

先说一下想法&#xff0c;小公司开发项目&#xff0c;参考若依框架使用的spring-cloud-starter-gateway和spring-cloud-starter-alibaba-nacos, 用到了nacos的配置中心和注册中心&#xff0c;有多个模块&#xff08;每个模块都是一个服务&#xff09;。 想本地开发&#xff0c;…

大模型训练_硬件微调知识增强

目录 关键硬件 大模型类型 垂域训练技术 微调技术 领域大模型训练trick 知识增强 关键硬件 GPU GPU擅长处理图形渲染和数据并行任务&#xff0c;可以同时处理大量的矩阵运算&#xff0c;在科学计算、人工智能、游戏开发等领域应用广泛。 显卡 显卡是一种完整的硬件设…

linux分配磁盘空间命令

使用命令lsblk查询linux磁盘空间时&#xff0c;发现空间并没有被分配完 如图&#xff0c;600G&#xff0c;但实际分配了一共199G&#xff0c;剩余500G&#xff0c;我们需要通过命令进行剩余存储的分配。 思路&#xff1a;创建新的分区->更新内核分区表->初始化新分区作…

【运维自动化-作业平台】魔法变量到底如何使用之主机列表类型

蓝鲸作业平台&#xff0c;以下简称作业平台或JOB平台 魔法变量&#xff1a;JOB平台执行引擎提供的特有的变量能力用法 脚本中使用&#xff0c;并且需要事先声明&#xff1a;job_import {{变量名}} 声明后&#xff0c;同样是使用 dollar 符 大括号&#xff1a;${变量名}来取值…

Windows重装后NI板卡LabVIEW恢复正常

在重新安装Windows系统后&#xff0c;NI&#xff08;National Instruments&#xff09;板卡能够恢复正常工作&#xff0c;通常是由于操作系统的重新配置解决了之前存在的硬件驱动、兼容性或配置问题。操作系统重装后&#xff0c;系统重新加载驱动程序、清理了潜在的冲突或损坏的…

Docker启动达梦 rman恢复

目录标题 1. 主库备份2. Docker启动备库3. 备库修改属组4. 开始恢复5. 连接数据库配置归档 & Open6. 检查数据 关于达梦数据库&#xff08;DMDBMS&#xff09;的主库备份、Docker启动备库、恢复备份以及配置归档和打开数据库的详细步骤。 1. 主库备份 # 使用达梦数据库备…

【C语言】_字符串拷贝函数strcpy

目录 1. 函数声明及功能 2. 使用示例 3. 注意事项 4. 模拟实现 4.1 第一版&#xff1a;基本功能判空const修饰 4.2 第二版&#xff1a;优化对于\0的单独拷贝 4.3 第三版&#xff1a;仿strcpy的char*返回值 1. 函数声明及功能 char * strcpy ( char * destination, cons…

XML序列化和反序列化的学习

1、基本介绍 在工作中&#xff0c;经常为了调通上游接口&#xff0c;从而对请求第三方的参数进行XML序列化&#xff0c;这里常使用的方式就是使用JAVA扩展包中的相关注解和类来实现xml的序列化和反序列化。 2、自定义工具类 import javax.xml.bind.JAXBContext; import javax.x…

js使用qrcode与canvas生成带logo的二维码

qrcode库 文档 https://www.npmjs.com/package/qrcode 安装 npm i qrcode 使用 errorCorrectionLevel: H // 容错率&#xff08;H是最高&#xff0c;其它看文档&#xff09; width: 200 // 大小 margin: 2 // 边距 import QRCode from qrcodeconst testFn async () > {c…

【计算机网络】lab5 ARP协议

&#x1f308; 个人主页&#xff1a;十二月的猫-CSDN博客 &#x1f525; 系列专栏&#xff1a; &#x1f3c0;计算机网络_十二月的猫的博客-CSDN博客 &#x1f4aa;&#x1f3fb; 十二月的寒冬阻挡不了春天的脚步&#xff0c;十二点的黑夜遮蔽不住黎明的曙光 目录 1. 前言 2.…

【React】脚手架进阶

目录 暴露webpack配置package.json的变化修改webpack.config.js配置less修改域名、端口号浏览器兼容处理处理跨域 暴露webpack配置 react-scripts对脚手架中的打包命令进行封装&#xff0c;如何暴露这些打包配置呢&#xff1f;上篇写到在package.json中的scripts配置项中有eje…

java项目之现代企业人力资源管理系统设计与实现(源码+文档)

大家好我是风歌&#xff0c;今天要和大家聊的是一款基于ssm的现代企业人力资源管理系统设计与实现。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 项目简介&#xff1a; 现代企业人力资源管理系统设计与实现的主要使用者分为管理员、经理和普通员工三个角…

2025.1.15——三、报错注入

一、基本操作&#xff1a;整理已知信息&#xff0c;本题为报错注入&#xff0c;需进一步确认回显方式 二、用updatexml()解题步骤 step 1&#xff1a;依据回显方式判断题目类型 键入&#xff1a;1、1 and 11 、id2-1 得到&#xff1a;查询正确的回显 键入&#xff1a;1’、…

【IDEA 2024】学习笔记--文件选项卡

在我们项目的开发过程中&#xff0c;由于项目涉及的类过多&#xff0c;以至于我们会打开很多的窗口。使用IDEA默认的配置&#xff0c;个人觉得十分不便。 目录 一、设置多个文件选项卡按照文件字母顺序排列 二、设置多个文件选项卡分行显示 一、设置多个文件选项卡按照文件字…

自己动手搭建“接入 AI Agent 的数字人”

前言 本文的实战案例来自于开源项目&#xff1a;https://github.com/wan-h/awesome-digital-human-live2d。该项目可以运用Dify编排框架和Live2D驱动模型搭建智能数字人&#xff0c;实现智能对话并动态交互&#xff0c;大家可以自行部署尝试&#xff0c;项目效果如下。 开源&a…

Flink链接Kafka

一、基于 Flink 的 Kafka 消息生产者 Kafka 生产者的创建与配置&#xff1a; 代码通过 FlinkKafkaProducer 创建 Kafka 生产者&#xff0c;用于向 Kafka 主题发送消息。Flink 执行环境的配置&#xff1a; 配置了 Flink 的检查点机制&#xff0c;确保消息的可靠性&#xff0c;支…

电脑有两张网卡,如何实现同时访问外网和内网?

要是想让一台电脑用两张网卡&#xff0c;既能访问外网又能访问内网&#xff0c;那可以通过设置网络路由还有网卡的 IP 地址来达成。 检查一下网卡的连接 得保证电脑的两张网卡分别连到外网和内网的网络设备上&#xff0c;像路由器或者交换机啥的。 给网卡配上不一样的 IP 地…

dockerfile2.0

dockerfile实现lnmp nginx centos7 mysql centos7 php centos7 自定义镜像来实现整个架构 cd /opt mkdir nginx mysql php cd nginx 拖入nginx和wordpress vim Dockerfile vim nginx.conf ↓ worker_processes 1; events {worker_connections 1024; } http {include …