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

探索C语言结构体:编程中的利器与艺术

pFp8UCq.jpg

✨✨ 欢迎大家来到贝蒂大讲堂✨✨

🎈🎈养成好习惯,先赞后看哦~🎈🎈

所属专栏:C语言学习
贝蒂的主页:Betty‘s blog

1. 常量与变量

1. 什么是结构体

在C语言中本身就自带了一些数据类型,如:char,int,float,double等数据类型都被称为内置类型,但是在实际生活中我们发现这些基本的数据类型是不够用的,当描述一个复杂的对象时候,如一个学生,一本书时,这时仅靠内置类型就有点捉襟见肘了。所以C语言除了内置类型之外又有了自定义类型,今天我们要学习的便是自定义类型之一——结构体

2. 结构体基础

2.1 结构体声明

结构体定义由关键字 struct 和结构体名组成,结构体名可以根据需要自行定义。struct 语句定义了一个包含多个成员的新的数据类型,struct 语句的格式如下:

struct tag {
member-list
member-list
member-list …
} variable-list ;

  • tag是结构体标签,定义你需要的结构体名,如book,student等。
  • member-list 是标准的变量定义,比如 int i; 或者 float f;,也可以定义数组char s[20]。
  • variable-list 结构变量,定义在结构的末尾,最后一个分号之前,您可以指定一个或多个结构变量,也可以省略。

下列是常见的结构体定义方式:

(1) 普通结构体
struct student
//声明一个学生的结构体
{int age;//年龄char sex[8];//性别int weight;//体重char tele[20];//电话
}s;
(2) 嵌套结构体

结构体和函数一样可以嵌套使用,也就是说在一个结构体中可以使用另外一个结构体,也可以包含指向自己结构体类型的指针,而通常这种指针的应用是为了实现一些更高级的数据结构如链表等。

struct student
{int age;//年龄char sex[8];//性别int weight;//体重char tele[20];//电话
};
struct people
{int num;//序号struct student s;//学生
};
struct list
{int num;//序号struct list* next;//指向自己的结构体指针
};
  • 但是结构体中不能包含一个同类型的结构体变量,因为这样结构体大小无法确定
struct node
{int num;struct node s;//错误定义
};
(3) 匿名结构体

匿名结构体是不定义结构体名称,而直接定义其成员的一种方式。这种结构体只能使用一次。并且两个匿名结构体的成员如果都相同的话,这两个匿名结构体也是不同的。

struct//匿名结构体
{int num;char name[20];//.....
};
struct
{int a;char b;float c;
}x;
struct
{int a;char b;float c;
}*p;
p = &x;//两种结构体不同无法赋值
(4) typedef简化结构体

因为结构体名称在使用时前缀较长,这时我们就可以使用typedef来简化。

typedef struct student
{int age;//年龄char sex[8];//性别int weight;//体重char tele[20];//电话
}stu;//之后就可以用stu代替struct student
  • 不能使用简化结构体提前在结构体内部创建变量
typedef struct student
{int age;//年龄char sex[8];//性别int weight;//体重char tele[20];//电话stu* next;//不能提前使用
}stu;

2.2 结构体变量的创建与初始化

在学习完结构体声明之后,我们就可以使用结构体创建变量和初始化变量。

typedef struct student
{int age;//年龄char sex[8];//性别int weight;//体重char tele[20];//电话
}stu;
struct people
{int num;//序号struct student s;//学生
};
int main()
{struct student s = { 20,"nan",50,"1233455" };//创建变量并初始化//struct student s;//s= { 20,"nan",50,"1233455" };错误stu t = { 18,"nan",45,"123444" };struct people p = { 1,{20,"nan",50,"1233455"} };//嵌套结构体的初始化return 0;
}
  • 结构体只能在创建变量时候初始化,不能先初始化,然后赋值

2.3 访问结构体成员

(1) 直接访问

结构体成员的直接访问是通过点操作符(.)访问的。点操作符接受两个操作数。如下所⽰:

struct Point
{int x;int y;
};
int main()
{struct Point p = { 1,2 };printf("x: %d y: %d\n", p.x, p.y);return 0;
}

输出结果:

x:1 y:2

(2) 间接访问

除了通过(.)操作符直接访问,我们也可以通过结构体地址,利用(->)操作符间接访问。

#include <stdio.h>
struct Point
{int x;int y;
};
int main()
{struct Point p = { 3, 4 };struct Point* ptr = &p;//结构体指针ptr->x = 1;ptr->y = 2;printf("x = %d y = %d\n", ptr->x, ptr->y);return 0;
}

输出结果:

x=1 y=2

3. 结构体内存对齐

3.1 内存对齐规则

在熟悉了结构体的基本应用之后,下面我们要深入讨论的就是结构体大小,那么结构体大小又该如何计算呢?首先让我们看一下下面这段代码。

struct example
{int num;char a;float b;int p;
}s;
int main()
{size_t sz = sizeof(s);printf("大小为%zd\n", sz);return 0;
}

输出结果:

  • 如果直接计算int,float类型4个字节,char类型一个字节,应该为13字节,而结果为16,所以C语言并不是直接分配大小的。

C语言分配结构体内存时,遵循的是内存对齐规则,那什么是内存对齐规则呢?

内存对齐规则:

  1. 结构体的第⼀个成员对⻬到和结构体变量起始位置偏移量为0的地址处
  2. 其他成员变量要对⻬到某个数字(对⻬数)的整数倍的地址处。
  3. 对⻬数=编译器默认的⼀个对⻬数与该成员变量⼤⼩的较⼩值。(VS 中默认的值为 8 ,Linux中gcc没有默认对齐数,对⻬数就是成员⾃⾝的⼤⼩)
  4. 结构体总⼤⼩为最⼤对⻬数(结构体中每个成员变量都有⼀个对⻬数,所有对⻬数中最⼤的)的整数倍。
  5. 如果嵌套了结构体的情况,嵌套的结构体成员对⻬到⾃⼰的成员中最⼤对⻬数的整数倍处,结构体的整体⼤⼩就是所有最⼤对⻬数(含嵌套结构体中成员的对⻬数)的整数倍。

3.2 宏——offsetof

  1. 头文件:#include <stddef.h>
  2. 声明:offsetof(type, member-designator)
  • type – 这是一个 class 类型,其中,member-designator 是一个有效的成员指示器。
  • member-designator – 这是一个 class 类型的成员指示器。
  1. 作用:计算结构体成员相对于起始位置的偏移量
  2. 返回值:该宏返回类型为 size_t 的值,表示 type 中成员的偏移量

代码示例:

#include<stddef.h>
struct S1
{char c1;int i;char c2;
};
int main()
{printf("%zd\n", offsetof(struct S1, c1));printf("%zd\n", offsetof(struct S1, i));printf("%zd\n", offsetof(struct S1, c2));return 0;
}

输出结果:

示意图:

  1. c1是一个char类型占一个字节,根据内存对齐第一条规则,放在偏移量为0处
  2. i是int类型占四个字节,根据内存对齐第二,三规则,默认对齐数是4,对齐至4的整数倍,中间浪费三个字节,偏移量为4.
  3. c2也是char类型占一个字节,默认对齐数是1的整数倍,偏移量为8。

3.2 实例应用

(1) 实例一
struct S1
{char c1;char c2;int i;
};
int main()
{printf("大小为%zd\n", sizeof(struct S1));//输出什么?return 0;
}

输出结果:

大小为8

解析:

  1. c1放在偏移量为0的位置,大小为一个字节。
  2. c2的默认对齐数为1,放在1的整数倍处,大小为一个字节。
  3. i的默认对齐数为4放在4的整数倍处,浪费2个字节,大小为4个字节、
  4. 最大对齐数为4,结构体大小为4的整数倍,现在大小为8刚好满足,所以这个结构体大小为8.
(2) 实例二
struct S2
{double i;char c1;char c3;
};
int main()
{printf("大小为%zd\n", sizeof(struct S2));//输出什么?return 0;
}

输出结果:

大小为16

解析:

  1. i放在偏移量为0处,大小为8。
  2. c1放在默认对齐数为1放在1的整数倍处,大小为一个字节。
  3. c2的默认对齐数为1,放在1的整数倍处,大小为一个字节。
  4. 最大默认对齐数为8,结构体大小为8的整数倍,现在大小为10所以浪费6个字节,大小为16。
(3) 实例三
struct S2
{double i;char c1;char c3;
};
struct S3
{struct S2 s;int d;char c3;
};
int main()
{printf("大小为%zd\n", sizeof(struct S3));//输出什么?return 0;
}

输出结果:

大小为24

解析:

  1. 结构体S2的大小为16,放在偏移量为0的位置。
  2. d的默认对齐数为4,放在4的整数倍16的位置,大小为4。
  3. c3的默认对齐数为1,放在1的整数倍20的位置,大小为1。
  4. 最大对齐数为8,结构体大小为8的整数倍,大小为24。

4. 位段

4.1 什么是位段

有些数据在存储时并不需要占用一个完整的字节,只需要占用一个或几个二进制位即可,所以C 语言有一种特别的数据结构名为位,允许我们按位对成员进行定义,指定其占用的位数,单位为比特位(bit)。一般是用来节约内存,与结构体有两个不同:

  1. 位段的成员必须是 int、unsigned int 或signed int ,char等整型。到了 C99,_Bool 也被支持了。
  2. 位段的成员名后边有⼀个冒号和⼀个数字。

比如:

struct A
{int _a : 2;//为a分配2个比特位int _b : 3;//为b分配3个比特位int _c : 4;//为c分配4个比特位int _d : 5;//为d分配5个比特位
};
  • 位段的⼏个成员可能共有同⼀个字节,这样有些成员的起始位置并不是某个字节的起始位置,那么这些位置处是没有地址的。内存中每个字节分配⼀个地址,⼀个字节内部的bit位是没有地址的。所以不能对位段的成员使⽤&操作符,这样就不能使⽤scanf直接给位段的成员输⼊值,只能是先输⼊放在⼀个变量中,然后赋值给位段的成员。
struct A
{int _a : 2;int _b : 5;int _c : 10;int _d : 30;
};
int main()
{struct A sa = { 0 };scanf("%d", &sa._b); //这是错误的//正确的⽰范int b = 0;scanf("%d", &b);sa._b = b;return 0;
}

4.2 位段的内存分配

  1. 位段的空间上是按照需要以4个字节( int )或者1个字节( char )的⽅式来开辟的。
  2. 位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使⽤位段。

如下所示,下面这个位段会占用多少内存空间?

struct A
{int _a : 2;int _b : 5;int _c : 10;int _d : 30;
};
int main()
{printf("大小为%zd\n", sizeof(struct A));//输出什么?return 0;
}

  • 2+5+10+30=47,而八个字节64个比特位,事实证明位段节约空间的能力有限的。

那么位段的分配到底是怎么样的呢?

当一个结构体包含两个位段,第二个位段比较大,无法容纳于第一个位段剩余的位时, 是舍弃剩余的位还是利用呢?

我们以下列程序举例:

struct S
{char a:3;char b:4;char c:5;char d:4;
};
//内存如何分配?
int main()
{struct S s = {0};s.a = 10;s.b = 12;s.c = 3;s.d = 4;return 0;
}
  1. 假设位段在一个字节内部是从高地址到低地址分配。
  2. 假设当一个结构体包含两个位段,第二个位段比较大,无法容纳于第一个位段剩余的位时, 是舍弃。

图示:

  1. 12先转换为二进制01010,因为假设位段在一个字节内部是从高地址到低地址分配,所以从右往左分配,又因为只给a分配了三个比特位,所以只存进010。
  2. 然后12转换为二进制01100,因为只分配给b四个比特位,所以只存进1100。
  3. 3转换为二进制为00011,因为分配了c五个比特位,所以存进00011。
  4. 最后4转换为二进制00100,分配进四个比特位0100。
  • 最后只需验证系统存在的数据是否为这样的数据就能验证我们的猜测是否正确了·。

  • 通过验证我们可以确定的是在VS2022环境下我们的猜想是正确的。

4.3 位段的缺陷

位段虽然能帮助我们节约内存,但是也有许多缺陷,尤其在跨平台问题上有很大的缺陷。

  1. int型位段成员会被当成有符号数还是无符号数是不确定的
  2. 位段中最大位数目是不确定的(在16位机器上int型最大为16,而在32为机器上int型最大为32,如若写成27,那么16位机器就会出问题)
  3. 位段的成员在内存中到底是从左向右分配,还是从右向左分配尚未定义
  4. 当一个结构体包含两个位段,第二个位段比较大,无法容纳于第一个位段剩余的位时, 是舍弃剩余的位还是利用,是不确定的。

5. 结构体传参

我们知道函数传参分为两种,一种是直接传参:直接传变量;一种是间接传参:通过传变量地址间接访问。

struct S1
{int p;int num;
};//结构体传参
void print1(struct S1 s)
{printf("%d\n", s.num);
}//结构体地址传参
void print2(struct S1* ps)
{printf("%d\n", ps->num);
}int main()
{struct S1 s = { 1,2 };print1(s);		//传整个结构体print2(&s);		//传地址return 0;
}

那么到底哪种方法好呢?答案自然是传地址更好,因为我们知道传参时形参是实参的临时拷贝,也就是说系统会将参数复制一份,当一个参数的数量极大时会造成不必要的内存分配,而传址调用系统只用分配四个字节或八个字节,大大节约内存。

相关文章:

探索C语言结构体:编程中的利器与艺术

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;C语言学习 贝蒂的主页&#xff1a;Betty‘s blog 1. 常量与变量 1. 什么是结构体 在C语言中本身就自带了一些数据类型&#x…...

Git介绍与常用命令总结

Git介绍与其常用命令总结 1、Git介绍2、Git的使用3、Git常用命令3.1 初始化仓库3.2 克隆仓库3.3 配置用户信息3.4 提交代码(Commit)3.5 推送代码(Push)3.6 拉取代码(Pull)3.7 分支(Branch)3.8 远程仓库(Remote)3.9 撤销回退本地改动3.10 更新本地仓库与远程仓库 1、Git介绍 Gi…...

机器学习 | 探索朴素贝叶斯算法的应用

朴素贝叶斯算法是一种基于贝叶斯定理和特征条件独立假设的分类算法。它被广泛应用于文本分类、垃圾邮件过滤、情感分析等领域&#xff0c;并且在实际应用中表现出色。 朴素贝叶斯法是基于贝叶斯定理与特征条件独立假设的分类方法&#xff1a; 1&#xff09;对于给定的待分类项r…...

【无刷电机学习】电流采样电路硬件方案

【仅作自学记录&#xff0c;不出于任何商业目的】 目录 AD8210 INA282 INA240 INA199 AD8210 【AD8210数据手册】 在典型应用中&#xff0c;AD8210放大由负载电流通过分流电阻产生的小差分输入电压。AD8210抑制高共模电压(高达65V)&#xff0c;并提供接地参考缓冲输出&…...

对于协同过滤算法我自己的一些总结和看法

文章目录 协同过滤算法的基本原理协同过滤算法的分类用户相似度计算UserCF && ItemCF应用场景 协同过滤算法的优缺点优点缺点 协同过滤算法的总结与展望Q&A 协同过滤算法的基本原理 关于协同过滤算法&#xff0c;我看过很多老师写的博客以及一些简单的教程&#x…...

数据库管理phpmyadmin

子任务1-PHPmyadmin软件的使用 本子任务讲解phpmyadmin的介绍和使用操作。 训练目标 1、掌握PHPmyadmin软件的使用方法。 步骤1 phpMyAdmin 介绍 phpmyadmin是一个用PHP编写的软件工具&#xff0c;可以通过web方式控制和操作MySQL数据库。通过phpMyAdmin可以完全对数据库进行…...

Oracle数据表ID自增操作

一、Oracle ID自增长功能介绍 Oracle数据库默认不支持像 SQLServer、MySQL中的自增长&#xff08;auto increment&#xff09;功能&#xff0c;即自动为每一行记录的自增长字段生成下一个值。 二、Oracle ID自增长方法 第一种&#xff0c;通过序列&#xff08;sequence&#…...

npm WARN deprecated uuid@3.4.0: Please upgrade to version 7 or higher

当使用npm下载vue3-lazy时出现一下错误时的解决方案 报错&#xff1a;npm WARN deprecated uuid3.4.0: Please upgrade to version 7 or higher 尝试使用过一下命令更新 npm install uuidlatest -g 是安装了最新版本的uuid&#xff0c; 再次下载已解决问题 ***但看某些播客依…...

第2节、让电机转起来【51单片机+L298N步进电机系列教程】

↑↑↑点击上方【目录】&#xff0c;查看本系列全部文章 摘要&#xff1a;本节介绍用简单的方式&#xff0c;让步进电机转起来。其目的之一是对电机转动有直观的感受&#xff0c;二是熟悉整个开发流程。本系列教程必要的51单片机基础包括IO口操作、中断、定时器三个部分&#…...

1154: 第多少天

题目描述 定义一个包括年、月、日的结构体变量&#xff0c;读入年、月、日&#xff0c;计算该日在当年中是第几天。注意闰年问题。 输入描述 三个整数&#xff0c;分别表示年、月、日。保证输入是实际存在的日期&#xff0c;且年份在1000至3000之间&#xff08;包含1000和30…...

【C语言初阶-const作用详解】const修饰变量、const修饰指针(图文详解版)

少年&#xff0c;做你认为对的事 目录 少年&#xff0c;做你认为对的事 1.const修饰变量 2.const修饰指针&#xff08;重要&#xff09; 代码1&#xff1a; 代码2&#xff1a; 代码3&#xff1a; ​编辑 3.结论 1.const修饰变量 const修饰变量将变量赋予了常量属性…...

线程协作工具类【CountDownLatch倒数门闩、Semaphore信号量、CyclicBarrier循环栏栅、Condition接口】

线程协作工具类 CountDownLatch倒数门闩Semaphore信号量CyclicBarrier循环栅栏CyclicBarrier和CountDownLatch区别: Condition接口(条件对象) 转自 极客时间 线程协作工具类就是帮助程序员更容易的让线程之间进行协作&#xff0c;来完成某个业务功能。 CountDownLatch倒数门闩…...

Python 函数式编程进阶:map、filter、reduce

Python 函数式编程进阶&#xff1a;map、filter、reduce 介绍map 函数作用和语法使用 map 函数Lambda 函数的配合应用 filter 函数作用和语法使用 filter 函数Lambda 函数的结合运用 reduce 函数作用和语法使用 reduce 函数典型应用场景 介绍 在函数式编程中&#xff0c;map、…...

大模型|基础_word2vec

文章目录 Word2Vec词袋模型CBOW Continuous Bag-of-WordsContinuous Skip-Gram存在的问题解决方案 其他技巧 Word2Vec 将词转化为向量后&#xff0c;会发现king和queen的差别与man和woman的差别是类似的&#xff0c;而在几何空间上&#xff0c;这样的差别将会以平行的关系进行表…...

14.2 url后端过滤器(❤❤)

14.2 过滤器 1. 过滤器Filter1.1 配置形式实现过滤器1.2 过滤器生命周期1.3 过滤器特性(面试点)1.4 注解形式实现过滤器1.5 两种实现的选择2. 应用2.1 字符集过滤:统一设置请求与响应字节编码1. 配置方式实现过滤器参数化:init-param标签关键代码完整代码2. 注解方式实现2.2 多…...

Leetcode 377 组合总和 Ⅳ

题意理解&#xff1a; 给你一个由 不同 整数组成的数组 nums &#xff0c;和一个目标整数 target 。请你从 nums 中找出并返回总和为 target 的元素组合的个数。 题目数据保证答案符合 32 位整数范围。 这道题目和凑零钱是一样的&#xff0c;需要求使用指定元素&#xff08;纸币…...

CleanMyMacX4.14.6如何清理mac垃圾内存

一直以来&#xff0c;苹果电脑的运行流畅度都很好&#xff0c;但是垃圾内存多了磁盘空间慢慢变少&#xff0c;还是会造成卡顿的。这篇文章就告诉大家电脑如何清理垃圾内存&#xff0c;电脑如何清理磁盘空间。 一、电脑如何清理垃圾内存 垃圾内存指的是各种缓存文件和系统垃圾…...

Java 学习和实践笔记(1)

2024年&#xff0c;决定好好学习计算机语言Java. B站上选了这个课程&#xff1a;【整整300集】浙大大佬160小时讲完的Java教程&#xff08;学习路线Java笔记&#xff09;零基础&#xff0c;就从今天开始学吧。 在这些语言中&#xff0c;C语言是最基础的语言&#xff0c;绝大多…...

【自然语言处理-工具篇】spaCy<1>--介绍及安装指南

目录 前言 安装指南 pip conda spaCy升级 总结 前言 spaCy是一个开源的自然语言处理库,用于处理和分析文本数据。它提供了许多功能,包括分词、词性标注...

LeetCode树总结

​​​​​​144. 二叉树的前序遍历 递归写法很简单&#xff0c;不再赘述。迭代写法需要用到一个栈&#xff0c;因为是根->左子树->右子树的顺序进行遍历&#xff0c;所以弹出当前结点后要先入栈右儿子&#xff0c;再入栈左儿子。 /*** Definition for a binary tree n…...

测试微信模版消息推送

进入“开发接口管理”--“公众平台测试账号”&#xff0c;无需申请公众账号、可在测试账号中体验并测试微信公众平台所有高级接口。 获取access_token: 自定义模版消息&#xff1a; 关注测试号&#xff1a;扫二维码关注测试号。 发送模版消息&#xff1a; import requests da…...

生成xcframework

打包 XCFramework 的方法 XCFramework 是苹果推出的一种多平台二进制分发格式&#xff0c;可以包含多个架构和平台的代码。打包 XCFramework 通常用于分发库或框架。 使用 Xcode 命令行工具打包 通过 xcodebuild 命令可以打包 XCFramework。确保项目已经配置好需要支持的平台…...

装饰模式(Decorator Pattern)重构java邮件发奖系统实战

前言 现在我们有个如下的需求&#xff0c;设计一个邮件发奖的小系统&#xff0c; 需求 1.数据验证 → 2. 敏感信息加密 → 3. 日志记录 → 4. 实际发送邮件 装饰器模式&#xff08;Decorator Pattern&#xff09;允许向一个现有的对象添加新的功能&#xff0c;同时又不改变其…...

SCAU期末笔记 - 数据分析与数据挖掘题库解析

这门怎么题库答案不全啊日 来简单学一下子来 一、选择题&#xff08;可多选&#xff09; 将原始数据进行集成、变换、维度规约、数值规约是在以下哪个步骤的任务?(C) A. 频繁模式挖掘 B.分类和预测 C.数据预处理 D.数据流挖掘 A. 频繁模式挖掘&#xff1a;专注于发现数据中…...

MySQL 8.0 OCP 英文题库解析(十三)

Oracle 为庆祝 MySQL 30 周年&#xff0c;截止到 2025.07.31 之前。所有人均可以免费考取原价245美元的MySQL OCP 认证。 从今天开始&#xff0c;将英文题库免费公布出来&#xff0c;并进行解析&#xff0c;帮助大家在一个月之内轻松通过OCP认证。 本期公布试题111~120 试题1…...

MySQL用户和授权

开放MySQL白名单 可以通过iptables-save命令确认对应客户端ip是否可以访问MySQL服务&#xff1a; test: # iptables-save | grep 3306 -A mp_srv_whitelist -s 172.16.14.102/32 -p tcp -m tcp --dport 3306 -j ACCEPT -A mp_srv_whitelist -s 172.16.4.16/32 -p tcp -m tcp -…...

DeepSeek 技术赋能无人农场协同作业:用 AI 重构农田管理 “神经网”

目录 一、引言二、DeepSeek 技术大揭秘2.1 核心架构解析2.2 关键技术剖析 三、智能农业无人农场协同作业现状3.1 发展现状概述3.2 协同作业模式介绍 四、DeepSeek 的 “农场奇妙游”4.1 数据处理与分析4.2 作物生长监测与预测4.3 病虫害防治4.4 农机协同作业调度 五、实际案例大…...

AGain DB和倍数增益的关系

我在设置一款索尼CMOS芯片时&#xff0c;Again增益0db变化为6DB&#xff0c;画面的变化只有2倍DN的增益&#xff0c;比如10变为20。 这与dB和线性增益的关系以及传感器处理流程有关。以下是具体原因分析&#xff1a; 1. dB与线性增益的换算关系 6dB对应的理论线性增益应为&…...

力扣热题100 k个一组反转链表题解

题目: 代码: func reverseKGroup(head *ListNode, k int) *ListNode {cur : headfor i : 0; i < k; i {if cur nil {return head}cur cur.Next}newHead : reverse(head, cur)head.Next reverseKGroup(cur, k)return newHead }func reverse(start, end *ListNode) *ListN…...

接口自动化测试:HttpRunner基础

相关文档 HttpRunner V3.x中文文档 HttpRunner 用户指南 使用HttpRunner 3.x实现接口自动化测试 HttpRunner介绍 HttpRunner 是一个开源的 API 测试工具&#xff0c;支持 HTTP(S)/HTTP2/WebSocket/RPC 等网络协议&#xff0c;涵盖接口测试、性能测试、数字体验监测等测试类型…...