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

【C语言】超详解strncpystrncatstrncmpstrerrorperror的使⽤和模拟实现

  🌈write in front :

🔍个人主页 : @啊森要自信的主页

✏️真正相信奇迹的家伙,本身和奇迹一样了不起啊!

欢迎大家关注🔍点赞👍收藏⭐️留言📝>希望看完我的文章对你有小小的帮助,如有错误,可以指出,让我们一起探讨学习交流,一起加油鸭。 请添加图片描述

文章目录

  • 📝前言
  • 🌠 库函数strncpy
    • 🌉strncpy 模拟实现
  • 🌠strncat 函数的使⽤
    • 🌉strncat 模拟实现
  • 🌠strncmp函数的使⽤
    • 🌉strncmp模拟实现
  • 🌠strerror
    • 🌉 perror
  • 🚩总结


📝前言

本小节,阿森继续和你一起学习5个字符串函数:strncpystrcnatstrncmp的使用和两种模拟实现方法,他们和strcpy等函数比较多了一个n ,实现方法有很大区别,还有strerrorperror的使用,学习这些库函数,可以更好的方便操作字符和字符串,文章干货满满,接下来我们就学习一下这些函数吧!


strcpystrcat这类函数不安全,因为它们在复制字符串时不检查目标缓冲区的大小,可能会导致缓冲区溢出
strncpystrncatstrncmp这类函数相对来说更安全,因为它们在复制/追加字符串时会限定最大长度参数n,避免无限制地写入目标缓冲区。
点击—>手把手教你配置VS的常见函数如何不报错!

在这里插入图片描述

🌠 库函数strncpy

strncpy函数用于将一个字符串拷贝到另一个字符串中,可以限定拷贝的字符数
函数原型:

 char * strncpy ( char * destination, const char * source, size_t num );dest - 目标字符串,用于接收拷贝内容。src - 源字符串,从中拷贝内容。 num - 要拷贝的字符数。

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

注意点:

  • 检查dest空间是否足以容纳srcn个字符及结尾'\0'strncpy不会检查dest的长度,如果dest空间不足可能会导致缓冲区溢出。

  • 拷⻉num个字符从源字符串到⽬标空间。拷贝num个就num个,不会拷贝多,也不会自己添加\0
    在这里插入图片描述

  • 如果源字符串的⻓度⼩于num,则拷⻉完源字符串之后,在⽬标的后边追加0,直到num个。
    在这里插入图片描述
    例子:

# define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
int main()
{char str1[] = "Hello World";char str2[7];str2[5] = '\0';strncpy(str2, str1, 5);printf("str1: %s\n", str1);printf("str2: %s\n", str2);return 0;
}

输出:

str1: Hello World
str2: Hello

🌉strncpy 模拟实现

对于strncpy函数,阿森给你带来了两种模拟实现方法,详解如下:

  • 主函数(两种模拟实现都可以用这个进行测试)
int main() 
{char dest[20]="xxxxxxxxxxxxxxxxx";char src[] = "hello";size_t set = strlen(src);printf("%d\n", set);my_strncpy(dest, src, 3);printf("%s\n", dest);my_strncpy(dest, src, 9);printf("%s\n", dest);return 0;
}
  1. 数组模拟实现
char* my_strncpy(char* dest, const char* src, size_t n) 
{if (dest == NULL || src == NULL) //检查dest和src参数是否合法,如果任意一个为NULL则直接返回NULL。{return NULL;}char* result = dest;// 保存dest的地址值,后面返回时使用size_t i;for ( i = 0; i < n && src[i] != '\0'; i++)  //使用for循环复制字符。{									     	 // i < n判断是否已经复制n个字符dest[i] = src[i];				          //  src[i] != '\0' 判断当前源字符串字符是否结束判断是否已经复制n个字符}//复制源字符串当前字符到目标字符串// 添加'\0'填充   (如果源字符串的⻓度⼩于`num`,则拷⻉完源字符串之后,在⽬标的后边追加`0`,直到`num`个。)while (i < n) 如果for循环结束但i未达到n,使用while循环填充'\0'。{dest[i++] = '\0';//将目标字符串当前位置字符填充为'\0'}							//dest[i++] = '\0'先dest[i]='\0',后i++return result;
}

输出:

5
helxxxxxxxxxxxxxx
hello

调试界面:
在这里插入图片描述

  1. 指针实现

char* my_strncpy(char* dest, const char* src, size_t n)
{assert(dest);//利用断言需要使用头文件#include<assert.h>assert(src);char* destPtr = dest;//定义dest和src的指针变量destPtr和srcPtr,用于遍历字符串。const char* srcPtr = src;while (n-- > 0) //使用while循环遍历n个字符{if (*srcPtr != '\0')  //检查当前源字符串srcPtr指向的字符是否为'\0'结束符{*destPtr++ = *srcPtr++;//如果不是结束符,就将源字符串当前字符复制到目标字符串,}										//并同时将两个指针前移到下一个字符。else //如果是结束符,进入else块{*destPtr++ = '\0';//将目标字符串当前字符设置为结束符'\0'}							  //然后destPtr再++}return dest; //返回目标字符串首地址。
}

输出:
在这里插入图片描述

*destPtr++ = *srcPtr++先进行一次赋值(*dest = *src),然后并使指针后移(dest=dest+1,src=src+1)
*destPtr++ = ‘\0’将目标字符串当前字符设置为结束符'\0',然后destPtr++

🌠strncat 函数的使⽤

strncat函数用于连接两个字符串,将源字符串src连接到目标字符串dest的结尾,最多连接n个字符。

strncat函数的原型:

char *strncat(char *dest, const char *src, size_t n);dest:目标字符串,其内容将在其后追加源字符串内容。src:源字符串,其内容将被追加到目标字符串结尾。 n:要从源字符串中追加到目标字符串中的最大字符数。
  • 返回值:
    函数返回目标字符串dest的指针。

例子:

#include <string.h>
int main()
{char dest[100] = "Hello";char src[] = " World";strncat(dest, src, 6);printf("%s\n", dest);
}

输出:

输出 Hello World

🌉strncat 模拟实现

  • 主函数
int main()
{char str1[100] = "hello";char str2[100] = " world";my_strncat(str1, str2, 5);printf("%s\n", str1);return 0;}
  1. 数组模拟实现
char* my_strncat(char* dst, const char* src, size_t n)
{char* tmp = dst;while (*dst)//使用while循环遍历dst字符串。{dst++;//找到字符串结束位置'\0'。}int i;for (i = 0; src[i] && i < n; i++)//  i < n 判断是否超过最大复制长度n{											 //src[i] 判断源字符串是否结束dst[i] = src[i];}dst[i] = 0;//在目标字符串末尾添加字符串结束标记'\0'。return tmp;
}

输出:

hello worl

在这里插入图片描述

  1. 指针实现
char* my_strncat(char* dest, const char* src, size_t n)
{//参数检查if (dest == NULL || src == NULL){return NULL;}char* ptr = dest;//找到目标字符串结尾while (*dest != '\0'){dest++;}while (n-- > 0 && *src != '\0'){*dest++ = *src++;}*dest = '\0';return ptr;//添加字符串结束符
}

输出:
在这里插入图片描述

首先,n-- 表示先使用 n 的值来进行比较是否>0,因为&& 是逻辑与运算符,*src != ‘\0’ 表示判断指针 src 所指向的字符是否为字符串的结束符 \0这两个条件验证真假后,最后 n 的值才减 1

🌠strncmp函数的使⽤

strncmp用于比较两个字符串的前n个字符。(比较的不是字符串的长度无关,只与对应位置的字符内容有关。)

strncmp函数原型:

int strncmp(const char *str1, const char *str2, size_t n);
str1 - 要比较的第一个字符串的指针
str2 - 要比较的第二个字符串的指针  
n - 将被比较的最大字符数

返回值:

  • 如果str1小于str2,返回值小于0
  • 如果str1大于str2,返回值大于0
  • 如果str1等于str2,返回值等于0

注意点:

  1. 如果n的值大于两个字符串中任意一个字符串的长度,比较将会超出字符串的范围,可能导致内存访问错误。因此,在使用strncmp函数时,需要确保n的值不会超过任意一个字符串的长度。

  2. strncmp函数返回的结果是一个整数,可以通过结果的正负值来判断两个字符串的大小关系。

  3. 比较规则与strcmp函数一致,按ASCII码顺序比较每个字符。

使用示例:

int main()
{char str1[] = "hello";//注意字符串结尾后面还有\0char str2[] = "hello world";int result1 = strncmp(str1, str2, 5);// 只比较前5个字符,结果为0,表示相等printf("%d\n", result1);int result2 = strncmp(str1, str2, 6);// 比较前6个字符,结果为负数,表示str1小于str2printf("%d\n", result2);}

输出:
在这里插入图片描述

🌉strncmp模拟实现

int my_strncmp(const char* s1, const char* s2, size_t n)
{int i = 0;//这是一个 for 循环,用于迭代比较两个字符串中的字符。for (; i < n && s1[i] != '\0' && s2[i] != '\0'; i++){if (s1[i] != s2[i]){return s1[i] - s2[i];//如果当前位置的两个字符不相等,返回它们的差值。}}if (i <= n){return s1[i] - s2[i];//如果 i 小于等于 n,但是循环结束了(即至少一个字符串已经达到结束符 ‘\0’),则返回当前位置字符的差值。}return 0;
}
int main()
{char s1[] = "hello";char s2[] = "helloworld";int result = my_strncmp(s1, s2, 5);printf("result = %d\n", result);return 0;
}

运行:
在这里插入图片描述

监视:
在这里插入图片描述

图解:
在这里插入图片描述

🌠strerror

错误码错误描述
0No error
1Operation not permitted
2No such file or directory
3No such process
4Interrupted function call
5Input/output error
6No such device or address
7Arg list too long
8Exec format error
9Bad file descriptor

strerror函数用于将错误码转换为对应的错误信息字符串。
函数原型如下:

char *strerror(int errnum);
errnum: 错误码号,通常是系统调用或库函数返回的错误号。   

strerror函数接受一个整型参数errnum,表示错误码。它会返回一个指向错误信息字符串的指针。

注意点:

  • 在不同的系统和C语⾔标准库的实现中都规定了⼀些错误码,⼀般是放在 errno.h 这个头⽂件中#include <errno.h>
  • C语⾔程序启动的时候就会使⽤⼀个全⾯的变量errno来记录程序的当前错误码,只不过程序启动的时候errno0,表⽰没有错误。
  • 当我们在使⽤标准库中的函数的时候发⽣了某种错误,就会讲对应
    的错误码,存放在errno
  • 以每⼀个错误码都是有对应的错误信息的
  • strerror函数返回的是一个静态字符串指针,不需要手动释放内存。

举栗子:

#include <errno.h>
#include <string.h>
#include <stdio.h>
int main()
{int i = 0;for (i = 0; i < 10; i++){printf("%d: %s\n",i, strerror(i));}return 0;
}

输出:
在这里插入图片描述

如何使用strerror函数打印打开文件失败的错误信息:

int main()
{FILE* pFile;pFile = fopen("unexist.txt", "r");//使用fopen函数打开文件"unexist.txt",以只读方式打开。if (pFile == NULL)//判断打开结果pFile是否为NULL,NULL表示打开失败。printf("Error opening file unexist.ent: %s\n", strerror(errno));elseprintf("打开文件成功\n");return 0;
}

输出:
在这里插入图片描述
分析:

Error opening file unexist.ent: No such file or directory说明打开文件"unexist.txt"失败,失败原因是文件不存在(ENOENT错误码)
在这里插入图片描述
如果加上"unexist.txt"该文件,就会显示打开成功!
在这里插入图片描述
在这里插入图片描述

🌉 perror

perror函数用于打印错误信息。它的功能与strerror函数类似,但打印方式不同。

perror函数原型:

void perror(const char *s);
s: 可选的错误前缀信息。

简意:

perror函数直接打印到标准错误输出,打印完参数部分的字符串后,再打印⼀个冒号和⼀个空格,再打印错误信息。(此代码结果为下面代码运行)
在这里插入图片描述

详解:
errno设置的错误号转换为错误描述字符串,然后打印到标准错误输出stderr上。如果s不为空,则在错误描述前加上s后跟 冒号 ":"。(stderr是预定义的一个文件输出流,它用于输出错误和诊断信息。stderr默认连接到控制台,输出到屏幕。所以向stderr输出的信息直接打印在屏幕上。)

使用perror函数需要包含错误头文件errno.h

栗子:

int main()
{FILE* pFile;pFile = fopen("unexist.txt", "r");if (pFile == NULL)//printf("Error opening file unexist.ent: %s\n", strerror(errno));perror("Error opening file:");elseprintf("打开文件成功\n");return 0;
}

运行结果:

在这里插入图片描述


🚩总结

这次阿森和你一起学习6个C语言中常用的基本字符操作函数,但阿森会慢慢和你一起学习。感谢你的收看,如果文章有错误,可以指出,我不胜感激,让我们一起学习交流,如果文章可以给你一个小小帮助,可以给博主点一个小小的赞😘

请添加图片描述

相关文章:

【C语言】超详解strncpystrncatstrncmpstrerrorperror的使⽤和模拟实现

&#x1f308;write in front :&#x1f50d;个人主页 &#xff1a; 啊森要自信的主页 ✏️真正相信奇迹的家伙&#xff0c;本身和奇迹一样了不起啊&#xff01; 欢迎大家关注&#x1f50d;点赞&#x1f44d;收藏⭐️留言&#x1f4dd;>希望看完我的文章对你有小小的帮助&am…...

【Spring Boot 】Spring Boot 常用配置总结

文章目录 前言1.多环境配置application.propertiesapplication.yaml 2.常用配置3.配置读取4.自定义配置 前言 在涉及项目开发时&#xff0c;通常我们会灵活地把一些配置项集中在一起&#xff0c;如果你的项目不是很大的情况下&#xff0c;那么通过配置文件集中不失为一个很好的…...

Day60力扣打卡

打卡记录 1682分了记录下&#xff0c;希望下回能突破1700捏&#x1f923;&#x1f923;。作为一个菜鸟&#x1f628;&#xff0c;知道自己不太行&#x1f62d;&#x1f44a;&#xff0c;从以前的周赛稳定1题到稳定2题&#x1f97a;&#xff0c;到现在的时有时无的3题&#x1f9…...

Axure的动态图使用以及说明

认识Axure动态图 Axure动态图是Axure中的一种功能&#xff0c;它允许用户在原型中添加动画效果和交互动作&#xff0c;使原型更加生动和具有真实的用户体验。用户可以通过添加动态图来展示页面过渡、按钮点击、下拉菜单等交互操作的效果。 这是&#xff1a;就是我们今天要叫的…...

力扣 | 437. 路径总和 III

437. 路径总和 III mport java.util.ArrayList; import java.util.List;/*** int的取值范围&#xff1a;* -2^31 ~ 2^31-1* <p>* -2147483648 ~ 2147483647&#xff08;约等于10的9次方&#xff09;* <p>* long long的取值范围&#xff1a;* -2^63 ~ (2^63-1&…...

如何部署自己的服务渲染页面为Pdf文档

前言 相信大家都觉得官方发布的文档生成模块https://docs.mendix.com/appstore/modules/document-generation/很有用&#xff0c;它能把Mendix页面像素级导出到Pdf文件中&#xff0c;这对于归档等业务非常有价值。但部署依赖公有云提供的渲染服务&#xff0c;而中国本土用户对…...

常用的调试方法(段错误产生原因)

C 语言中常用的调试技巧和 demo C语言中常用的调试方法 打印调试信息 GDB 调试器 编写单元测试 段错误产生原因 初学时两种常用的段错误调试方法 C 语言中常用的调试技巧和 demo 当程序员进行调试时&#xff0c;他们通常会使用一些调试语句或技巧来帮助他们理解代码的执行过程…...

[云原生] Docker 入门指南:镜像、容器、卷和网络解析

Docker 是一种流行的容器化平台&#xff0c;它以其强大的功能和易用性在软件开发和部署领域广受欢迎。本文将带领您逐步探索 Docker 中的四个核心概念&#xff1a;镜像、容器、卷和网络。通过了解这些概念的是什么、为什么以及如何使用&#xff0c;您将能够更好地理解和利用 Do…...

机器学习-聚类问题

前言 聚类算法又叫做”无监督分类“&#xff0c;目标是通过对无标记训练样本来揭示数据的内在性质及 规律&#xff0c;为进一步的数据分析提供基础。 Kmeans 作为聚类算法的典型代表&#xff0c;Kmeans可以说是最简单的聚类算法&#xff0c;没有之一&#xff0c;那她是怎么完…...

leetcode9.回文数java解法

力扣&#xff08;LeetCode&#xff09;官网 - 全球极客挚爱的技术成长平台 给你一个整数 x &#xff0c;如果 x 是一个回文整数&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 回文数是指正序&#xff08;从左向右&#xff09;和倒序&#xff08;从右向左&…...

图论专栏一《图的基础知识》

图论&#xff08;Graph Theory&#xff09;是数学的一个分支。它以图为研究对象。图论中的图是由若干给定的点及连接两点的线所构成的图形&#xff0c;这种图形通常用来描述某些实体之间的某种特定关系&#xff0c;用点代表实体&#xff0c;用连接两点的线表示两个实体间具有的…...

得帆云为玉柴打造CRM售后服务管理系统,实现服务全过程管理|基于得帆云低代码的CRM案例系列

广西玉柴机器股份有限公司 广西玉柴机器股份有限公司始建于1992年&#xff0c;是国内行业首家赴境外上市的中外合资企业&#xff0c;产品远销亚欧美非等180多个国家和地区。公司总部设在广西玉林市&#xff0c;下辖11家子公司&#xff0c;生产基地布局广西、江苏、安徽、山东等…...

智能优化算法应用:基于蝠鲼觅食算法3D无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于蝠鲼觅食算法3D无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于蝠鲼觅食算法3D无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.蝠鲼觅食算法4.实验参数设定5.算法结果6.…...

vue2 以及 vue3 自定义组件使用 v-model使用默认值以及自定义事件

vue2 以及 vue3 自定义组件使用 v-model使用默认值以及自定义事件 1. vue2 自定义组件的 v-model vue2官网&#xff0c;自定义组件官方解释&#xff1a;一个组件上的 v-model 默认会利用名为 value 的 prop 和名为 input 的事件上代码代码中使用了 element-ui 子组件 使用默…...

《PCL多线程加速处理》-滤波-统计滤波

《PCL多线程加速处理》-滤波-统计滤波 一、效果展示二、实现方式三、代码一、效果展示 提升速度随着点云越多效果越明显 二、实现方式 1、原始的统计滤波实现方式 #include <pcl/filters/statistical_outlier_removal.h>pcl::PointCloud<pcl::PointXYZ...

插入排序——直接插入排序和希尔排序(C语言实现)

文章目录 前言直接插入排序基本思想特性总结代码实现 希尔排序算法思想特性总结代码实现 前言 本博客插入排序动图和希尔排序视频参考大佬java技术爱好者&#xff0c;如有侵权&#xff0c;请联系删除。 直接插入排序 基本思想 直接插入排序是一种简单的插入排序法&#xff…...

【Linux系统化学习】进程地址空间 | 虚拟地址和物理地址的关系

个人主页点击直达&#xff1a;小白不是程序媛 Linux专栏&#xff1a;Linux系统化学习 代码仓库&#xff1a;Gitee 目录 虚拟地址和物理地址 页表 进程地址空间 进程地址空间存在的意义 虚拟地址和物理地址 我们在学习C/C的时候肯定都见过下面这张有关于内存分布的图片&a…...

Navicat 技术指引 | 适用于 GaussDB 分布式的模型功能

Navicat Premium&#xff08;16.3.3 Windows 版或以上&#xff09;正式支持 GaussDB 分布式数据库。GaussDB 分布式模式更适合对系统可用性和数据处理能力要求较高的场景。Navicat 工具不仅提供可视化数据查看和编辑功能&#xff0c;还提供强大的高阶功能&#xff08;如模型、结…...

四十五、Redis主从

目录 1、数据同步原理 &#xff08;1&#xff09;全量同步 &#xff08;2&#xff09;增量同步 &#xff08;3&#xff09;优化Redis主从集群 &#xff08;4&#xff09;什么时候执行全量同步 &#xff08;5&#xff09;什么时候执行增量同步 2、流程 1、数据同步原理 &…...

Spring源码学习一

IOC容器概述 ApplicationContext接口相当于负责bean的初始化、配置和组装的IoC容器. Spring为ApplicationContext提供了一些开箱即用的实现, 独立的应用可以使用 ClassPathXmlApplicationContext或者FileSystemXmlApplicationContext&#xff0c;web应用在web.xml配置监 听&am…...

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…...

19c补丁后oracle属主变化,导致不能识别磁盘组

补丁后服务器重启&#xff0c;数据库再次无法启动 ORA01017: invalid username/password; logon denied Oracle 19c 在打上 19.23 或以上补丁版本后&#xff0c;存在与用户组权限相关的问题。具体表现为&#xff0c;Oracle 实例的运行用户&#xff08;oracle&#xff09;和集…...

JavaSec-RCE

简介 RCE(Remote Code Execution)&#xff0c;可以分为:命令注入(Command Injection)、代码注入(Code Injection) 代码注入 1.漏洞场景&#xff1a;Groovy代码注入 Groovy是一种基于JVM的动态语言&#xff0c;语法简洁&#xff0c;支持闭包、动态类型和Java互操作性&#xff0c…...

零门槛NAS搭建:WinNAS如何让普通电脑秒变私有云?

一、核心优势&#xff1a;专为Windows用户设计的极简NAS WinNAS由深圳耘想存储科技开发&#xff0c;是一款收费低廉但功能全面的Windows NAS工具&#xff0c;主打“无学习成本部署” 。与其他NAS软件相比&#xff0c;其优势在于&#xff1a; 无需硬件改造&#xff1a;将任意W…...

stm32G473的flash模式是单bank还是双bank?

今天突然有人stm32G473的flash模式是单bank还是双bank&#xff1f;由于时间太久&#xff0c;我真忘记了。搜搜发现&#xff0c;还真有人和我一样。见下面的链接&#xff1a;https://shequ.stmicroelectronics.cn/forum.php?modviewthread&tid644563 根据STM32G4系列参考手…...

Java 语言特性(面试系列1)

一、面向对象编程 1. 封装&#xff08;Encapsulation&#xff09; 定义&#xff1a;将数据&#xff08;属性&#xff09;和操作数据的方法绑定在一起&#xff0c;通过访问控制符&#xff08;private、protected、public&#xff09;隐藏内部实现细节。示例&#xff1a; public …...

【WiFi帧结构】

文章目录 帧结构MAC头部管理帧 帧结构 Wi-Fi的帧分为三部分组成&#xff1a;MAC头部frame bodyFCS&#xff0c;其中MAC是固定格式的&#xff0c;frame body是可变长度。 MAC头部有frame control&#xff0c;duration&#xff0c;address1&#xff0c;address2&#xff0c;addre…...

Python:操作 Excel 折叠

💖亲爱的技术爱好者们,热烈欢迎来到 Kant2048 的博客!我是 Thomas Kant,很开心能在CSDN上与你们相遇~💖 本博客的精华专栏: 【自动化测试】 【测试经验】 【人工智能】 【Python】 Python 操作 Excel 系列 读取单元格数据按行写入设置行高和列宽自动调整行高和列宽水平…...

uni-app学习笔记二十二---使用vite.config.js全局导入常用依赖

在前面的练习中&#xff0c;每个页面需要使用ref&#xff0c;onShow等生命周期钩子函数时都需要像下面这样导入 import {onMounted, ref} from "vue" 如果不想每个页面都导入&#xff0c;需要使用node.js命令npm安装unplugin-auto-import npm install unplugin-au…...

关于iview组件中使用 table , 绑定序号分页后序号从1开始的解决方案

问题描述&#xff1a;iview使用table 中type: "index",分页之后 &#xff0c;索引还是从1开始&#xff0c;试过绑定后台返回数据的id, 这种方法可行&#xff0c;就是后台返回数据的每个页面id都不完全是按照从1开始的升序&#xff0c;因此百度了下&#xff0c;找到了…...