由浅到深认识C语言(11):结构体

该文章Github地址:https://github.com/AntonyCheng/c-notes

在此介绍一下作者开源的SpringBoot项目初始化模板(Github仓库地址:https://github.com/AntonyCheng/spring-boot-init-template & CSDN文章地址:https://blog.csdn.net/AntonyCheng/article/details/136555245),该模板集成了最常见的开发组件,同时基于修改配置文件实现组件的装载,除了这些,模板中还有非常丰富的整合示例,同时单体架构也非常适合SpringBoot框架入门,如果觉得有意义或者有帮助,欢迎Star & Issues & PR!

上一章:由浅到深认识C语言(10):字符串处理函数

11.结构体

概念以及定义:

C 语言允许用户自己指定这样一种数据结构,它由不同类型的数据组合成一个整体,以便引用,这些组合在一个整体中的数据是互相联系的,这样的数据结构称为结构体,它相当于其它高级语言中记录;

11.1.结构体定义形式

**形式一:**先定义结构体类型,再定义变量(推荐);

//定义结构体类型
struct 结构体类型{成员一;成员二;……
};
//定义变量
struct 结构体类型 变量名;

示例如下:

struct stu{int id;char name[5];int age;
};
struct stu s1;

此时结构体的数据类型就是定义时中括号前的内容,即 struct stu

**形式二:**定义类型的同时定义变量;

//定义类型,同时定义变量
struct 结构体类型{成员一;成员二;……
}变量名;

示例如下:

struct stu{int id;char name[5];int age;
}s1;

**形式三:**定义一次性结构体;

//定义类型,但不带结构体类型,同时定义变量
struct {成员一;成员二;……
}变量名;

示例如下:

struct{int id;char name[5];int age;
}s1;

不能用一次性结构体定义其他变量,例如 s2 、s3 ……

注意事项

  • 定义结构体类型的时候,不要给成员变量赋值,因为定义结构体类型的时候并没有分配空间,所以结构体类型不占空间,结构体变量才占空间
  • 结构体内的成员拥有独立的空间;
  • 定义结构体类型是一条语句,所以要记住结构体后的那一个 ;

实例展示

#include<stdio.h>struct stu
{int id;char name[32];int age;
};void test() {printf("sizeof(struct stu) = %d\n", sizeof(struct stu));return;
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

11.2.结构体使用法则

格式:类似于Java中类的思想;

结构体变量.结构体成员

访问结构体成员时是通过结构体变量来访问的,而且一定要遵循成员自身的类型;

重点又是字符串在结构体中的操作,字符串除了初始化能够直接赋值之外,其他情况的赋值必须想其他方法,而在结构体中是不能在结构体中成员进行初始化的,所以我们就要将字符串拷贝进去;

示例如下:

#include<stdio.h>
#include<string.h>struct stu
{int id;char name[32];int age;
};void test() {struct stu xiaochen;xiaochen.id = 9;strcpy_s(xiaochen.name, 32, "晓晨");xiaochen.age = 21;printf("xiaochen.id = %d\n", xiaochen.id);printf("xiaochen.name = %s\n", xiaochen.name);printf("xiaochen,age = %d\n", xiaochen.age);return;
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

11.3.结构体的操作

初始化结构体

定义结构体类型的时候我们是不能将其初始化的,但是在定义变量时可以对其进行初始化;

初始化示例如下:

struct stu{int id;char name[32];char sex;
};
struct stu s = {2020510,"xiaochen",'男'};

清空结构体

格式如下:

struct stu{int id;char name[32];char sex;
};
struct stu s = {2020510,"xiaochen",'男'};
memset(&s, 0, sizeof(s));

11.4.结构体变量操作

获取键盘输入

示例如下:

#include<stdio.h>
#include<string.h>struct stu
{int id;char name[32];int age;
};void test() {struct stu xiaochen;memset(&xiaochen, 0, sizeof(xiaochen));printf("从键盘输入xiaochen的基本情况:");scanf_s("%d %s %d", &xiaochen.id, xiaochen.name, 32, &xiaochen.age);printf("xiaochen.id = %d\n", xiaochen.id);printf("xiaochen.name = %s\n", xiaochen.name);printf("xiaochen.age = %d\n", xiaochen.age);
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

变量之间的赋值

例如此时有两个结构体变量,我们想将其中一个的值复制给另外一个,此时就要使用到变量之间的赋值运算;

**方式一:**逐个成员更换;

#include<stdio.h>
#include<string.h>struct stu
{int id;char name[32];int age;
};void test() {struct stu s1 = { 2022,"小红",21 };struct stu s2;s2.age = s1.age;s2.id = s1.id;strcpy_s(s2.name, 32, s1.name);printf("s2.age = %d , s2.id = %d , s2.name = %s\n", s2.age, s2.id, s2.name);
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

**方式二:**相同类型的结构体变量可以整体赋值;(推荐)

#include<stdio.h>
#include<string.h>struct stu
{int id;char name[32];int age;
};void test() {struct stu s1 = { 2022,"小红",21 };struct stu s2;s2 = s1;printf("s2.age = %d , s2.id = %d , s2.name = %s\n", s2.age, s2.id, s2.name);
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

**方式三:**使用内存拷贝函数 memcpy() 从原理出发实现赋值;

#include<stdio.h>
#include<string.h>struct stu
{int id;char name[32];int age;
};void test() {struct stu s1 = { 2022,"小红",21 };struct stu s2;memcpy(&s2, &s1, sizeof(s1));printf("s2.age = %d , s2.id = %d , s2.name = %s\n", s2.age, s2.id, s2.name);
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

11.5.结构体数组操作

结构体数组本质就是装有结构体的数组容器;

图解:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

当我们要去提取 arr[2] 中的 name ,即用 arr[2].name

示例如下:

#include<stdio.h>
#include<string.h>struct stu
{int id;char name[32];int age;
};void test() {struct stu arr[5] = {{100,"xiaofa",18},{101,"xiaohong",19},{102,"xiaowang",18},{103,"xiaochen",19},{104,"xiaowu",20}};for (int i = 0; i < 5; i++) {printf("arr[%d].id = %d , arr[%d].name = %s , arr[%d].age = %d\n",i,arr[i].id,i,arr[i].name,i,arr[i].age);}
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

结构体数组从键盘输入:

#include<stdio.h>
#include<string.h>struct stu
{int id;char name[32];int age;
};void test() {struct stu arr[5];int n = sizeof(arr) / sizeof(arr[0]);memset(arr, 0, sizeof(arr));printf("请一次输入%d个学生的学号,名字以及年龄:\n", n);for (int i = 0; i < n; i++) {printf("请输入第%d个学生的信息:\n",i+1);scanf_s("%d %s %d", &arr[i].id, arr[i].name, 32, &arr[i].age);}for (int i = 0; i < n; i++) {printf("arr[%d].id = %d , arr[%d].name = %s , arr[%d].age = %d\n",i,arr[i].id,i,arr[i].name,i,arr[i].age);}
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

11.6.结构体指针操作

顾名思义,结构体指针本质是指针,该指针指向结构体的地址,即数据类型是一个 结构体*

写法类比:

//定义结构体:
struct struct_name{int id;};
//定义结构体变量:
struct struct_name a;
//定义结构体指针变量:
struct struct_name* p;
//结构体变量调用结构体元素:
a.id;
//结构体指针变量调用结构体元素:
a->id;  //也可以写作(*a).id;

图解如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

示例一:结构体指针;

#include<stdio.h>
#include<string.h>struct stu
{int id;char name[32];int age;
}; void test() {struct stu xiaochen = { 100,"晓晨",18 };printf("id = %d,name = %s,age = %d\n", xiaochen.id, xiaochen.name, xiaochen.age);struct stu* p = &xiaochen;p->age = 20;p->id = 101;strcpy_s(p->name, 32, "安东尼");printf("id = %d,name = %s,age = %d\n", xiaochen.id, xiaochen.name, xiaochen.age);
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

案例二:从堆区为结构体申请一个空间;

#include<stdio.h>
#include<stdlib.h>
#include<string.h>struct stu
{int id;char name[32];int age;
}; void test() {struct stu* p = NULL;int n = 0;printf("请输入人数个数:");scanf_s("%d", &n);//从堆区申请结构体空间p = (struct stu*)calloc(n, sizeof(struct stu));if (p == NULL) {perror("err");return;}//获取键盘输入for (int i = 0; i < n; i++) {printf("请输入第%d个学生信息:", i + 1);scanf_s("%d %s %d", &(p+i)->id, (p+i)->name,32, &(p+i)->age);}//遍历键盘输入for (int i = 0; i < n; i++) {printf("p.id = %d  p.name = %s  p.age = %d\n", (p+i)->id, (p+i)->name, (p+i)->age);}//释放空间if (p != NULL) {free(p);p = NULL;}
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

示例三:从堆区为结构体数组申请一个空间,并且分函数实现(空间复杂度尽可能地小);

#include<stdio.h>
#include<stdlib.h>
#include<string.h>struct stu
{int id;char name[32];int age;
};struct stu* get_mem(int n) {return (struct stu*)calloc(n, sizeof(struct stu));
}struct stu* set_data(struct stu* p, int* n) {for (int i = 0; i < *n; i++) {printf("请输入第%d个同学信息:", i + 1);scanf_s("%d %s %d", &(p + i)->id, (p + i)->name, 32, &(p + i)->age);}return p;
}void print_data(const struct stu* p, int* n) {//constfor (int i = 0; i < *n; i++) {printf("student%d.id = %d  student%d.name = %s student%d.age = %d\n", i + 1, (p + i)->id, i + 1, (p + i)->name, i + 1, (p + i)->age);}return;
}void test() {printf("请输入学生的个数:\n");int n;scanf_s("%d", &n);struct stu* arr = NULL;//根据学生个数申请空间arr = get_mem(n);if (arr == NULL) {perror("err");return;}//为申请的空间赋值arr = set_data(arr, &n);//输出空间内容print_data(arr, &n);//释放空间if (arr != NULL) {free(arr);arr = NULL;}
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

结构体指针作为函数的参数

**示例一:**定义一个函数,给结构体成员获取键盘输入;

#include<stdio.h>
#include<stdlib.h>
#include<string.h>struct stu
{int id;char name[32];int age;
}; void method(struct stu* student) {printf("请输入学生的信息(学号 名字 年龄):");scanf_s("%d %s %d", &student->id, student->name, 32, &student->age);printf("xiaochen.id = %d  xiaochen.name = %s  xiaochen.age = %d\n", student->id, student->name, student->age);
}void test() {struct stu xiaochen;method(&xiaochen);
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

11.7.typedef取别名

使用步骤:

  • 先用已有的类型定义一个相对应的变量
  • 用别名替换变量名
  • 在整个表达式前添加 typedef

注意:typedef 不能创造新的类型,只是给现有类型取别名;

给 int 类型取别名:

#include<stdio.h>typedef int INT;void test() {INT num = 10;printf("INT num = %d\n", num);
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

给数组取别名:

#include<stdio.h>typedef int INT[5];void test() {INT num = {1,2,3,4,5};  //此时 arr 就是一个拥有5个元素,每个元素为 int 型的数组,arr 是数组首地址for (int i = 0; i < 5; i++) {printf("%d ", *(num + i));}printf("\n");
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

给指针取别名:

#include<stdio.h>typedef int* INT;void test() {INT num = NULL;int a = 10;num = &a;printf("*num = %d\n", *num);
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

给函数指针取别名:

#include<stdio.h>int sum(int a, int b) {return a + b;
}typedef int (*SUM)(int,int) ;
//SUM 是一个函数指针类型,该函数必须有两个 int 形参以及一个 int返回值void test() {SUM p = sum;printf("*num = %d\n", p(100,200));
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

给结构体类型取别名:

#include<stdio.h>
#include<string.h>typedef struct stu
{int id;char name[32];int age;
}STU; //此时的STU不是结构体变量,而是结构体类型void test() {STU xiaochen = { 0 };xiaochen.age = 19;xiaochen.id = 101;strcpy_s(xiaochen.name, 32, "晓晨");printf("%s %d %d\n", xiaochen.name, xiaochen.id, xiaochen.age);
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

11.8.结构体内存对齐

引入实例:

#include<stdio.h>
struct data1
{char ch1;  //1Bchar ch2;  //1Bchar ch3;  //1Bshort num; //2B
};
struct data2
{char ch1;  //1Bchar ch2;  //1Bchar ch3;  //1Bint num;   //4B
};
struct data3
{char ch1;  //1Bchar ch2;  //1Bchar ch3;  //1Bdouble num;  //8B
};void test() {struct data1 d1;printf("sizeof(d1) = %d\n", sizeof(d1));struct data2 d2;printf("sizeof(d2) = %d\n", sizeof(d2));struct data3 d3;printf("sizeof(d3) = %d\n", sizeof(d3));
}int main(int argc, char* argv[]) {test();return;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

从上面例子所打印结果我们可以得到一个观察结论:

结构体的大小由占内存最大的那个基本数据类型所决定,即结构体的单位内存大小就是内存最大的基本数据类型的内存大小,这就是结构体的内存向大对齐;

结构体内存分布

示例如下:

struct data{char c;  //1Bint i;  //4B
};

方法如下:

  1. 确定分配单位:每一行应该分配的字节数;

    由结构体中最大的基本类型长度决定;

  2. 确定成员的起始位置的偏移量 == 成员的基本类型含零整数倍

  3. 收尾工作:结构体总大小 == 分配单位的整数倍;

案例一:

typedef struct {int a;char b;short c;char d;
}DATA;

画图如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

证实如下:

#include<stdio.h>typedef struct {int a;char b;short c;char d;
}DATA;void test() {DATA data = { 0,0,0,0 };printf("%u\n", &data.a);printf("%u\n", &data.b);printf("%u\n", &data.c);printf("%u\n", &data.d);
}int main(int argc, char* argv[]) {test();return;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

案例二:

typedef struct {char a;short b;short c;int d;char e;
}DATA;

画图如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

案例三:

typedef struct {char a;short b;char c;short d;
}DATA;

画图如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

结构体内存分布的作用

便于指针作用于结构体中;

示例如下:

一个结构体中有如下成员:

typedef struct {char a;int b;short c;
}DATA;

我们要通过指针去取到 short c

#include<stdio.h>typedef struct {char a;int b;short c;
}DATA;void test() {DATA data = { 'a',10,20 };char* p = NULL;p = &data.a;p = p + 8;printf("data.c = %d\n", *(short*)p);
}int main(int argc, char* argv[]) {test();return;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

11.9.结构体嵌套结构体

图解如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

示例如下:

#include<stdio.h>typedef struct {int a;int b;
}DATA1;typedef struct {int c;int d;DATA1 e;  //结构体变量c作为DATA2的成员,这就是结构体嵌套结构体
}DATA2;void test() {DATA2 data = {0,0,{0,0}};printf("请输入四个数字:\n");scanf_s("%d %d %d %d", &data.e.a, &data.e.b, &data.c, &data.d);printf("你输入的四个数是:\n");printf("data.e.a = %d\ndata.e.b = %d\ndata.c = %d\ndata.d = %d\n", data.e.a, data.e.b, data.c, data.d);
}int main(int argc, char* argv[]) {test();return;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

嵌套结构内存分布

该内存分布与单层结构体的内存分布方式大同小异;

方法如下:

  1. 确定分配单位:每一行应该分配的字节数;

    由所有结构体中最大的基本类型长度决定的;

  2. 确定成员的偏移量 == 自身类型的含零整数倍;

    结构体成员偏移量 == 被嵌套的结构体中最大的基本类型含零整数倍;

    结构体成员中的成员偏移量是相对于被嵌套的结构体而言的;

  3. 收尾工作:结构体的总大小 == 分配单位的整数倍;

    结构体成员的总大小 == 被嵌套的结构体里面最大基本类型整数倍;

案例一:

typedef struct {short d;char e;
}DATA2;typedef struct {short a;int b;DATA2 c;short f;
}DATA1;

画图如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

证实如下:

#include<stdio.h>typedef struct {short d;char e;
}DATA2;typedef struct {short a;int b;DATA2 c;short f;
}DATA1;void test() {DATA1 data = { 0,0,{0,0},0 };printf("%u\n", &data.a);printf("%u\n", &data.b);printf("%u\n", &data.c.d);printf("%u\n", &data.c.e);printf("%u\n", &data.f);
}int main(int argc, char* argv[]) {test();return;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

案例二:

typedef struct {short d;char e;
}DATA2;typedef struct {int a;short b;DATA2 c;short f;
}DATA;

画图如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

11.10.强制类型对齐

为了控制C语言中数据类型对齐不统一的问题,C语言中自带了一条处理语言:

#pragma pack (value);
//value只能是2^n,常用的几个是1,2,4,8;

这样声明就能够将该文件中的数据类型强制对齐于某一个值,但是要注意以下事项

指定对齐方式(value)与数据类型对齐默认值相比取较小值;

**解释说明:**如果我们指定的是 4,但是系统默认对齐值是 2 ,那么我们所设置的对齐方式无效,系统会默认进行比较,并且选择较小的值,如果我们指定的是 2,但是系统默认对齐值是 4,那么我们所设置的对齐方式就会生效;

  • 例一:指定值是 1,则 short、int、float 等均为 1;
  • 例二:指定值是 2,则 char 仍然为 1,short 为 2,int 也为 2;

下面是具体步骤:

  1. 确定分配单位:每一行应该分配的字节数 ==> min(value,默认分配单位);
  2. 成员偏移量 == 成员自身类型的含零整数倍;
  3. 收尾工作:总大小就是分配单位的整数倍;

那么我们可以通过强制类型对齐来计算一个结构体的真实大小,即不含有空地值的大小:

#include<stdio.h>
#pragma pack (1)
typedef struct {char a;    //1Bint b;     //4Bshort c;   //2B
}DATA;void test() {DATA data = { 'a',10,20 };printf("the real size of data is %d Bite\n", sizeof(data));
}int main(int argc, char* argv[]) {test();return 0;
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

无论是系统默认对齐,还是强制类型转换都有可能会改变一个结构体的内存大小,除此之外,调整结构体的成员顺序也有可能改变一个结构体的大小,但是这是一个时间复杂度与空间复杂度的相互牵制问题,如果想让空间复杂度小一点,那么就尽量把数据类型大小相同的或者相近的成员在编写代码阶段编在一起;

下一章:由浅到深认识C语言(12):位段/位域

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

前端React篇之React的生命周期有哪些?

目录 React的生命周期有哪些&#xff1f;挂载阶段&#xff08;Mounting&#xff09;更新阶段&#xff08;Updating&#xff09;卸载阶段&#xff08;Unmounting&#xff09;错误处理阶段&#xff08;Error Handling&#xff09; React常见的生命周期React主要生命周期 React的生…

神经网络中激活函数的绘制——阶跃函数、sigmoid函数、ReLU函数

一、阶跃函数 import numpy as np import matplotlib.pylab as plt def step_function(x):return np.array(x>0)x np.arange(-5.0,5.0,0.1) y step_function(x) plt.plot(x, y) plt.ylim(-0.1, 1.1) plt.show() 二、sigmoid函数 import numpy as np import matplotlib.p…

如果要做优化,CSS提高性能的方法有哪些?

文章目录 一、前言二、实现方式内联首屏关键CSS异步加载CSS资源压缩合理使用选择器减少使用昂贵的属性不要使用import其他 三、总结参考文献 一、前言 每一个网页都离不开css&#xff0c;但是很多人又认为&#xff0c;css主要是用来完成页面布局的&#xff0c;像一些细节或者优…

智慧公厕对于智慧城市管理的意义

近年来&#xff0c;智慧城市的概念不断被提及&#xff0c;而智慧公厕作为智慧城市管理的重要组成部分&#xff0c;其在监测、管理和养护方面发挥着重要的作用。智慧公厕不仅是城市市容提升的重要保障&#xff0c;还能提升城市环境卫生管理的质量&#xff0c;并有效助力创造清洁…

【代码随想录】【回溯算法】补day24:组合问题以及组合的优化

回溯算法&#xff1a;递归函数里面嵌套着for循环 给定两个整数 n 和 k&#xff0c;返回 1 … n 中所有可能的 k 个数的组合。 示例: 输入: n 4, k 2 输出: [ [2,4], [3,4], [2,3], [1,2], [1,3], [1,4], ] 包含组合问题和组合问题的剪枝优化 class solution:def combine(se…

离线LaTex公式识别工具V0.9开发

离线LaTex公式识别工具V0.9开发 文章目录 离线LaTex公式识别工具V0.9开发工程来源GUI界面streamlit 模型文件离线调用Pix2TexttexifyNougat-LaTex 打包成exepyinstaller打包方法pyinstaller打包遇到的问题File "subprocess.py", line 1420, in _execute_child FileNo…

数据结构的概念大合集03(栈)

概念大合集03 1、栈1.1 栈的定义和特点1.2 栈的基础操作1.3 栈的顺序存储1.3.1 顺序栈1.3.2 栈空&#xff0c;栈满&#xff0c;进栈&#xff0c;出栈的基本思想1.3.3 共享栈1.3.3.1 共享栈的4要素 1.4 栈的链式存储1.4.1 链栈的实现1.4.2 链栈的4个要素 1、栈 1.1 栈的定义和特…

软考高级:软件工程螺旋模型概念和例题

作者&#xff1a;明明如月学长&#xff0c; CSDN 博客专家&#xff0c;大厂高级 Java 工程师&#xff0c;《性能优化方法论》作者、《解锁大厂思维&#xff1a;剖析《阿里巴巴Java开发手册》》、《再学经典&#xff1a;《Effective Java》独家解析》专栏作者。 热门文章推荐&am…

科研绘图一:箱线图(添加贝赛尔曲线)

R语言绘图系列—箱线图贝赛尔曲线 &#xff08;一&#xff09;: 科研绘图一&#xff1a;箱线图&#xff08;添加贝赛尔曲线&#xff09; 文章目录 R语言绘图系列---箱线图贝赛尔曲线&#xff08;一&#xff09;: 科研绘图一&#xff1a;箱线图&#xff08;添加贝赛尔曲线&…

【人工智能】英文学习材料(每日一句)

#学习笔记# 目录 1.Natural Language Processing&#xff0c;NLP&#xff08;自然语言处理&#xff09; 2.Machine Learing&#xff0c;ML&#xff08;机器学习&#xff09; 3.Neural Networks&#xff08;神经网络&#xff09; 4.Deep Learing&#xff08;深度学习&#…

GiT: Towards Generalist Vision Transformer through Universal Language Interface

GiT: Towards Generalist Vision Transformer through Universal Language Interface 相关链接&#xff1a;arxiv github 关键字&#xff1a;Generalist Vision Transformer (GiT)、Universal Language Interface、Multi-task Learning、Zero-shot Transfer、Transformer 摘要 …

浏览器如何进行静态资源缓存?—— 强缓存 协商缓存

在平时使用浏览器排查问题的过程中&#xff0c;我们有时会看到浏览器网络请求中出现304状态码&#xff0c;那么是什么情况下出现304呢&#xff1f;下面是关于这一现象的解释&#xff1a; 浏览器如何进行静态资源缓存&#xff1f;—— 强缓存 & 协商缓存 状态码 304浏览器如…