【C语言进阶篇】模拟实现通讯录 (内附源码)


在这里插入图片描述

🎬 鸽芷咕:个人主页

 🔥 个人专栏:《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.mzph.cn/news/24433.shtml

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

相关文章

计算机网络的定义和分类

计算机网络的定义和分类 计算机网络的定义 计算机网络的精确定义并未统一计算机网络最简单的定义是&#xff1a;一些互相连接的、自治的计算机的集合 互连:指计算机之间可以通过有线或无线的方式进行数据通信自治:是指独立的计算机&#xff0c;它有自己的硬件和软件&#xff…

Python语法:... for ... in ... if ...

Python中&#xff0c;for...in...[if]...语句是一种简洁的构建List的方法&#xff0c;从for给定的List中选择出满足if条件的元素组成新的List&#xff0c;其中if是可以省略的。下面举几个简单的例子进行说明 [for in ]: ...for ....in..... 语句. 实例如下&#xff1a; (1) …

PHP实现首字母头像

<?php $name"哈哈"; $logoletter_avatar($name);echo <img src".$logo." style" border-radius: 50%;">;function letter_avatar($text) {$total unpack(L, hash(adler32, $text, true))[1];$hue $total % 360;list($r, $g, $b) hs…

【go-zero】docker镜像直接部署API与RPC服务 如何实现注册发现?docker network 实现 go-zero 注册发现

一、场景&问题 使用docker直接部署go-zero微服务会发现API无法找到RPC服务 1、API无法发现RPC服务 用docker直接部署 我们会发现API无法注册发现RPC服务 原因是我们缺少了docker的network网桥 2、系统内查看 RPC服务运行正常API服务启动,通过docker logs 查看日志还是未…

Linux学习之正则表达式元字符和grep命令

cat /etc/redhat-release看到操作系统的版本是CentOS Linux release 7.6.1810 (Core)&#xff0c;uname -r可以看到内核版本是3.10.0-957.21.3.el7.x86_64。 正则表达式是一种搜索字符串的模式&#xff0c;通俗点理解&#xff0c;也就是普通字符和元字符共同组成的字符集合匹…

模板方法模式——定义算法的框架

1、简介 1.1、概述 模板方法模式是结构最简单的行为型设计模式&#xff0c;在其结构中只存在父类与子类之间的继承关系。通过使用模板方法模式&#xff0c;可以将一些复杂流程的实现步骤封装在一系列基本方法中。在抽象父类中提供一个称之为模板方法的方法来定义这些基本方法…

vscode插件不能搜索安装

1 现象 vscode搜索自己的插件&#xff0c;报错&#xff1a; Error while fetching extensions. HXR failed2 原因 之前用vscode开发golang语言&#xff0c;设置了proxy代理&#xff0c;所以导致错误&#xff0c;删除即可 重启vscode 3 结果

SSM(Vue3+ElementPlus+Axios+SSM前后端分离)--功能实现【四】

文章目录 SSM--功能实现实现功能06-修改家居信息需求分析/图解思路分析代码实现注意事项和细节 实现功能07-删除家居信息需求分析/图解思路分析代码实现 实现功能08-分页显示列表需求分析/图解思路分析代码实现完成测试分页显示效果 SSM–功能实现 实现功能06-修改家居信息 需…

git之reflog分析

写在前面 本文一起看下reflog命令。 1&#xff1a;场景描述 在开发的过程中&#xff0c;因为修改错误&#xff0c;想要通过git reset命令恢复到之前的某个版本&#xff0c;但是选择提交ID错误&#xff0c;导致多恢复了一个版本&#xff0c;假定&#xff0c;该版本对应的内容…

macOS 环境变量加载探究

使用 macOS 安装环境&#xff0c;见到过很数种环境变量配置方法&#xff0c;每次也都是按照别人的代码&#xff0c;人家配置在哪 我就配置在哪&#xff0c;其实不太清楚有什么区别&#xff0c;决定记录下。 本机 macOS 13.3&#xff0c;从 macOS Catalina(10.15) 开始&#xf…

用html+javascript打造公文一键排版系统15:一键删除所有空格

现在我们来实现一键删除所有空格的功能。 一、使用原有的代码来实现&#xff0c;测试效果并不理想 在这之前我们已经为String对象编写了一个使用正则表达式来删除所有空格的方法&#xff1a; //功能&#xff1a;删除字符串中的所有空格 //记录&#xff1a;20230726创建 Stri…

腾讯云-宝塔添加MySQL数据库

1. 数据库菜单 2. 添加数据库 3. 数据库添加成功 4. 上传数据库文件 5. 导入数据库文件 6. 开启数据库权限 7. 添加安全组 (宝塔/腾讯云) 8. Navicat 连接成功

python 封装sql 增删改查连接MySQL

select * from Teacher limit 10 连接字符串配置MysqlConfig.py class MysqlConfig:HOST 192.168.56.210PORT 3306USER rootPASSWORD 1qaz0987654321DBStudentDBCHARSET utf8封装增删改查MysqlConnection.py Author: tkhywang 2810248865qq.com Date: 2023-06-19 15:44:48 Las…

rest api client code generator

一、搜索&#xff1a;REST API Client Code Generator 二、 安装成功后 配置java环境和node环境

线性代数 | 机器学习数学基础

前言 线性代数&#xff08;linear algebra&#xff09;是关于向量空间和线性映射的一个数学分支。它包括对线、面和子空间的研究&#xff0c;同时也涉及到所有的向量空间的一般性质。 本文主要介绍机器学习中所用到的线性代数核心基础概念&#xff0c;供读者学习阶段查漏补缺…

743. 网络延迟时间

有 n 个网络节点&#xff0c;标记为 1 到 n。 给你一个列表 times&#xff0c;表示信号经过 有向 边的传递时间。 times[i] (ui, vi, wi)&#xff0c;其中 ui 是源节点&#xff0c;vi 是目标节点&#xff0c; wi 是一个信号从源节点传递到目标节点的时间。 现在&#xff0c;…

JNI之Java实现蓝牙交互

蓝牙概述 蓝牙&#xff0c;是一种支持设备短距离通信&#xff08;一般10m内&#xff09;的无线电技术&#xff0c;能在包括移动电话、PDA、无线耳机、笔记本电脑、相关外设等众多设备之间&#xff0c;通过蓝牙设备之间的无线通信实现数据传输&#xff0c;实现数据传输&#xf…

【计算机视觉|语音分离】期望在嘈杂环境中聆听:一个用于语音分离的不依赖于讲话者的“音频-视觉模型”

本系列博文为深度学习/计算机视觉论文笔记&#xff0c;转载请注明出处 标题&#xff1a;Looking to Listen at the Cocktail Party: A Speaker-Independent Audio-Visual Model for Speech Separation 链接&#xff1a;Looking to listen at the cocktail party: a speaker-in…

【网络工程】网络流量分析工具 Wireshark

文章目录 第一章&#xff1a;WireShark介绍第二章&#xff1a;WireShark应用第三章&#xff1a;Wireshark 实战 第一章&#xff1a;WireShark介绍 Wireshark (前身 Ethereal)&#xff1a;它是一个强大的网络包分析工具 ! 此工具主要是用来捕获网络数据包的&#xff0c;并且自动…

zookeeper --- 基础篇

一、zookeeper简介 1.1、什么是zookeeper zookeeper官网&#xff1a;https://zookeeper.apache.org/ 大数据生态系统里的很多组件的命名都是某种动物或者昆虫&#xff0c;他是用来管 Hadoop&#xff08;大象&#xff09;、Hive(蜜蜂)、Pig(小 猪)的管理员。顾名思义就是管理…