【C语言进阶篇】结构体都学完了,那么现造第一个通讯录呢?(内附源码)

news2025/1/18 9:59:41

在这里插入图片描述

🎬 鸽芷咕:个人主页

 🔥 个人专栏:《C语言初阶篇》 《C语言进阶篇》

⛺️生活的理想,就是为了理想的生活!

文章目录

  • 📋 前言
  • 一 、 通讯录的简介
    • 1.1 联系人的类型定义
    • 1.2 通讯录的定义
    • 1.3 通讯录要实现的功能
  • 二 、 如何实现这些功能
    • 2.1 test.c 的实现
      • mian() 函数的实现
      • menu() 菜单函数的实现
      • test() 通讯录选择的实现
    • 2.2 Contact.h 的声明
    • 2.3 Contact.c 功能函数的定义
      • 0️⃣ 初始化通讯录
      • 1️⃣ 新增联系人的实现
      • 2️⃣ 删除联系人的实现
      • 3️⃣查询联系人的实现
      • 4️⃣ 修改联系人的实现
      • 5️⃣ 查看所有联系人
      • 6️⃣ 排序联系人
  • 三、通讯录功能的测试
      • 1️⃣ 新增联系人的测试
      • 2️⃣ 删除联系人的测试
      • 3️⃣查询联系人的测试
      • 4️⃣ 修改联系人的测试
      • 5️⃣ 查看所有联系人
      • 6️⃣ 排序联系人
  • 四、通讯录整体工程
        • test.c
        • contact.c
        • contact.h
  • 📝全篇总结

📋 前言

  🌈hello! 各位宝子们大家好啊,结构体我们都学完了,那么我们今天就来点实战把!
  ⛳️给大家现编一个通讯录,其实并不难只需要用到我们的结构体知识就可以,大家一起动动手吧!
  📚本期文章收录在《C语言进阶篇》,大家有兴趣可以看看呐
  ⛺️ 欢迎铁汁们 ✔️ 点赞 👍 收藏 ⭐留言 📝!

🔥 注:结构体的文章在这里嗷!《结构体的万字解析》

一 、 通讯录的简介

通讯录大家可以说是在熟悉不过了,那么今天就来用我们所学的C语言知识实现一下。通讯录无非就是增加联系人和删除等,增删查改这些功能。>

  • 而每个 联系人 又是 不同元素 的集合
  • 这时我们的结构体就排上用场了

1.1 联系人的类型定义

既然是联系人,那么我们相信大家一定储存的都是联系人的:

  • 姓名 年龄 性别 电话 地址
  • 这些基本的元素,这些知道了我们的结构体也就可以定义了

📚 代码演示:

由于数组的数字使用起来不方便更改和没有什么特殊意义,所以我们就把这些数组可以定义的宏来。

  • 可以让数组大小更容易更改
  • 还可以让别人一眼就知道这些数字的含义
#define MAX 1000
#define MAX_NAME  20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30

//联系人的类型定义
typedef struct PeoInfo
{
	char name[MAX_NAME];
	int age;
	char sex[MAX_SEX];
	char tele[MAX_TELE];
	char addr[MAX_ADDR];
}PeoInfo;

1.2 通讯录的定义

联系人的类型我们知道了那么通讯录还不要创建嘛,首先我们需要一个数组来存放每个人的信息!和一个整形来统计我们存放了多少人。

  • 那么就用 PeoInfo data[100]来负责存放100个联系人的信息。
  • int sz,来统计存放了多少个人
  • 这里由于不知道到底存放多少个字节合适所以用宏定义一个MAX定义为100也好修改

📚 代码演示:

//通讯录的定义
#define MAX 100
typedef struct Contact
{
	PeoInfo data[MAX];//指向了存放数据的空间
	int sz;//记录当前存放的有效元素的个数
}Contact;

1.3 通讯录要实现的功能

上面我们给大家说了,通讯录的一些功能大家都清楚那么该怎么实现呢?今天就来给大家用多文件的形式一起实现一下!

  • 我们来先看一下大纲是怎么样的

在这里插入图片描述

二 、 如何实现这些功能

2.1 test.c 的实现

在这个文件里面就是我们的主文件用来负责测试和调用函数的,main()函数就在次文件里面包含着!

  • 首先、这个文件要完成菜单的选择以及测试函数的的调用
  • 二、是操作通讯的选项实现

mian() 函数的实现

main函数的功能很简单就是调用test() 函数进行测试就好了其他什么也不用干!

📚 代码演示:

int main()
{
	test();
	return 0;
 }

menu() 菜单函数的实现

这里做一个简易的菜单就好了,只需要让使用者知道每个选项该怎么操作就好了!

📚 代码演示:

//菜单
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");
}

test() 通讯录选择的实现

既然是通讯录的选择,那么我们一般使用的都是多分支语句 switch 所以我们这次也按照 switch 来实现每次操作的选择。

  • 由于 switch 语句里面的 1 2 3 数字的含义不是很明确
  • 所以我们使用枚举来列举一下,这样每个选项是什么意思
  • 就一目了然了

🔥 注:不会枚举的可以看看这篇文章《枚举 联合 位段》里面有详细讲解哦!一看就懂!

📚 代码演示:


//枚举选项
enum  OPTION
{
	  EXIT,
	  ADD,  
	  DEL,  
	  SEARCH,  
	  MODIFY,  
	  SHOW,  
	  SORT,  
};


//测试通讯录
void test()
{
	int input = 0;
	Contact con;
	InitContact(&con);
	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);
}

2.2 Contact.h 的声明

好了主文件我们的编写完了,接下来就是对我们所调用的函数进行声明和定义。

  • 而点h 文件刚好是用来声明函数的
  • 下面我们就把需要调用的函数先声明一下后面去实现

🔥 注:由于很多头文件我们,每个文件都要调用,而Contact.h 这个文件我们也需要调用。

  • 所以就把一些都要用的声明提前写到点 h 的文件里面
  • 例如前面的声明,和宏这些放到点 h 的文件里面使用起来方便一些

📚 代码演示:

#pragma once
#include <string.h>
#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define MAX 1000
#define MAX_NAME  20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30

#define DEFAULT_SZ 3
#define INC_SZ 2

//联系人的类型定义
typedef struct PeoInfo
{
	char name[MAX_NAME];
	int age;
	char sex[MAX_SEX];
	char tele[MAX_TELE];
	char addr[MAX_ADDR];
}PeoInfo;

//动态通讯录
typedef struct Contact
{
	PeoInfo* data;//指向了存放数据的空间
	int sz;//记录当前存放的有效元素的个数
	int capacity;//通讯录当前的最大容量
}Contact;

//枚举选项
enum  OPTION
{
	  EXIT,
	  ADD,  
	  DEL,  
	  SEARCH,  
	  MODIFY,  
	  SHOW,  
	  SORT,  

};
//初始化通讯录函数
void InitContact(Contact* pc);
//增加联系人
void AddContact(Contact* pc);
//删除指定联系人
void DelContact(Contact* pc);
//查询联系人
void SearchContact(const Contact* pc);
//修改联系人
void ModifyContact(Contact* pc);
//显示所有联系人
void ShowContact(const Contact* pc);
//排序结构体
void SortContact(Contact* pc);

2.3 Contact.c 功能函数的定义

  ⛳️ 通讯录的大体框架我们都搭建起来了,接下来我们就是各种函数的实现。然后通讯录的整体工程就完成啦!

0️⃣ 初始化通讯录

这里没什么可注意的,唯一需要注意的一点就是:

  • assert()断言一下确保程序的可执行性
  • 和 结构体数组初始化要用 memset 函数初始化

📚 代码演示:

void InitContact(Contact* pc)
{
	assert(pc);
	memset(pc->data, 0, sizeof(pc->data));
	pc->sz = 0;
}

1️⃣ 新增联系人的实现

新增的大概思想就是根据我们的 sz 联系人的个数来做数组下标。然后进行访问存储数据。

  • 而这里要注意的是确保一下容量会不会慢一旦满了我们就提示
  • 通讯录已满,无法添加联系人。这才是最合理的思想

📚 代码演示:

void AddContact(Contact* pc)
{
	assert(pc);
	if (pc->sz == MAX)
	{
		printf("通讯录已满,无法添加\n");
		return;
	}

	printf("请输入名字:>");
	scanf("%s", pc->data[pc->sz].name);
	printf("请输入年龄:>");
	scanf("%d", &(pc->data[pc->sz].age));
	printf("请输入性别:>");
	scanf("%s", pc->data[pc->sz].sex);
	printf("请输入电话:>");
	scanf("%s", pc->data[pc->sz].tele);
	printf("请输入地址:>");
	scanf("%s", pc->data[pc->sz].addr);

	pc->sz++;
	printf("成功增加联系人\n");
}

2️⃣ 删除联系人的实现

删除联系人怎么实现呢?这里是不是得先写一个查找函数先来找到我们需要删除联系人的下标然后再进行删除。

  • 一 、这里要注意的意思如果联系人为空就无法删除
  • 二,查找函数的实现不需要声明因为我们只需要在这一个文件下用不需要跨文件使用
  • 三 、 删除完联系人后我们需要把删除的联系人的那个节点后面的联系人都给向左填充,这样我们的通讯录才又是一个有序数组!
  • 还有一个重要的点是 删除完联系人,我们的有效人数 sz 也要减一

📚 代码演示:

// 查询联系人
int FindByName(const Contact* pc, char name[])
{
	//找到要删除的人
	int i = 0;

	for (i = 0; i < pc->sz; i++)
	{
		if (strcmp(pc->data[i].name, name) == 0)
		{
			return i;
		}
	}
	return -1;
}

//删除指定联系人
void DelContact(Contact* pc)
{
	if (pc->sz == 0)
	{
		printf("通讯录为空,无法删除!\n");
		return;
	}
	char name[MAX_NAME] = { 0 };
	assert(pc);
	//删除联系人
	printf("请输入要删除人的名字:>");
	scanf("%s", name);
	//查询联系人
	int del = FindByName(pc, name);
	if (del == -1)
	{
		printf("要删除的人不存在,删除失败!\n");
		return;
	}
	int i = 0;
	for (i = del; i < pc->sz-1; i++)
	{
		pc->data[i] = pc->data[i + 1];
	}
	pc->sz--;
	printf("成功删除联系人!\n");
}

3️⃣查询联系人的实现

这个就比较简单了,前面我们已经简单的实现了一个查找函数用来查找下标。

  • 这里需注意的是由于我们只需要查找联系人而不用修改
  • 所以我们在接收只指针的时候要记得,使用 const 进行修饰
  • 来确保指针指向的内容不会被我们改变而引发程序错乱

📚 代码演示:

//查询联系人
void SearchContact(const Contact* pc)
{
	char name[MAX_NAME] = { 0 };
	printf("请输入要查找人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("要查找的人不存在!\n");
		return;
	}
	else
	{
		printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
		printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",
			pc->data[pos].name,
			pc->data[pos].age,
			pc->data[pos].sex,
			pc->data[pos].tele,
			pc->data[pos].addr);
	}

}

4️⃣ 修改联系人的实现

修改联系人我们该怎么办呢?答案肯定还是使用下标的方法进行更改了!诶这里大家有没有发现我们查询函数的便捷性,所以像这种只要多次使用的功能一定要封装成函数。

  • 这样使用起来就会方便很多

📚 代码演示:

//修改联系人
void ModifyContact(Contact* pc)
{
	char name[MAX_NAME] = { 0 };
	printf("请输入要修改人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("要修改的人不存在!\n");
		return;
	}
	else
	{
		printf("请输入名字:->");
		scanf("%s", pc->data[pos].name);
		printf("请输入年龄:->");
		scanf("%d", &(pc->data[pos].age));
		printf("请输入性别:->");
		scanf("%s", pc->data[pos].sex);
		printf("请输入电话:->");
		scanf("%s", pc->data[pos].tele);
		printf("请输入地址:->");
		scanf("%s", pc->data[pos].addr);
		printf("联系人修改成功!\n");
		return;
	}
}

5️⃣ 查看所有联系人

这里更加简单只需要使用循环来遍历我们的数组就可以了。

  • 而这里我们也是只访问并不修改,所以使用指针接收的时候
  • 一定要用 const 来修饰我们的指针确保指针指向的内容不会改变

📚 代码演示:

//显示所有联系人
void ShowContact(const Contact* pc)
{
	int i = 0;
	//打印列标题
	printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
	//打印数据
	for (i = 0; i < pc->sz; i++)
	{
		printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n", 
			pc->data[i].name, 
			pc->data[i].age, 
			pc->data[i].sex, 
			pc->data[i].tele,
			pc->data[i].addr);


	}

}

6️⃣ 排序联系人

这里就需要用到我们 库函数 qsort 函数一键排序了,十分的方便

  • 不会的可以去看一下博主前面的文章,看完秒会

🔥 注:文章链接在这里。《qsort的使用详解》

📚 代码演示:

int cmp(const void* p1, const void* p2)
{
	return  (*((PeoInfo*)p1)->name, *((PeoInfo*)p2)->name);
}

//结构体比较函数
void SortContact(Contact* pc)
{
	qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp);
	printf("已按照姓名排序成功!\n");
}

三、通讯录功能的测试

所有函数都实现了那么我们就来查看一下每个功能的是否正常呢?接下来我们就来测试一下!

1️⃣ 新增联系人的测试

  • 这里我们就可以看到新增联系人的函数实现成功了!

在这里插入图片描述

2️⃣ 删除联系人的测试

  ⛳️这里就可以看到本来我们是有3个联系人的,然后进删除选了就只剩俩个联系人了!

在这里插入图片描述

3️⃣查询联系人的测试

在这里插入图片描述

4️⃣ 修改联系人的测试

这里我们就把翠花的信息重新修改,为小美了,说明这个函数也实现了

在这里插入图片描述

5️⃣ 查看所有联系人

在这里插入图片描述

6️⃣ 排序联系人

在这里插入图片描述

四、通讯录整体工程

test.c

#define _CRT_SECURE_NO_WARNINGS 1
//实现一个通讯录
// 名字
// 年龄
// 性别
// 电话
// 地址
// 
// 通讯录的功能可以存放
//增加联系人
// 删除联系人
// 修改联系人
// 查找联系人
// 显示所有联系人
// 排序功能
//
#include <stdio.h>
#include "contact.h"

//菜单
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");
}

//测试通讯录
void test()
{
	int input = 0;
	Contact con;
	InitContact(&con);
	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);
}
int main()
{
	test();
	return 0;
 }

contact.c

#define _CRT_SECURE_NO_WARNINGS 1
#include "contact.h"


//初始化通讯录函数
void InitContact(Contact* pc)
{
	assert(pc);
	pc->data = (PeoInfo*)malloc(DEFAULT_SZ *sizeof(PeoInfo));
	if (pc->data == NULL)
	{
		perror("InitContact");
		return;
	}
	pc->sz = 0;
	pc->capacity = DEFAULT_SZ;

}

//检查容量
int  CheckCapacity(Contact* pc)
{
	if (pc->sz == pc->capacity)
	{
		PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + INC_SZ) * sizeof(PeoInfo));
		if (ptr == NULL)
		{
			perror("CheckCapacity");
			return 0;
		}
		else
		{
			pc->data = ptr;
			pc->capacity += INC_SZ;
			printf("增容成功!\n");
			return 1;
		}
		
	}
	return 1;
}
//增加联系人
void AddContact(Contact* pc)
{
	assert(pc);
	CheckCapacity(pc);

	if (0 == CheckCapacity(pc))
	{
		return ;
	}
	else
	{
		printf("请输入名字:->");
		scanf("%s", pc->data[pc->sz].name);
		printf("请输入年龄:->");
		scanf("%d", &(pc->data[pc->sz].age));
		printf("请输入性别:->");
		scanf("%s", pc->data[pc->sz].sex);
		printf("请输入电话:->");
		scanf("%s", pc->data[pc->sz].tele);
		printf("请输入地址:->");
		scanf("%s", pc->data[pc->sz].addr);

		pc->sz++;
		printf("成功增加联系人\n");
	}

}

//显示所有联系人
void ShowContact(const Contact* pc)
{
	int i = 0;
	//打印列标题
	printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
	//打印数据
	for (i = 0; i < pc->sz; i++)
	{
		printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n", 
			pc->data[i].name, 
			pc->data[i].age, 
			pc->data[i].sex, 
			pc->data[i].tele,
			pc->data[i].addr);


	}

}

// 查询联系人
int FindByName(const Contact* pc, char name[])
{
	//找到要删除的人
	int i = 0;

	for (i = 0; i < pc->sz; i++)
	{
		if (strcmp(pc->data[i].name, name) == 0)
		{
			return i;
		}
	}
	return -1;
}

//删除指定联系人
void DelContact(Contact* pc)
{
	if (pc->sz == 0)
	{
		printf("通讯录为空,无法删除!\n");
		return;
	}
	char name[MAX_NAME] = { 0 };
	assert(pc);
	//删除联系人
	printf("请输入要删除人的名字:>");
	scanf("%s", name);
	//查询联系人
	int del = FindByName(pc, name);
	if (del == -1)
	{
		printf("要删除的人不存在,删除失败!\n");
		return;
	}
	int i = 0;
	for (i = del; i < pc->sz-1; i++)
	{
		pc->data[i] = pc->data[i + 1];
	}
	pc->sz--;
	printf("成功删除联系人!\n");
}

//查询联系人
void SearchContact(const Contact* pc)
{
	char name[MAX_NAME] = { 0 };
	printf("请输入要查找人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("要查找的人不存在!\n");
		return;
	}
	else
	{
		printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
		printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",
			pc->data[pos].name,
			pc->data[pos].age,
			pc->data[pos].sex,
			pc->data[pos].tele,
			pc->data[pos].addr);
	}

}

//修改联系人
void ModifyContact(Contact* pc)
{
	char name[MAX_NAME] = { 0 };
	printf("请输入要修改人的名字:>");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1)
	{
		printf("要修改的人不存在!\n");
		return;
	}
	else
	{
		printf("请输入名字:->");
		scanf("%s", pc->data[pos].name);
		printf("请输入年龄:->");
		scanf("%d", &(pc->data[pos].age));
		printf("请输入性别:->");
		scanf("%s", pc->data[pos].sex);
		printf("请输入电话:->");
		scanf("%s", pc->data[pos].tele);
		printf("请输入地址:->");
		scanf("%s", pc->data[pos].addr);
		printf("联系人修改成功!\n");
		return;
	}
}

//
//int cmp(const void* p1, const void* p2)
//{
//	return   (((Contact*)p1)->data)->age- (((Contact*)p2)->data)->age;
//}

int cmp(const void* p1, const void* p2)
{
	return  (*((PeoInfo*)p1)->name, *((PeoInfo*)p2)->name);
}

//结构体比较函数
void SortContact(Contact* pc)
{
	qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp);
	printf("已按照姓名排序成功!\n");
}

void DestroyContact(Contact* pc)
{
	free(pc->data);
	pc->sz = 0;
	pc->capacity = 0;
}

contact.h

#pragma once
#include <string.h>
#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define MAX 1000
#define MAX_NAME  20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30

#define DEFAULT_SZ 3
#define INC_SZ 2

//联系人的类型定义
typedef struct PeoInfo
{
	char name[MAX_NAME];
	int age;
	char sex[MAX_SEX];
	char tele[MAX_TELE];
	char addr[MAX_ADDR];
}PeoInfo;

//动态通讯录
typedef struct Contact
{
	PeoInfo* data;//指向了存放数据的空间
	int sz;//记录当前存放的有效元素的个数
	int capacity;//通讯录当前的最大容量
}Contact;

//枚举选项
enum  OPTION
{
	  EXIT,
	  ADD,  
	  DEL,  
	  SEARCH,  
	  MODIFY,  
	  SHOW,  
	  SORT,  

};

//初始化通讯录函数
void InitContact(Contact* pc);
//增加联系人
void AddContact(Contact* pc);
//删除指定联系人
void DelContact(Contact* pc);
//查询联系人
void SearchContact(const Contact* pc);
//修改联系人
void ModifyContact(Contact* pc);
//显示所有联系人
void ShowContact(const Contact* pc);
//排序结构体
void SortContact(Contact* pc);
void DestroyContact(Contact* pc);

📝全篇总结

✅ 归纳:
好了以上就是用简易通讯录的实现是不是很有趣呢!本期只用到了指针和结构体的内容!下期给大家带来动态通讯录的改造!
  初始化通讯录
  新增联系人的实现
  删除联系人的实现
  查询联系人的实现
  修改联系人的实现
  查看所有联系人
  排序联系人
☁️ 好了把上面的知识全部掌握我相信各位铁汁们,对结构体和指针的应用更加得心应手了!
看到这里了还不给博主扣个:
⛳️ 点赞☀️收藏 ⭐️ 关注

💛 💙 💜 ❤️ 💚💓 💗 💕 💞 💘 💖
拜托拜托这个真的很重要!
你们的点赞就是博主更新最大的动力!
有问题可以评论或者私信呢秒回哦。
在这里插入图片描述

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

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

相关文章

Android平台如何实时叠加电量信息和设备信号状态到GB28181接入端

技术背景 我们在Android平台实现GB28181设备接入&#xff0c;把摄像头和麦克风数据&#xff0c;采集过去&#xff0c;用于移动单兵、智能车载、智慧安防、智能家居、工业仿真等行业时&#xff0c;发现大多场景对视频水印的要求越来越高&#xff0c;从之前的固定位置静态文字水…

Python(六十)字典的创建

❤️ 专栏简介&#xff1a;本专栏记录了我个人从零开始学习Python编程的过程。在这个专栏中&#xff0c;我将分享我在学习Python的过程中的学习笔记、学习路线以及各个知识点。 ☀️ 专栏适用人群 &#xff1a;本专栏适用于希望学习Python编程的初学者和有一定编程基础的人。无…

Kubernetes——理论基础

Kubernetes——理论基础 一、Kubernetes 概述1.K8S 是什么&#xff1f;2.为什么要用 K8S?3.Kubernetes 主要功能 二、Kubernetes 集群架构与组件三、Master 组件1.Kube-apiserver2.Kube-controller-manager3.Kube-scheduler4.配置存储中心——etcd 四、Node 组件1.Kubelet2.Ku…

城市供水管网水力模型的基本概念及理论

1.1引言 城市供水管网系统由大量管材各异、管径各异、铺设年代各异的管道&#xff0c;泵站&#xff0c;阀门&#xff0c; 水塔等多元素构成&#xff0c;因此决定了供水管网系统是一个拓扑结构庞杂、运行工况多变的巨系统。以前国内供水公司对铺设在地面以下的供水管网多以经验…

HCIP的mgre实验

题目 拓扑图 IP地址配置和缺省 R1 [r1]int g0/0/1 [r1-GigabitEthernet0/0/1]ip add 192.168.1.1 24 Aug 2 2023 20:38:20-08:00 r1 %%01IFNET/4/LINK_STATE(l)[0]:The line protocol IP on the interface GigabitEthernet0/0/1 has entered the UP state. [r1-GigabitEtherne…

【雕爷学编程】Arduino动手做(181)---Maixduino AI开发板4

37款传感器与执行器的提法&#xff0c;在网络上广泛流传&#xff0c;其实Arduino能够兼容的传感器模块肯定是不止这37种的。鉴于本人手头积累了一些传感器和执行器模块&#xff0c;依照实践出真知&#xff08;一定要动手做&#xff09;的理念&#xff0c;以学习和交流为目的&am…

云环境中使用飞蛾火焰和萨尔普群算法组合的工作流调度(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

AcWing 202. 最幸运的数字

AcWing 202. 最幸运的数字 思路&#xff1a; Code: #include<bits/stdc.h> using namespace std; typedef long long LL; int gcd(LL n,int m) {return m?gcd(m,n%m):n; } LL get_euler(LL x) { //求欧拉函数LL resx;for(int i2;i<x/i;i) {if(x%i0) {while(x%i0)…

经典CNN(三):DenseNet算法实战与解析

&#x1f368; 本文为&#x1f517;365天深度学习训练营中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊|接辅导、项目定制 1 前言 在计算机视觉领域&#xff0c;卷积神经网络&#xff08;CNN&#xff09;已经成为最主流的方法&#xff0c;比如GoogleNet&#xff0c;…

结算功能实现(小兔鲜儿)【Vue3】

退出登录 - 清空购物车列表 业务需求 在用户退出登录时,除了清除用户信息之外,也需要把购物车数据清空 // 清除购物车const clearCart () > {cartList.value []}// 退出时清除用户信息const clearUserInfo () > {userInfo.value {}// 执行清除购物车的actioncartS…

2023年攻防演练利器之必修高危漏洞合集(包含详细修复建议)

2023年攻防演练利器之必修高危漏洞合集&#xff08;包含详细修复建议&#xff09; 下载地址见盘&#xff1a; 链接&#xff1a;https://pan.baidu.com/s/1WeCC7oXFieDs4tBOh7bmWA?pwd12u4 提取码&#xff1a;12u4

PyTorch 中的累积梯度

https://stackoverflow.com/questions/62067400/understanding-accumulated-gradients-in-pytorch 有一个小的计算图&#xff0c;两次前向梯度累积的结果&#xff0c;可以看到梯度是严格相等的。 代码&#xff1a; import numpy as np import torchclass ExampleLinear(torch…

MongoDB文档--基本安装-linux安装(mongodb环境搭建)-docker安装(挂载数据卷)-以及详细版本对比

阿丹&#xff1a; 前面了解了mongodb的一些基本概念。本节文章对安装mongodb进行讲解以及汇总。 官网教程如下&#xff1a; 安装 MongoDB - MongoDB-CN-Manual 版本特性 下面是各个版本的选择请在安装以及选择版本的时候参考一下&#xff1a; MongoDB 2.x 版本&#xff1a…

TensorRT学习笔记--基于YoloV8检测图片和视频

1--完整项目 完整项目地址&#xff1a;https://github.com/liujf69/TensorRT-Demo git clone https://github.com/liujf69/TensorRT-Demo.gitcd TRT_YoloV8 2--模型转换 cd yolov8python gen_wts.py 3--编译项目 mkdir buildcd build cmake .. # 需要更改 CMakeLists.txt…

postgresSQL Extended Query执行过程和sharding-proxy的处理

pg Extended Query PostgreSQL: Documentation: 15: 55.2. Message Flow 多个阶段&#xff0c;可复用 Parse → DESCRIBE statement → SYNC Parse 解析&#xff0c; 将 sql 文本字符串&#xff0c;解析成 named preparedStatement 语句&#xff08;生命周期随session&#x…

数据安全能力框架模型-详细解读(一)

8月30日&#xff0c;奇安信集团正式发布“数据安全能力框架”&#xff0c;以及“数据安全概念运行图”&#xff08;数据安全ConOps&#xff09;&#xff0c;旨在为数字化转型不断深入的大型政企客户以及业内伙伴&#xff0c;提供基于甲方视角的数据安全全面图景&#xff0c;以及…

GESP2023年6月C++一级客观题

一、单选题&#xff08;每题 2 分&#xff0c;共 30 分&#xff09; 以下不属于计算机输出设备的有&#xff08; &#xff09;。 A. 麦克风 B. 音箱 C. 打印机 D. 显示器 ChatGPT 是 OpenAI 研发的聊天机器人程序&#xff0c;它能通过理解和学习人类的语言 来进行对话&#xf…

ES6之Promise、Class类与模块化(Modules)

目录 PromiseClass类extendssuper Modules 模块系统export default 和对应importexport 和 import Promise Promise 是 ES6 引入的一种用于处理异步操作的对象。 它解决了传统回调函数&#xff08;callback&#xff09;模式中容易出现的回调地狱和代码可读性差的问题。 Promis…

Vue.js2+Cesium 四、WMS 服务加载,控制自图层显隐

Vue.js2Cesium 四、WMS 服务加载&#xff0c;控制自图层显隐 Demo <template><divid"cesium-container"style"width: 100%; height: 100%;"><div class"layer_container"><button id"btn">清除</button&g…

大模型开发(十六):从0到1构建一个高度自动化的AI项目开发流程(中)

全文共1w余字&#xff0c;预计阅读时间约40~60分钟 | 满满干货(附代码)&#xff0c;建议收藏&#xff01; 本文目标&#xff1a;通过LtM提示流程实现自动构建符合要求的函数&#xff0c;并通过实验逐步完整测试code_generate函数功能。 代码下载点这里 一、介绍 此篇文章为…