动态通讯录及程序保存在文件中

目录

一、结构体改造及增容函数

1.结构体部分

2.初始化函数及增容函数

二、信息添加及销毁和排序

1.信息添加函数(Add)

2.销毁函数(Destroy)

3.排序部分(qsort)

三、通讯录信息保存

1.保存在文件中(输出操作)

2.加载通讯录(输入操作)

四、整理通讯录

1.在菜单按0后退出程序

2.其他方式退出程序

3.打开程序就加载通讯录

4.完整通讯录代码


前言:这一次是在之前静态的通讯录基础上进行改造;变成动态通讯录,当空间不够时,可以完成自动增容,并且将通讯录中的信息保存在文件中,退出或关闭程序就不会再丢失数据。

先看静态通讯录的代码:

#define _CRT_SECURE_NO_WARNINGS 1#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<stdlib.h>#define NAME_MAX 20
#define SEX_MAX 6
#define TELE_MAX 20
#define ADDR_MAX 20#define MAX 100typedef struct PeoInfo
{char name[NAME_MAX];//姓名char sex[SEX_MAX];//性别int age;//年龄char tele[TELE_MAX];//电话char addr[ADDR_MAX];//地址
}PeoInfo;typedef struct contact
{PeoInfo data[MAX];//通讯录int sz;//记录通讯录的个数
}contact;//函数的声明
//初始化void InitContact(contact* pc);//添加用户信息void AddContact(contact* pc);//打印通讯录void ShowContact(contact* pc);//删除联系人void DelContact(contact* pc);//查找某个联系人void SearchContact(contact* pc);//修改联系人void ModifyContact(contact* pc);//联系人排序void SortContact(contact* pc);//函数功能的实现void InitContact(contact* pc)
{assert(pc);memset(pc->data,0,sizeof(pc->data));pc->sz = 0;
}
void ShowContact(contact* pc)
{assert(pc);if (pc->sz == 0){printf("通讯录为空\n");return;}printf("%-10s %-5s %-10s %-15s %-10s\n","名字","性别","年龄","电话","住址");int i = 0;for (i=0;i<pc->sz;i++){printf("%-10s %-5s %-10d %-15s %-10s\n",pc->data[i].name, pc->data[i].sex, pc->data[i].age,pc->data[i].tele, pc->data[i].addr);}printf("\n");
}void AddContact(contact* pc)
{assert(pc);int adds;back:if (pc->sz == MAX){printf("通讯录已满,存入信息失败\n");return;}printf("请输入姓名>:");scanf("%s", pc->data[pc->sz].name);printf("请输入性别>:");scanf("%s", pc->data[pc->sz].sex);printf("请输入年龄>:");scanf("%d", &(pc->data[pc->sz].age));printf("请输入电话>:");scanf("%s", pc->data[pc->sz].tele);printf("请输入住址>:");scanf("%s", pc->data[pc->sz].addr);pc->sz++;printf("信息添加成功\n");printf("是否继续添加联系人信息1/0:");scanf("%d",&adds);if (adds == 1)goto back;else{return;}
}
//查看某个联系人是否存在
static int FindContact(contact* pc,char name[])
{assert(pc);int i = 0;for (i=0;i<pc->sz;i++){if (strcmp(name, pc->data[i].name) == 0)return i;}return -1;
}
//删除联系人
void DelContact(contact* pc)
{assert(pc);if (pc->sz == 0){printf("通讯录为空,删除失败\n");return;}printf("请输入你要删除的联系人:");char name[NAME_MAX];scanf("%s",name);int ret = FindContact(pc,name);if (ret == -1){printf("联系人不存在,删除失败\n");return;}int i = 0;for (i=ret;i<pc->sz-1;i++){pc->data[i] = pc->data[i + 1];}pc->sz--;printf("删除联系人成功\n");
}
//查找联系人
void SearchContact(contact* pc)
{assert(pc);printf("请输入你要查找联系人的名字:");char name[NAME_MAX];scanf("%s", name);int ret = FindContact(pc, name);if (ret == -1){printf("联系人不存在,查找失败\n");return;}printf("查找成功:\n");printf("%-10s %-5s %-10s %-15s %-10s\n", "名字", "性别", "年龄", "电话", "住址");printf("%-10s %-5s %-10d %-15s %-10s\n",pc->data[ret].name, pc->data[ret].sex, pc->data[ret].age,pc->data[ret].tele, pc->data[ret].addr);
}
//修改联系人
void ModifyContact(contact* pc)
{assert(pc);printf("请输入你要查找联系人的名字:");char name[NAME_MAX];scanf("%s", name);int ret = FindContact(pc, name);if (ret == -1){printf("联系人不存在,修改失败\n");return;}printf("联系人存在:");printf("%-10s %-5s %-10d %-15s %-10s\n",pc->data[ret].name, pc->data[ret].sex, pc->data[ret].age,pc->data[ret].tele, pc->data[ret].addr);printf("请修改姓名>:");scanf("%s", pc->data[ret].name);printf("请修改性别>:");scanf("%s", pc->data[ret].sex);printf("请修改年龄>:");scanf("%d", &(pc->data[ret].age));printf("请修改电话>:");scanf("%s", pc->data[ret].tele);printf("请修改住址>:");scanf("%s", pc->data[ret].addr);printf("\n修改成功");
}
//分类菜单
void menu2()
{printf("*************************\n");printf("**** 1.名字   2.年龄 ****\n");printf("*************************\n");}
//名字排序
int qsort_cmp_name(const void* e1,const void* e2)
{return strcmp((((contact*)e1)->data)->name, (((contact*)e2)->data)->name);
}
//年龄排序
int qsort_cmp_age(const void* e1,const void* e2)
{return (((contact*)e1)->data)->age - (((contact*)e2)->data)->age;
}
//联系人排序
void SortContact(contact* pc)
{int input2;menu2();printf("请选择排序方式:");scanf("%d",&input2);switch (input2){case 1:qsort(pc->data, pc->sz, sizeof(pc->data[0]), qsort_cmp_name);case 2:qsort(pc->data, pc->sz, sizeof(pc->data[0]), qsort_cmp_age);defualt:printf("选择错误\n");break;}
}//主函数及菜单
void menu()
{printf("********************************\n");printf("**** 1. add      2. del     ****\n");	printf("**** 3. search   4. modify  ****\n");printf("**** 5. show     6. sort    ****\n");printf("**** 0. exit                ****\n");printf("********************************\n");
}
enum Option
{
EXIT,//退出
ADD,//增加
DEL,//删除联系人
SEARCH,//查找联系人
MODIFY,//修改指定联系人
SHOW,//打印联系人
SORT,//分类
};
int main()
{contact con;InitContact(&con);int input;do{menu();printf("请输入你的选择>:");scanf("%d",&input);switch (input){case ADD:AddContact(&con);break;case DEL:DelContact(&con);break;case SEARCH:SearchContact(&con);break;case MODIFY:ModifyContact(&con);break;case SHOW:ShowContact(&con);break;case SORT:SortContact(&con);break;case EXIT:printf("你已选择退出程序\n");break;default:printf("选择错误,请重新选择\n");break;}} while (input);return 0;
}

我们将对其进行改造。

一、结构体改造及增容函数

1.结构体部分

typedef struct PeoInfo
{char name[NAME_MAX];//姓名char sex[SEX_MAX];//性别int age;//年龄char tele[TELE_MAX];//电话char addr[ADDR_MAX];//地址
}PeoInfo;
typedef struct contact
{PeoInfo* data;//通讯录指针int capacity;//通讯录最大容量int sz;//记录通讯录的个数
}contact;

(1)通讯录的内容不需要修改。

(2)因为数组是不可能动态变化大小,所以改成指针,可以指向一块空间,就可以使用realloc进行增容。

2.初始化函数及增容函数

(1)初始化函数

#define INCAPA 3//capacity初始容量//动态初始化版本
void InitContact(contact* pc)
{assert(pc);pc->sz = 0;pc->capacity = INCAPA;pc->data = calloc(pc->capacity, sizeof(PeoInfo));if (pc->data == NULL){perror("InitContact->calloc");return 1;}
}

(1)sz刚开始是0,capacity开始的容量我们赋值INCAPA,也就是3

(2)使用calloc将开辟好的一块空间赋值给data

(2)增容函数 

//增容函数
void CheckCapacity(contact* pc)
{if (pc->sz == pc->capacity){PeoInfo* str = (PeoInfo*)realloc(pc->data, (pc->capacity + 5) * sizeof(PeoInfo));if (str != NULL){pc->data = str;pc->capacity += 5;printf("增容成功\n");}else{perror("CheckCapacity->realloc");return;}}
}

(1)该函数用来扩容

(2)sz==capacity说明通讯录已满需要扩容,每次增加5个空间

二、信息添加及销毁和排序

1.信息添加函数(Add)

//动态版本
void AddContact(contact* pc)
{assert(pc);CheckCapacity(pc);//检查通讯录是否满int adds;
back:printf("请输入姓名>:");scanf("%s", pc->data[pc->sz].name);printf("请输入性别>:");scanf("%s", pc->data[pc->sz].sex);printf("请输入年龄>:");scanf("%d", &(pc->data[pc->sz].age));printf("请输入电话>:");scanf("%s", pc->data[pc->sz].tele);printf("请输入住址>:");scanf("%s", pc->data[pc->sz].addr);pc->sz++;printf("信息添加成功\n");printf("是否继续添加联系人信息1/0:");scanf("%d", &adds);if (adds == 1)goto back;else{return;}
}

(1)在添加信息的时候,检查容量是否已满,满则扩容

2.销毁函数(Destroy)

//销毁通讯录
void DestroyContact(contact* pc)
{free(pc->data);pc->data = NULL;pc->sz = 0;pc->capacity = 0;
}

calloc和realloc开辟的内存为动态内存,程序结束需要及时释放

3.排序部分(qsort)

void menu2()
{printf("*************************\n");printf("**** 1.名字   2.年龄 ****\n");printf("*************************\n");
}
//动态排序
//名字排序
int qsort_cmp_name(const void* e1, const void* e2)
{return strcmp(((PeoInfo*)e1)->name, ((PeoInfo*)e2)->name);
}
//年龄排序
int qsort_cmp_age(const void* e1, const void* e2)
{return ((PeoInfo*)e1)->age - ((PeoInfo*)e2)->age;
}
//联系人排序
void SortContact(contact* pc)
{int input2;menu2();printf("请选择排序方式:");scanf("%d", &input2);switch (input2){case 1:qsort(pc->data, pc->sz, sizeof(PeoInfo), qsort_cmp_name); break;case 2:qsort(pc->data, pc->sz, sizeof(PeoInfo), qsort_cmp_age); break;defualt:printf("选择错误\n");break;}
}

这个排序代码同样适用于静态通讯录。

三、通讯录信息保存

1.保存在文件中(输出操作)

为了将通讯录中的信息保存下来,所以我们需要将程序写入文件中,从而可以达到保存数据的目的,下面是文件保存的代码:

//通讯录信息保存
void SaveContact(contact* pc)
{assert(pc);//打开文件FILE* pf = fopen("contact.txt", "wb");if (pf == NULL){perror("SaveContact");return 1;}//写文件int i = 0;for (i = 0; i < pc->sz; i++){fwrite(pc->data + i, sizeof(PeoInfo), 1, pf);//fwrite写文件更方便}//关闭文件fclose(pf);pf = NULL;
}

(1)我们利用fwrite函数以二进制的形式将数据写入contact.txt的文本中、

(2)每一次写一个人的信息

2.加载通讯录(输入操作)

单单将数据保存到文件中还不行,下次再将程序运行起来,还是一个空的通讯录,因为上次的数据在文件中,所以我们需要将文件中的数据输入程序中,这就是读文件的操作:

//加载通讯录
void LoadContact(contact* pc)
{assert(pc);FILE* pf = fopen("contact.txt", "rb");if (pf == NULL){perror("LoadContact");return 1;}PeoInfo tmp = { 0 };//用来存放fread读写的信息while (fread(&tmp, sizeof(PeoInfo), 1, pf)){//检查容量CheckCapacity(pc);pc->data[pc->sz] = tmp;pc->sz++;}fclose(pf);pf = NULL;
}

(1)我们将文件中的通讯录信息加载出来,需要考虑通讯录的空间是否足够,否则需要增容

(2)然后同理将通讯录的信息一个个输出出来

四、整理通讯录

上述已经写好了文件保存的程序,接下来就需要调用,当程序退出时就可以及时保存到文件中。

退出的情况有很多种:

1.在菜单按0后退出程序

下面的代码是正常退出程序时的操作:

case EXIT:SaveContact(&con);//将通讯录保存于文件中DestroyContact(&con);printf("你已选择退出程序\n");break;

2.其他方式退出程序

暴力退出,如:直接叉掉程序、直接关掉软件(VS),这些也需要单独考虑

(1)增加信息后暴力退出

那我们就在增加信息函数后面加上一个文件保存函数,每次调用增容函数后就及时保存

case ADD:AddContact(&con);SaveContact(&con);break;

(2)删除信息后暴力退出

删除信息后不及时保存,然后暴力退出依旧达不到删除信息的目的,所以也需要在调用删除函数后面补上文件保存函数。

case DEL:DelContact(&con);SaveContact(&con);break;

(3)排序后暴力退出

排序同理,排完序就及时保存。

case SORT:SortContact(&con);SaveContact(&con);break;

(4)修改信息后暴力退出

修改同理,修改后及时保存信息 

case MODIFY:ModifyContact(&con);SaveContact(&con);break;

3.打开程序就加载通讯录

当每次把程序运行起来之后,就希望已经将文件中的信息加载到程序中了,所以需要在初始化函数处进行改造,加上加载文件的函数接口即可

void InitContact(contact* pc)
{assert(pc);pc->sz = 0;pc->capacity = INCAPA;pc->data = calloc(pc->capacity, sizeof(PeoInfo));if (pc->data == NULL){perror("InitContact->calloc");return 1;}//每次初始化前可以先打开之前的通讯录LoadContact(pc);
}

4.完整通讯录代码

#define _CRT_SECURE_NO_WARNINGS 1#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<stdlib.h>#define NAME_MAX 20
#define SEX_MAX 6
#define TELE_MAX 20
#define ADDR_MAX 30#define MAX 100#define INCAPA 3//capacity初始容量typedef struct PeoInfo
{char name[NAME_MAX];//姓名char sex[SEX_MAX];//性别int age;//年龄char tele[TELE_MAX];//电话char addr[ADDR_MAX];//地址
}PeoInfo;//静态的通讯录
//typedef struct contact
//{
//	PeoInfo data[MAX];//通讯录
//	int sz;//记录通讯录的个数
//}contact;//动态版本
typedef struct contact
{PeoInfo* data;//通讯录int capacity;//通讯录最大容量int sz;//记录通讯录的个数
}contact;//初始化
void InitContact(contact* pc);//添加用户信息
void AddContact(contact* pc);//打印通讯录
void ShowContact(contact* pc);//删除联系人
void DelContact(contact* pc);//查找某个联系人
void SearchContact(contact* pc);//修改联系人
void ModifyContact(contact* pc);//联系人分类
void SortContact(contact* pc);//增容函数
void CheckCapacity(contact* pc);//销毁通讯录
void DestroyContact(contact* pc);//文件保存函数
void SaveContact(contact* pc);//加载通讯录
void LoadContact(contact* pc);#define _CRT_SECURE_NO_WARNINGS 1//静态初始化
//void InitContact(contact* pc)
//{
//	assert(pc);
//	memset(pc->data, 0, sizeof(pc->data));
//	pc->sz = 0;
//}
//动态初始化版本
void InitContact(contact* pc)
{assert(pc);pc->sz = 0;pc->capacity = INCAPA;pc->data = calloc(pc->capacity, sizeof(PeoInfo));if (pc->data == NULL){perror("InitContact->calloc");return 1;}//每次初始化前可以先打开之前的通讯录LoadContact(pc);
}//加载通讯录
void LoadContact(contact* pc)
{assert(pc);FILE* pf = fopen("contact.txt", "rb");if (pf == NULL){perror("LoadContact");return 1;}PeoInfo tmp = { 0 };//用来存放fread读写的信息while (fread(&tmp, sizeof(PeoInfo), 1, pf)){//检查容量CheckCapacity(pc);pc->data[pc->sz] = tmp;pc->sz++;}fclose(pf);pf = NULL;
}//销毁通讯录
void DestroyContact(contact* pc)
{free(pc->data);pc->data = NULL;pc->sz = 0;pc->capacity = 0;
}void ShowContact(contact* pc)
{assert(pc);if (pc->sz == 0){printf("通讯录为空\n");return;}printf("%-10s %-5s %-10s %-15s %-10s\n", "名字", "性别", "年龄", "电话", "住址");int i = 0;for (i = 0; i < pc->sz; i++){printf("%-10s %-5s %-10d %-15s %-10s\n",pc->data[i].name, pc->data[i].sex, pc->data[i].age,pc->data[i].tele, pc->data[i].addr);}printf("\n");
}
//静态版本
//void AddContact(contact* pc)
//{
//	assert(pc);
//	int adds;
//back:
//	if (pc->sz == MAX)
//	{
//		printf("通讯录已满,存入信息失败\n");
//		return;
//	}
//
//	printf("请输入姓名>:");
//	scanf("%s", pc->data[pc->sz].name);
//	printf("请输入性别>:");
//	scanf("%s", pc->data[pc->sz].sex);
//	printf("请输入年龄>:");
//	scanf("%d", &(pc->data[pc->sz].age));
//	printf("请输入电话>:");
//	scanf("%s", pc->data[pc->sz].tele);
//	printf("请输入住址>:");
//	scanf("%s", pc->data[pc->sz].addr);
//	pc->sz++;
//	printf("信息添加成功\n");
//
//	printf("是否继续添加联系人信息1/0:");
//	scanf("%d", &adds);
//	if (adds == 1)
//		goto back;
//	else
//	{
//		return;
//	}
//}//增容函数
void CheckCapacity(contact* pc)
{if (pc->sz == pc->capacity){PeoInfo* str = (PeoInfo*)realloc(pc->data, (pc->capacity + 5) * sizeof(PeoInfo));if (str != NULL){pc->data = str;pc->capacity += 5;printf("增容成功\n");}else{perror("CheckCapacity->realloc");return;}}
}//动态版本
void AddContact(contact* pc)
{assert(pc);CheckCapacity(pc);//检查通讯录是否满int adds;
back:printf("请输入姓名>:");scanf("%s", pc->data[pc->sz].name);printf("请输入性别>:");scanf("%s", pc->data[pc->sz].sex);printf("请输入年龄>:");scanf("%d", &(pc->data[pc->sz].age));printf("请输入电话>:");scanf("%s", pc->data[pc->sz].tele);printf("请输入住址>:");scanf("%s", pc->data[pc->sz].addr);pc->sz++;printf("信息添加成功\n");printf("是否继续添加联系人信息1/0:");scanf("%d", &adds);if (adds == 1)goto back;else{return;}
}
//通讯录信息保存
void SaveContact(contact* pc)
{assert(pc);//打开文件FILE* pf = fopen("contact.txt", "wb");if (pf == NULL){perror("SaveContact");return 1;}//写文件int i = 0;for (i = 0; i < pc->sz; i++){fwrite(pc->data + i, sizeof(PeoInfo), 1, pf);//fwrite写文件更方便}//关闭文件fclose(pf);pf = NULL;
}//查看某个联系人是否存在
static int FindContact(contact* pc, char name[])
{assert(pc);int i = 0;for (i = 0; i < pc->sz; i++){if (strcmp(name, pc->data[i].name) == 0)return i;}return -1;
}
//删除联系人
void DelContact(contact* pc)
{assert(pc);if (pc->sz == 0){printf("通讯录为空,删除失败\n");return;}printf("请输入你要删除的联系人:");char name[NAME_MAX];scanf("%s", name);int ret = FindContact(pc, name);if (ret == -1){printf("联系人不存在,删除失败\n");return;}int i = 0;for (i = ret; i < pc->sz - 1; i++){pc->data[i] = pc->data[i + 1];}pc->sz--;printf("删除联系人成功\n");
}
//查找联系人
void SearchContact(contact* pc)
{assert(pc);printf("请输入你要查找联系人的名字:");char name[NAME_MAX];scanf("%s", name);int ret = FindContact(pc, name);if (ret == -1){printf("联系人不存在,查找失败\n");return;}printf("查找成功:\n");printf("%-10s %-5s %-10s %-15s %-10s\n", "名字", "性别", "年龄", "电话", "住址");printf("%-10s %-5s %-10d %-15s %-10s\n",pc->data[ret].name, pc->data[ret].sex, pc->data[ret].age,pc->data[ret].tele, pc->data[ret].addr);
}
//修改联系人
void ModifyContact(contact* pc)
{assert(pc);printf("请输入你要查找联系人的名字:");char name[NAME_MAX];scanf("%s", name);int ret = FindContact(pc, name);if (ret == -1){printf("联系人不存在,修改失败\n");return;}printf("联系人存在:");printf("%-10s %-5s %-10d %-15s %-10s\n",pc->data[ret].name, pc->data[ret].sex, pc->data[ret].age,pc->data[ret].tele, pc->data[ret].addr);printf("请修改姓名>:");scanf("%s", pc->data[ret].name);printf("请修改性别>:");scanf("%s", pc->data[ret].sex);printf("请修改年龄>:");scanf("%d", &(pc->data[ret].age));printf("请修改电话>:");scanf("%s", pc->data[ret].tele);printf("请修改住址>:");scanf("%s", pc->data[ret].addr);printf("\n修改成功\n");
}
//分类菜单
void menu2()
{printf("*************************\n");printf("**** 1.名字   2.年龄 ****\n");printf("*************************\n");}
//静态排序
//名字排序
//int qsort_cmp_name(const void* e1, const void* e2)
//{
//	return strcmp((((contact*)e1)->data)->name, (((contact*)e2)->data)->name);
//}
年龄排序
//int qsort_cmp_age(const void* e1, const void* e2)
//{
//	return (((contact*)e1)->data)->age - (((contact*)e2)->data)->age;
//}
联系人排序
//void SortContact(contact* pc)
//{
//	int input2;
//	menu2();
//	printf("请选择排序方式:");
//	scanf("%d", &input2);
//	switch (input2)
//	{
//	case 1:qsort(pc->data, pc->sz, sizeof(pc->data[0]), qsort_cmp_name); break;
//	case 2:qsort(pc->data, pc->sz, sizeof(pc->data[0]), qsort_cmp_age); break;
//	defualt:printf("选择错误\n");
//		break;
//	}
//}//动态排序
//名字排序
int qsort_cmp_name(const void* e1, const void* e2)
{return strcmp(((PeoInfo*)e1)->name, ((PeoInfo*)e2)->name);
}
//年龄排序
int qsort_cmp_age(const void* e1, const void* e2)
{return ((PeoInfo*)e1)->age - ((PeoInfo*)e2)->age;
}
//联系人排序
void SortContact(contact* pc)
{int input2;menu2();printf("请选择排序方式:");scanf("%d", &input2);switch (input2){case 1:qsort(pc->data, pc->sz, sizeof(PeoInfo), qsort_cmp_name); break;case 2:qsort(pc->data, pc->sz, sizeof(PeoInfo), qsort_cmp_age); break;defualt:printf("选择错误\n");break;}
}void menu()
{printf("********************************\n");printf("**** 1. add      2. del     ****\n");printf("**** 3. search   4. modify  ****\n");printf("**** 5. show     6. sort    ****\n");printf("**** 0. exit                ****\n");printf("********************************\n");
}
enum Option
{EXIT,//退出ADD,//增加DEL,//删除联系人SEARCH,//查找联系人MODIFY,//修改指定联系人SHOW,//打印联系人SORT,//分类
};
int main()
{contact con;InitContact(&con);int input=0;do{menu();printf("请输入你的选择>:");scanf("%d", &input);switch (input){case ADD:AddContact(&con);SaveContact(&con);break;case DEL:DelContact(&con);SaveContact(&con);break;case SEARCH:SearchContact(&con);break;case MODIFY:ModifyContact(&con);SaveContact(&con);break;case SHOW:ShowContact(&con);break;case SORT:SortContact(&con);SaveContact(&con);break;case EXIT:SaveContact(&con);//将通讯录保存于文件中DestroyContact(&con);printf("你已选择退出程序\n");break;default:printf("选择错误,请重新选择\n");break;}} while (input);return 0;
}

运行结果:

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

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

相关文章

Flutter笔记:光影动画按钮、滚动图标卡片组等

Flutter笔记 scale_design更新&#xff1a;光影动画按钮、滚动图标卡片组 作者&#xff1a;李俊才 &#xff08;jcLee95&#xff09;&#xff1a;https://blog.csdn.net/qq_28550263 邮箱 &#xff1a;291148484163.com 本文地址&#xff1a;https://blog.csdn.net/qq_28550263…

SpringData、SparkStreaming和Flink集成Elasticsearch

本文代码链接&#xff1a;https://download.csdn.net/download/shangjg03/88522188 1 Spring Data框架集成 1.1 Spring Data框架介绍 Spring Data是一个用于简化数据库、非关系型数据库、索引库访问&#xff0c;并支持云服务的开源框架。其主要目标是使得对数据的访问变得方便快…

成集云 | 英克对接零售O2O+线上商城 | 解决方案

方案介绍 零售O2O线上商城是一种新型的商业模式&#xff0c;它通过线上和线下的融合&#xff0c;提供更加便捷的购物体验。其中&#xff0c;O2O指的是线上与线下的结合&#xff0c;通过互联网平台与实体店面的结合&#xff0c;实现线上线下的互动和协同。线上商城则是指通过互…

Git的进阶操作,在idea中部署gie

&#x1f3c5;我是默&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; ​​ &#x1f31f;在这里&#xff0c;我要推荐给大家我的专栏《git》。&#x1f3af;&#x1f3af; &#x1f680;无论你是编程小白&#xff0c;还是有一定基础的程序员&#xff0c;这…

[工业自动化-9]:西门子S7-15xxx编程 - PLC主站 - 信号量:模拟量

目录 前言&#xff1a; 一、模拟量模块 1.1 概述 1.2 安装 1.3 模拟量链接线 二、模拟量常见问题 2.1 两线制、四线制&#xff08;电流&#xff09; 2.2 模拟量模块的参数 2.3 差分信号与单端信号 三、如何防止电磁干扰 3.1 概述 3.2 工业现场的电磁干扰源来源 3.…

Hive3 on Spark3配置

1、软件环境 1.1 大数据组件环境 大数据组件版本Hive3.1.2Sparkspark-3.0.0-bin-hadoop3.2 1.2 操作系统环境 OS版本MacOSMonterey 12.1Linux - CentOS7.6 2、大数据组件搭建 2.1 Hive环境搭建 1&#xff09;Hive on Spark说明 Hive引擎包括&#xff1a;默认 mr、spark、…

IP行业API助力于网络分析和数据挖掘

引言 在当今数字化时代&#xff0c;数据成为了企业、科研机构和政府决策者的重要资源&#xff0c;而IP行业API则成为了数据分析及挖掘的工具之一。IP行业API是一种能够查询IP地址所属的行业分类信息的应用程序接口&#xff0c;它能够提供在网络分析、用户行为分析及大数据挖掘…

ChatRule:基于知识图推理的大语言模型逻辑规则挖掘11.10

ChatRule&#xff1a;基于知识图推理的大语言模型逻辑规则挖掘 摘要引言相关工作初始化和问题定义方法实验 摘要 逻辑规则对于揭示关系之间的逻辑联系至关重要&#xff0c;这可以提高推理性能并在知识图谱&#xff08;KG&#xff09;上提供可解释的结果。虽然已经有许多努力&a…

【LeetCode刷题笔记】堆和优先级队列

358. K 距离间隔重排字符串 解题思路: 大根堆 + 队列 , 1)首先 计数数组 统计 每个字符出现的次数 ,然后将 计数 > 0 的 字符 和 次数 一起放入 大根堆 ,大根堆中

高阶组件和Hooks

目录 1. 高阶组件&#xff08;Higher-Order Components&#xff09; 1.1 创建高阶组件 1.2 使用高阶组件 2. Hooks 2.1 使用useState Hook管理状态 2.2 创建自定义Hook 结论 1. 高阶组件&#xff08;Higher-Order Components&#xff09; 高阶组件是一个接受一个组件作为…

【Bug】Python利用matplotlib绘图无法显示中文解决办法

一&#xff0c;问题描述 当利用matplotlib进行图形绘制时&#xff0c;图表标题&#xff0c;坐标轴&#xff0c;标签中文无法显示&#xff0c;显示为方框&#xff0c;并报错 运行窗口报错&#xff1a; 这是中文字体格式未导入的缘故。 二&#xff0c;解决方案 在代码import部…

在 Arduino IDE 2.0 中安装 ESP32 板(Windows、Mac OS X、Linux)

有一个新的 Arduino IDE——Arduino IDE 2.0&#xff08;测试版&#xff09;。在本教程中&#xff0c;您将学习如何在 Arduino IDE 2.0 中安装 ESP32 板并将代码上传到板。本教程与 Windows、Mac OS X 和 Linux 操作系统兼容。 据 Arduino 网站称&#xff1a;“ Arduino IDE 2.…

vscode调试react 最初的源码

如果直接在react项目中打点调试, 调试的是 react-dom.development.js, 而源码里这些逻辑是分散在不同的包里的,如何才能够调试 React 最初的源码呢&#xff1f; JS 代码经过编译&#xff0c;会产生目标代码&#xff0c;但同时也会产生 sourcemap。sourcemap 的作用就是映射目…

DAY50 309.最佳买卖股票时机含冷冻期 + 714.买卖股票的最佳时机含手续费

309.最佳买卖股票时机含冷冻期 题目要求&#xff1a;给定一个整数数组&#xff0c;其中第 i 个元素代表了第 i 天的股票价格 。 设计一个算法计算出最大利润。在满足以下约束条件下&#xff0c;你可以尽可能地完成更多的交易&#xff08;多次买卖一支股票&#xff09;: 你不…

HCIP---VRRP

文章目录 前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 一. VRRP概述 VRRP---虚拟路由器冗余协议 VRRP&#xff08;Virtual Router Redundancy Protocol&#xff09;是一种用于在多个路由器之间创建虚拟路由器的协议。 VRRP使用了一系列协议来实现路…

新零售时代,传统便利店如何转型?

在零售批发业&#xff0c;如何降低各环节成本、提高业务运转效率、更科学地了解客户服务客户&#xff0c;是每家企业在激烈竞争中需要思考的课题。 对零售批发企业来说&#xff0c;这些问题或许由来已久&#xff1a; &#xff08;1&#xff09;如何对各岗位的员工进行科学的考…

嵌入式系统中,输入网址之后,发生了什么?

让我们一步一步地来看这个过程。 步骤1&#xff1a; 用户在浏览器中输入一个URL&#xff08;比如www.bytebytego.com&#xff09;&#xff0c;然后按下回车键。首先&#xff0c;我们需要将这个URL转换成一个IP地址。通常&#xff0c;这个映射关系会被存储在缓存中&#xff0c…

uniapp和vue3+ts开发小程序,使用vscode提示声明变量冲突解决办法

在uniapp中&#xff0c;我们可能经常会遇到需要在不用的环境中使用不同变量的场景&#xff0c;例如在VUE3中的小程序环境使用下面的方式导入echarts&#xff1a; const echarts require(../../static/echarts.min); 如果不是小程序环境则使用下面的方式导入echarts&#xff…

Azure 机器学习 - 有关为 Azure 机器学习配置 Kubernetes 群集的参考

目录 受支持的 Kubernetes 版本和区域建议的资源计划ARO 或 OCP 群集的先决条件禁用安全增强型 Linux (SELinux)ARO 和 OCP 的特权设置 收集的日志详细信息Azure 机器学习作业与自定义数据存储连接支持的 Azure 机器学习排斥和容许最佳实践 通过 HTTP 或 HTTPS 将其他入口控制器…

【计算机网络笔记】Internet网络的网络层——IP协议之IP数据报的结构

系列文章目录 什么是计算机网络&#xff1f; 什么是网络协议&#xff1f; 计算机网络的结构 数据交换之电路交换 数据交换之报文交换和分组交换 分组交换 vs 电路交换 计算机网络性能&#xff08;1&#xff09;——速率、带宽、延迟 计算机网络性能&#xff08;2&#xff09;…