【C语言】通讯录1.0 (静态版)

news2024/9/27 7:18:31

前言

通讯录是一种记录联系人信息的工具,包括姓名、电话号码、电子邮件地址、住址等。
此通讯录是基于自定义类型的基础上进行制作,通讯录(静态版),后期会进行通讯录的更新
****** 有需要源代码,见文章末尾 ******


系列文章目录

第一篇:【C语言】通讯录1.0 (静态版)
第二篇:【C语言】通讯录2.0 (动态增长版)
第三篇:【C语言】通讯录2.0 (文件存储版)


文章目录

  • 前言
  • 系列文章目录
  • 一、什么是通讯录
  • 二、静态版、动态增长版和文件存储版的区别
    • 1. 静态版
    • 2. 动态增长版
    • 3. 文件存储版
  • 三、通讯录模块组成(图文)
    • 1. 通讯录文件构成
    • 2. 通讯录个人信息
    • 3. 通讯录功能模块
  • 四、逻辑测试文件(主函数)
    • 1. 选择菜单界面
    • 2. 选择功能实现
    • 3. 结构体建立联个人信息
    • 4. 结构体建立一个通讯录
    • 5. 初始化通讯录
  • 五、功能函数实现
    • 1. 添加
    • 2. 删除
    • 3. 查找
    • 4. 修改
    • 5. 显示(所有信息)
  • 五、所有文件代码
    • 1. 头文件代码
    • 2. 函数代码
    • 3. 测试逻辑代码
  • 总结


一、什么是通讯录

通讯录是一种记录联系人信息的工具,包括姓名、电话号码、电子邮件地址、住址等。通讯录可以帮助人们管理自己的联系人,让人们更轻松地与他人保持联系。通讯录可以在手机、电脑、笔记本等设备上保存,也可以在云端储存和同步,方便用户随时查看和更新联系人信息。

二、静态版、动态增长版和文件存储版的区别

C语言静态版、动态增长版和文件存储版的区别如下:

1. 静态版

  1. 静态版:在程序编译时就确定了内存大小,程序运行期间内存大小不会发生变化,因此对于需要处理大量数据或者不确定数据大小的情况不适用。

2. 动态增长版

  1. 动态增长版:可以在程序运行期间根据需要动态增加内存大小,因此适用于处理不确定数据大小的情况。但是动态增长的内存需要手动释放,否则会导致内存泄漏。

3. 文件存储版

  1. 文件存储版:将数据存储在文件中,可以持久保存数据并随时读取。但是存储在文件中的数据需要进行IO操作,因此相比于内存操作来说效率较低。此外,文件存储版不适用于需要频繁修改的数据。

三、通讯录模块组成(图文)

1. 通讯录文件构成

在这里插入图片描述

2. 通讯录个人信息

在这里插入图片描述

3. 通讯录功能模块

在这里插入图片描述

四、逻辑测试文件(主函数)

1. 选择菜单界面

进入通讯录出现选择菜单界面
菜单效果:
在这里插入图片描述

代码如下:
建立一个空类型函数,进行简单的打印即可

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");

}

2. 选择功能实现

进入菜单后,选择所需要的功能进入,用switch语句实现,用枚举类型定义[0-6] 7个数字方便后期编码方便

枚举定义数字代码:
0 EXIT,1 ADD,2 DEL,3 SEARCH,4 MODIFY,5 SHOW,6 SORT 用字母代表各case选项。

enum OPTION
{
	EXIT,//0
	ADD,
	DEL,
	SEARCH,
	MODIFY,
	SHOW,
	SORT
};

选择代码:
input等于0的时候,do…while语句,while(input)为假,跳出循环

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:
			printf("功能待开发\n");
			break;
		case EXIT:
			printf("成功退出通讯录\n");
			break;
		default:
			printf("输入错误,请重新输入\n");
			break;
		}
	} while (input);
	
}

3. 结构体建立联个人信息

这里使用结构体建立一个联系人所需要的元素信息,用typedef改名成PeoInfo方便后期编写代码
代码如下:

#define MAX 100
#define NAME 10
#define SEX  5
#define TELE 12
#define ADDR 30
typedef struct PeoInfo
{
	char name[NAME];
	int age;
	char sex[SEX];
	char tele[TELE];
	char addr[ADDR];
}PeoInfo;

4. 结构体建立一个通讯录

PeoInfo data[MAX]代表通讯录可以存放联系人的数量

int sz表示通讯录现在已经存入的人数数量

typedef改名成Contact方便后期编写代码

代码如下:

//建立通讯录

typedef struct Contact
{
	PeoInfo data[MAX]; //通讯录数量
	int sz;				//目前通讯录内的人数	
}Contact;

5. 初始化通讯录

建立一个通讯录为con

InitContact函数中进行初始化,

使用内存函数对结构体各元素进行初始化

代码如下:

逻辑测试文件

Contact con;
InitContact(&con);

头文件声明

//初始化通讯录
void InitContact(Contact* pc);

函数实现

void InitContact(Contact* pc)
{
	assert(pc);   //断言
	memset(pc->data, 0, sizeof(pc->data));   //内存函数  data初始化为0  
	pc->sz = 0;
}

五、功能函数实现

1. 添加

第一步,先assert()进行断言判断pc指针
第二步,判断通讯录是否为空,为空返回
第三步,如果不为空,修改pc所指向的data
第四步,data[pc->sz].修改的元素,然后sz++,代表通讯录联系人加1

代码如下
头文件:

//增加联系人
void AddContact(Contact* pc);

函数实现:

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++;    //通讯录加1
	printf("联系人增加成功\n");
}

2. 删除

第一步,删除,需要找到相对于的名字
第二步,编写一个查找名字的函数,用来进行查找,查找到后,返回对应的下标
第三步,编写一个for循环,将返回下标后的所有元素进行向前移动
代码如下
头文件:

//删除指定联系人
void DelContact(Contact* pc);

函数实现:

static int Findname(const Contact*pc,char na[])
{
	int i = 0;
	assert(pc && na);
	for (i = 0; i < pc->sz; i++)
	{
		if (strcmp(pc->data[i].name, na) == 0)

		{
			return i;
		}

	}
	return -1;
}
void DelContact(Contact* pc)
{
	if (pc->sz == 0)
	{
		printf("通讯录为空\n");

		return;
	}
	char name[NAME] = { 0 };
	assert(pc);
	//输入要查找的联系人名字
	printf("请输入要查找的名字:>");
	scanf("%s", &name);
	//找到要查找的联系人
	int del = Findname(pc, name);
	//删除坐标位子的联系人 ,将后面的联系人进行代替其位置
	if (del == -1)
	{
		printf("找不到,此人不存在\n");
		return;
	}
	else
	{
		int i = 0;
		for (i = del; i < pc->sz; i++)
		{
			pc->data[i] = pc->data[i + 1];
		}
		pc->sz--;
	}
	
	printf("成功删除联系人\n");
}

3. 查找

第一步,使用查找函数进行查找,找到后返回下标
第二步,进行该下标的各元素打印
代码如下
头文件:

//查找指定联系人
void SearchContact(const Contact* pc);

函数实现:

void SearchContact(const Contact* pc)
{
	if (pc->sz == 0)
	{
		printf("通讯录为空\n");

		return;
	}
	char name[NAME] = { 0 };
	assert(pc);
	//输入要查找的联系人名字
	printf("请输入要查找的名字:>");
	scanf("%s", &name);
	//找到要查找的联系人
	int i = Findname(pc, name);
	if (i == -1)
	{
		printf("找不到,此人不存在\n");
		return;
	}
	else
	{
		printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
		printf("%-10s\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);
	}
	
	printf("成功找到联系人\n");
}

4. 修改

第一步,使用上面的查找函数,进行查找需要修改人的名字,返回下标
第二步,对返回的下标内元素,进行修改
代码如下
头文件:

//修改指定联系人
void ModifyContact(Contact* pc);

函数实现:

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

5. 显示(所有信息)

第一步,打印所有元素,编写一个for循环,打印sz所有的个数
代码如下
头文件:

//显示所有联系人的信息
void ShowContact(const Contact* pc);

函数实现:

void ShowContact(const Contact* pc)
{
	assert(pc);
	printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
	int i = 0;
	for (i = 0; i < pc->sz; i++)
	{
		printf("%-10s\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);
	}
	printf("通讯录展示完毕\n");
}

五、所有文件代码

1. 头文件代码

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

#define MAX 100
#define NAME 10
#define SEX  5
#define TELE 12
#define ADDR 30
//使用枚举  定义选择   
enum OPTION
{
	EXIT,//0
	ADD,
	DEL,
	SEARCH,
	MODIFY,
	SHOW,
	SORT
};

//个人信息类型声明
typedef struct PeoInfo
{
	char name[NAME];
	int age;
	char sex[SEX];
	char tele[TELE];
	char addr[ADDR];
}PeoInfo;

//建立通讯录

typedef struct Contact
{
	PeoInfo data[MAX]; //通讯录数量
	int sz;				//目前通讯录内的人数	
}Contact;

//函数声明

//初始化通讯录
void InitContact(Contact* pc);

//增加联系人
void AddContact(Contact* pc);

//显示所有联系人的信息
void ShowContact(const Contact* pc);

//删除指定联系人
void DelContact(Contact* pc);

//查找指定联系人
void SearchContact(const Contact* pc);

//修改指定联系人
void ModifyContact(Contact* pc);

2. 函数代码

#define _CRT_SECURE_NO_WARNINGS 1
#include"addbook.h"
void InitContact(Contact* pc)
{
	assert(pc);   //断言
	memset(pc->data, 0, sizeof(pc->data));   //内存函数  data初始化为0  
	pc->sz = 0;
}


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++;    //通讯录加1
	printf("联系人增加成功\n");
}
//搜索名字找通讯录函数
static int Findname(const Contact*pc,char na[])
{
	int i = 0;
	assert(pc && na);
	for (i = 0; i < pc->sz; i++)
	{
		if (strcmp(pc->data[i].name, na) == 0)

		{
			return i;
		}

	}
	return -1;
}
void DelContact(Contact* pc)
{
	if (pc->sz == 0)
	{
		printf("通讯录为空\n");

		return;
	}
	char name[NAME] = { 0 };
	assert(pc);
	//输入要查找的联系人名字
	printf("请输入要查找的名字:>");
	scanf("%s", &name);
	//找到要查找的联系人
	int del = Findname(pc, name);
	//删除坐标位子的联系人 ,将后面的联系人进行代替其位置
	if (del == -1)
	{
		printf("找不到,此人不存在\n");
		return;
	}
	else
	{
		int i = 0;
		for (i = del; i < pc->sz; i++)
		{
			pc->data[i] = pc->data[i + 1];
		}
		pc->sz--;
	}
	
	printf("成功删除联系人\n");
}
void ShowContact(const Contact* pc)
{
	assert(pc);
	printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
	int i = 0;
	for (i = 0; i < pc->sz; i++)
	{
		printf("%-10s\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);
	}
	printf("通讯录展示完毕\n");
}
void SearchContact(const Contact* pc)
{
	if (pc->sz == 0)
	{
		printf("通讯录为空\n");

		return;
	}
	char name[NAME] = { 0 };
	assert(pc);
	//输入要查找的联系人名字
	printf("请输入要查找的名字:>");
	scanf("%s", &name);
	//找到要查找的联系人
	int i = Findname(pc, name);
	if (i == -1)
	{
		printf("找不到,此人不存在\n");
		return;
	}
	else
	{
		printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");
		printf("%-10s\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);
	}
	
	printf("成功找到联系人\n");
}
void ModifyContact(Contact* pc)
{
	assert(pc);
	char name[NAME] = { 0 };
	printf("请输入要修改人的名字:>");
	scanf("%s", &name);
	int mod = Findname(pc, name);
	if (mod == -1)
	{
		printf("找不到,不存在\n");
		return;
	}
	else
	{
		printf("请输入姓名:>");
		scanf("%s", pc->data[mod].name);
		printf("请输入年龄:>");
		scanf("%d", &(pc->data[mod].age));
		printf("请输入性别:>");
		scanf("%s", pc->data[mod].sex);
		printf("请输入电话:>");
		scanf("%s", pc->data[mod].tele);
		printf("请输入地址:>");
		scanf("%s", pc->data[mod].addr);
		   
		printf("联系人修改成功\n");
	}
}

3. 测试逻辑代码

#define _CRT_SECURE_NO_WARNINGS 1
#include"addbook.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:
			printf("功能待开发\n");
			break;
		case EXIT:
			printf("成功退出通讯录\n");
			break;
		default:
			printf("输入错误,请重新输入\n");
			break;
		}
	} while (input);
	
}
int main()
{
	test();
	return 0;
}

总结

本期博客,是通讯录1.0(静态版),是对前面所学知识进行复习,编写通讯录时有助于理解自定义类型的学习和了解,后期会对现在的通讯录进行更新!!!


如这篇博客对大家有帮助的话,希望 三连 支持一下 !!! 如果有错误感谢大佬的斧正 如有 其他见解发到评论区,一起学习 一起进步。

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

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

相关文章

大于号在python中怎么打,python大于等于怎么写

大家好&#xff0c;小编为大家解答python中大于并小于一个数代码的问题。很多人还不知道python中大于等于且小于等于&#xff0c;现在让我们一起来看看吧&#xff01; 1、python 中不等于怎么表示 #!/usr/bin/python a1 b2 if ab: print "a 等于 b" if a!b: print &…

ArcGIS Pro 制作一张立体地形图

在各位关掉文章之前,先把成果贴上来 下面开始操作步骤贴图,这个真的很简单,没有什么复杂的软件联动和操作 这是哥斯达黎加部分区域的30mDEM,数据链接我放在最后。 首先,找到工具【栅格函数】—【统计分析】,选择下载好的栅格,领域设置行列数都改为6,点击创建新图层。然…

从官网认识 JDK,JRE,JVM 三者的关系

点击下方关注我&#xff0c;然后右上角点击...“设为星标”&#xff0c;就能第一时间收到更新推送啦~~~ JVM 是一些大厂面试必问点&#xff0c;要想解决 OOM、性能调优方面的问题&#xff0c;掌握 JVM 知识必不可少&#xff0c;从今天开始&#xff0c;将为大家介绍 JVM 的常用知…

ShardingSphere Pipeline 兼容 MySQL 时间类型 解读

背景 ShardingSphere 在日常中&#xff0c;开发者经常会用到时间类型&#xff0c;如果熟悉的时间类型涉及到了时区、精度&#xff0c;哪种时间类型更合适&#xff1f;JDBC 驱动中又有哪些注意事项&#xff1f;因此&#xff0c;对数据库时间类型有更深入的了解&#xff0c;有助于…

android framework车载桌面CarLauncher的TaskView详细源码分析

1、构建相关的TaskView&#xff0c;装载到对应的ViewGroup b站免费视频教程讲解&#xff1a; https://www.bilibili.com/video/BV1wj411o7A9/ //packages/apps/Car/Launcher/src/com/android/car/carlauncher/CarLauncher.java void onCreate() { //ignoresetContentView(R.…

管理类联考——写作——记忆篇——论证有效性分析——析错口诀

析错口诀 概念不明确&#xff0c;我就说它概念模糊&#xff0c;并做不利它的解释。【有概念模糊之嫌&#xff0c;A是理解1&#xff1f;还是理解2&#xff1f;】概念有变换&#xff0c;我就说它混淆概念&#xff0c;并指出混淆的环节。&#xff08;概念推概念&#xff0c;我就看…

类方法(成员方法)与构造方法的区别与联系

&#xff08;粗略理解为&#xff1a;除了构造方法以外的所有方法都是成员方法&#xff09; 区别&#xff1a; 构造方法在创建对象时用new调用&#xff0c;是为了给对象的数据进行初始化&#xff0c;没有返回值。 成员方法是实现对类中成员变量的操作&#xff0c;提供某些功能…

【图论】kruskal算法

一.介绍 Kruskal&#xff08;克鲁斯卡尔&#xff09;算法是一种用于解决最小生成树问题的贪心算法。最小生成树是指在一个连通无向图中&#xff0c;选择一棵包含所有顶点且边权重之和最小的树。 下面是Kruskal算法的基本步骤&#xff1a; 将图中的所有边按照权重从小到大进行…

JUC并发工具类

一、ReentrantLock 特点&#xff1a;独占、可重入、公平/非公平、可中断、支持多个条件变量 1、常用api ReentrantLock实现了Lock接口&#xff0c;Lock类规范定义了如下方法 lock()&#xff1a;获取锁&#xff0c;调用该方法的线程会获取锁&#xff0c;当锁获得后&#xff0…

想做上位机,学C#还是QT?

学习C#还是Qt&#xff0c;取决于你的具体需求和偏好。 如果你计划开发跨平台的桌面应用程序&#xff0c;并且希望使用一种更轻量级、直观的界面框架&#xff0c;那么Qt可能是一个不错的选择。Qt是一个功能丰富且成熟的跨平台框架&#xff0c;支持多种开发语言&#xff08;包括…

第J2周:ResNet50V2算法实战与解析

&#x1f368; 本文为&#x1f517;365天深度学习训练营 中的学习记录博客&#x1f366; 参考文章&#xff1a;365天深度学习训练营-第J2周&#xff1a;ResNet50V2算法实战与解析&#x1f356; 原作者&#xff1a;K同学啊|接辅导、项目定制 目录 一、论文解读1. ResNetV2结构与…

【需求响应】一种新的需求响应机制DR-VCG研究

目录 1 主要内容 2 部分代码 3 程序结果 4 程序链接 1 主要内容 该程序对应文章《Contract Design for Energy Demand Response》&#xff0c;电力系统需求响应&#xff08;DR&#xff09;用来调节用户对电能的需求&#xff0c;即在预测的需求高于电能供应时&#xff0c;希…

VS Code调试Darknet

一、安装插件 二、连接服务器 三、调试darknet工程 {"version": "2.0.0","options": {"cwd": "${workspaceFolder}"},"tasks": [{"label": "clean","type": "shell",&qu…

数据结构之动态顺序表(附带完整程序)

&#x1f388;基本概念 &#x1f308;一.线性表、顺序表的定义 ☀️&#xff08;1&#xff09;线性表&#xff1a; 是n个具有相同特性的数据元素的有限序列。线性表在逻辑上是线性结构&#xff0c;但在物理上存储时&#xff0c;通常以数组和链式结构的形式存储。 ☀️&…

C# 关于使用newlife包将webapi接口寄宿于一个控制台程序、winform程序、wpf程序运行

C# 关于使用newlife包将webapi接口寄宿于一个控制台程序、winform程序、wpf程序运行 安装newlife包 Program的Main()函数源码 using ConsoleApp3; using NewLife.Log;var server new NewLife.Http.HttpServer {Port 8080,Log XTrace.Log,SessionLog XTrace.Log }; serv…

hcip——ospf综合

要求 1. 搭建toop 2.地址规划 协议范围路由器地址 RIP 172.16.0.0 17 R12 loop0&#xff1a;172.16.0.0 24 loop1&#xff1a;172.16.1.0 24 OSPF 172.16.128.0 17 area1 172.16.144.0 20 R1 g0:172.16.144.1 24 loop0:172.16.145.1 24 R2 g0:172.16.144.2 24 loop:172…

iOS - Apple开发者账户添加新测试设备

获取UUID 首先将设备连接XCode&#xff0c;打开Window -> Devices and Simulators&#xff0c;通过下方位置查看 之后登录(苹果开发者网站)[https://developer.apple.com/account/] &#xff0c;点击设备 点击加号添加新设备 填写信息之后点击Continue&#xff0c;并一路继续…

Golang Devops项目开发(1)

1.1 GO语言基础 1 初识Go语言 1.1.1 开发环境搭建 参考文档&#xff1a;《Windows Go语言环境搭建》 1.2.1 Go语言特性-垃圾回收 a. 内存自动回收&#xff0c;再也不需要开发人员管理内存 b. 开发人员专注业务实现&#xff0c;降低了心智负担 c. 只需要new分配内存&#xff0c;…

Android系统服务之AMS

目录 概述 重点和难点问题 启动方式 main入口&#xff1a; run方法&#xff1a; BootstrapSevices 小结&#xff1a; 与其他线程的通信原理 参考文档&#xff1a; 概述 AMS是Android系统主要负责四大组件的启动&#xff0c;切换&#xff0c;调度以及应用程序进程管理和调度等工…

watch避坑,使用computed进行处理数据

业务场景&#xff1a;在vue中监听el-input 中的字数有没有超过60&#xff0c;如果超过60字时将60后面的字变为 “>>” 符号&#xff0c;以此实现预览苹果手机推送摘要场景。 错误&#xff1a;开始的逻辑是使用watch监听&#xff0c;检查length超过60直接 加上符号&#x…