通讯录(纯C语言实现)

相信大家都有过通讯录,今天我来带大家实现以下最简单的通讯录,通过本篇文章,相信可以让大家对C语言有进一步的认识。

话不多说,我们先放函数的实现

#define  _CRT_SECURE_NO_WARNINGS 1
#include "Contact.h"


int CheakCapacity(Contact* ps);

void LoadContact(Contact* ps)
{
	FILE* pf = fopen("Contact.dat", "rb");
	if (pf == NULL)
	{
		perror("LoadContact");
		return;
	}
	PeoInfo tmp = { 0 };
	while (fread(&tmp,sizeof(PeoInfo),1,pf))
	{
		CheakCapacity(ps);
		ps->date[ps->size] = tmp;
		ps->size++;
	}

	fclose(pf);
	pf = NULL;

	
}

void InitContact(Contact* ps)
{
	assert(ps);
	ps->date = NULL;
	ps->size = ps->capacity = 0;
	LoadContact(ps);
}

int CheakCapacity(Contact* ps)
{
	assert(ps);
	if (ps->capacity == ps->size)
	{
		int newcapacity = ps->capacity == 0 ? 4 : ps->capacity * 2;
		PeoInfo* tmp = (PeoInfo*)realloc(ps->date, newcapacity * sizeof(PeoInfo));
		if (tmp == NULL)
		{
			perror("CheakCapacity");
			return 0;
		}
		else
		{
			ps->date = tmp;
			ps->capacity = newcapacity;
			printf("增容成功\n");
			return 1;
		}
	}
	return 1;
}

void AddContact(Contact* ps)
{
	assert(ps);
	if (CheakCapacity(ps) == 0)
	{
		return;
	}

	printf("请输入增加的姓名:>");
	scanf("%s", ps->date[ps->size].name);
	printf("请输入增加的年龄:>");
	scanf("%d", &ps->date[ps->size].age);
	printf("请输入增加的性别:>");
	scanf("%s", ps->date[ps->size].sex);
	printf("请输入增加的电话:>");
	scanf("%s", ps->date[ps->size].tele);
	printf("请输入增加的地址:>");
	scanf("%s", ps->date[ps->size].addr);
	
	ps->size++;
	printf("增加成功\n");
}

void ShowContact(Contact* ps)
{
	assert(ps);

	printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");

	for (int i =0; i<ps->size; i++)
	{
		printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
			ps->date[i].name,
			ps->date[i].age,
			ps->date[i].sex,
			ps->date[i].tele,
			ps->date[i].addr
			);
	}
}

int FindByName(const Contact* ps,char name[])
{
	int i = 0;
	for (i =0; i<ps->size; i++)
	{
		if (strcmp(ps->date[i].name,name)==0)
		{
			return i;
		}
	}
	return -1;
}

void DeleteContact(Contact* ps)
{
	assert(ps);
	if (ps->size == 0)
	{
		printf("通讯录为空,无法删除\n");
		return;
	}

	char name[20] = { 0 };
	printf("请输入你要删除的姓名:>");
	scanf("%s", name);

	int ret = FindByName(ps, name);
	if (ret == -1)
	{
		printf("要删除的人不存在\n");
		return;
	}

	for (int i =ret; i<ps->size; i++)
	{
		ps->date[i] = ps->date[i + 1];
	}

	ps->size--;
	printf("删除成功\n");
}

void SearchContact(Contact* ps)
{
	assert(ps);
	char name[20] = { 0 };
	printf("请输入你要删除的姓名:>");
	scanf("%s", name);

	int pos = FindByName(ps, name);
	if (pos == -1)
	{
		printf("要查找的人不存在\n");
		return;
	}

	printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
		ps->date[pos].name,
		ps->date[pos].age,
		ps->date[pos].sex,
		ps->date[pos].tele,
		ps->date[pos].addr
	);
}

void ModifyContact(Contact* ps)
{
	assert(ps);
	char name[20] = { 0 };
	printf("请输入你要修改的姓名:>");
	scanf("%s", name);

	int pos = FindByName(ps, name);
	if (pos == -1)
	{
		printf("要修改的人不存在\n");
		return;
	}

	printf("请输入修改的姓名:>");
	scanf("%s", ps->date[pos].name);
	printf("请输入修改的年龄:>");
	scanf("%d", &ps->date[pos].age);
	printf("请输入修改的性别:>");
	scanf("%s", ps->date[pos].sex);
	printf("请输入修改的电话:>");
	scanf("%s", ps->date[pos].tele);
	printf("请输入修改的地址:>");
	scanf("%s", ps->date[pos].addr);
}

void SortContact(Contact* ps)
{
	int i = 0;
	int j = 0;
	for (i=0; i<ps->size-1; i++)
	{
		for (j =0; j<ps->size-1-i; j++)
		{
			if (strcmp(ps->date[j].name, ps->date[j+1].name) > 0)
			{
				PeoInfo tmp = ps->date[j];
				ps->date[j] = ps->date[j + 1];
				ps->date[j + 1] = tmp;
			}
		}
	}
	printf("排序成功\n");
}

void DestoryContact(Contact* ps)
{
	free(ps->date);
	ps->date = NULL;
	ps->capacity = ps->size = 0;
}

void SaveContact(Contact* ps)
{
	FILE* pf = fopen("Contact.dat", "wb");
	if (pf == NULL)
	{
		perror("SaveContact");
		return;
	}
	for (int i =0; i<ps->size; i++)
	{
		fwrite(ps->date+ i, sizeof(PeoInfo), 1, pf);
	}

	fclose(pf);
	pf = NULL;
}

 

#define  _CRT_SECURE_NO_WARNINGS 1
#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");

}

void Text()
{
	int input = 0;
	Contact con;
	InitContact(&con);
	do 
	{
		menu();
		printf("请输入你的选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case ADD:
			AddContact(&con);
			break;
		case DEL:
			DeleteContact(&con);
			break;
		case SEARCH:
			SearchContact(&con);
			break;
		case MODIFY:
			ModifyContact(&con);
			break;
		case SHOW:
			ShowContact(&con);
			break;
		case SORT:
			SortContact(&con);
			break;
		case EXIT:
			SaveContact(&con);
			DestoryContact(&con);
			printf("退出通讯录成功\n");
			break;
		default:
			printf("你选择的有误,请重新输入\n");
			break;

		}
	} while (input);
}

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

 是不是看到这里会感到很害怕??不用怕,跟着我的思路,你也可以实现它,我带着你一步一步实现每一个功能

  •  我们要实现这个功能,首先我们来看下面的代码,首先我们应该先选择,这里我选择了do while的语句,要实现这个功能,我们就用到了menu这个函数来打印菜单,其实很简单,我就用了printf函数来实现。
void Text()
{
	int input = 0;
	Contact con;
	InitContact(&con);
	do 
	{
		menu();
		printf("请输入你的选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case ADD:
			AddContact(&con);
			break;
		case DEL:
			DeleteContact(&con);
			break;
		case SEARCH:
			SearchContact(&con);
			break;
		case MODIFY:
			ModifyContact(&con);
			break;
		case SHOW:
			ShowContact(&con);
			break;
		case SORT:
			SortContact(&con);
			break;
		case EXIT:
			SaveContact(&con);
			DestoryContact(&con);
			printf("退出通讯录成功\n");
			break;
		default:
			printf("你选择的有误,请重新输入\n");
			break;

		}
	} while (input);
}

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

}

  • 我们直接来实现通讯录的基本功能,我先用struct来定义了一个人的基本信息,然后再用struct来包含人的基本信息,但是我还加上了size和capcacity,size是用来记录存储了多少个人,而capacity是用来说明有多少个空间
    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 *date;
    	int size;
    	int capacity;
    }Contact;

  •  我们一开始用通讯录不要忘了要初始化,我在这里是把ps->date指向的空间置位了NULL,size和capacity初始化为0,因为我们是要弄一个动态的通讯录,所以我们特意用结构体的指针date来设计。
  • 我们实现一个通讯录,我们先设想一个场景,如果你的手机关机了,重启后是不是通讯录里面的信息还是存在的,所以我们也要实现这样的功能。
  • 我在这里是用到了文件操作,我先创建了结构体的tmp临时变量,我用fread来操作,如果freaed的返回值不是0,我们就将数据拷贝到tmp中。
  • while (fread(&tmp,sizeof(PeoInfo),1,pf))
    	{
    		CheakCapacity(ps);
    		ps->date[ps->size] = tmp;
    		ps->size++;
    	}

    下面是这个功能的全部代码

  • void LoadContact(Contact* ps)
    {
    	FILE* pf = fopen("Contact.dat", "rb");
    	if (pf == NULL)
    	{
    		perror("LoadContact");
    		return;
    	}
    	PeoInfo tmp = { 0 };
    	while (fread(&tmp,sizeof(PeoInfo),1,pf))
    	{
    		CheakCapacity(ps);
    		ps->date[ps->size] = tmp;
    		ps->size++;
    	}
    
    	fclose(pf);
    	pf = NULL;
    
    	
    }
    
    void InitContact(Contact* ps)
    {
    	assert(ps);
    	ps->date = NULL;
    	ps->size = ps->capacity = 0;
    	LoadContact(ps);
    }

  •  接着我们来看第二个功能,增加人的信息,我们在一开始增加信息的时候,要想到如果空间满了的话,就要考虑扩容。所以,我一开始就判断是否要扩容,因为一开始size和capacity都是0,所以一开始就要扩容,我是malloc了一个空间,如果满了的话,我就扩二倍。最后通过返回值来判断是否扩容成功了。
int CheakCapacity(Contact* ps)
{
	assert(ps);
	if (ps->capacity == ps->size)
	{
		int newcapacity = ps->capacity == 0 ? 4 : ps->capacity * 2;
		PeoInfo* tmp = (PeoInfo*)realloc(ps->date, newcapacity * sizeof(PeoInfo));
		if (tmp == NULL)
		{
			perror("CheakCapacity");
			return 0;
		}
		else
		{
			ps->date = tmp;
			ps->capacity = newcapacity;
			printf("增容成功\n");
			return 1;
		}
	}
	return 1;
}
  • 然后,接着实现add函数,其实很简单,我们一开始的size是0,所以每当我们增加一个信息,ps->size就要++,而ps->date指向的就是人信息的那片空间,ps->date【ps->size】后面再加上我们要增加的信息,就完成了我们add函数的功能。

void AddContact(Contact* ps)
{
	assert(ps);
	if (CheakCapacity(ps) == 0)
	{
		return;
	}

	printf("请输入增加的姓名:>");
	scanf("%s", ps->date[ps->size].name);
	printf("请输入增加的年龄:>");
	scanf("%d", &ps->date[ps->size].age);
	printf("请输入增加的性别:>");
	scanf("%s", ps->date[ps->size].sex);
	printf("请输入增加的电话:>");
	scanf("%s", ps->date[ps->size].tele);
	printf("请输入增加的地址:>");
	scanf("%s", ps->date[ps->size].addr);
	
	ps->size++;
	printf("增加成功\n");
}


  •  第二个删除的功能,我的思想就是先创建一个数组,然后用数组和通讯录中名字相比较看是否相等。然后返回要删除的下标
int FindByName(const Contact* ps,char name[])
{
	int i = 0;
	for (i =0; i<ps->size; i++)
	{
		if (strcmp(ps->date[i].name,name)==0)
		{
			return i;
		}
	}
	return -1;
}
  • 最后到删除的操作就是后面往前面移动,然后ps->size--就可以了。
void DeleteContact(Contact* ps)
{
	assert(ps);
	if (ps->size == 0)
	{
		printf("通讯录为空,无法删除\n");
		return;
	}

	char name[20] = { 0 };
	printf("请输入你要删除的姓名:>");
	scanf("%s", name);

	int ret = FindByName(ps, name);
	if (ret == -1)
	{
		printf("要删除的人不存在\n");
		return;
	}

	for (int i =ret; i<ps->size; i++)
	{
		ps->date[i] = ps->date[i + 1];
	}

	ps->size--;
	printf("删除成功\n");
}

 


  •  第三个search功能的实现,也是和删除差不多的操作,不过我们是查找功能,所以我们最后是直接printf来打印出我们要查找的信息,这里我也用到了FindByName函数的复用。

void SearchContact(Contact* ps)
{
	assert(ps);
	char name[20] = { 0 };
	printf("请输入你要删除的姓名:>");
	scanf("%s", name);

	int pos = FindByName(ps, name);
	if (pos == -1)
	{
		printf("要查找的人不存在\n");
		return;
	}

	printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
		ps->date[pos].name,
		ps->date[pos].age,
		ps->date[pos].sex,
		ps->date[pos].tele,
		ps->date[pos].addr
	);
}


  •  第四个修改的功能,因为我们也要找到要修改的下标,所以我用到了函数的复用,我们只要在要修改的下标中重新输入自己想修改的值就可以了。
void ModifyContact(Contact* ps)
{
	assert(ps);
	char name[20] = { 0 };
	printf("请输入你要修改的姓名:>");
	scanf("%s", name);

	int pos = FindByName(ps, name);
	if (pos == -1)
	{
		printf("要修改的人不存在\n");
		return;
	}

	printf("请输入修改的姓名:>");
	scanf("%s", ps->date[pos].name);
	printf("请输入修改的年龄:>");
	scanf("%d", &ps->date[pos].age);
	printf("请输入修改的性别:>");
	scanf("%s", ps->date[pos].sex);
	printf("请输入修改的电话:>");
	scanf("%s", ps->date[pos].tele);
	printf("请输入修改的地址:>");
	scanf("%s", ps->date[pos].addr);
}


 是不是发现其实函数的实现很多都是相似的, 所以不用怕,接着往后看。

  •  第五个函数:就是我们要展示信息,我们直接用for循环遍历一遍就行了,只不过我们为了好看一点,用到了左对齐,至于长度可以根据你来实现。最后的效果就是下面这样。

 

void ShowContact(Contact* ps)
{
	assert(ps);

	printf("%-10s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");

	for (int i =0; i<ps->size; i++)
	{
		printf("%-10s\t%-4d\t%-5s\t%-12s\t%-30s\n",
			ps->date[i].name,
			ps->date[i].age,
			ps->date[i].sex,
			ps->date[i].tele,
			ps->date[i].addr
			);
	}
}


  •  第六个函数:我们要排序名字的大小,这里我直接用了冒泡排序,可以直接把名字的大小排出来。只不过这里的时间复杂度是O(N^2),效率是很低的,也可以用快排来实现,效率可以更高一点
void SortContact(Contact* ps)
{
	int i = 0;
	int j = 0;
	for (i=0; i<ps->size-1; i++)
	{
		for (j =0; j<ps->size-1-i; j++)
		{
			if (strcmp(ps->date[j].name, ps->date[j+1].name) > 0)
			{
				PeoInfo tmp = ps->date[j];
				ps->date[j] = ps->date[j + 1];
				ps->date[j + 1] = tmp;
			}
		}
	}
	printf("排序成功\n");
}


  •  因为我们是动态来实现通讯录的,所以我用到了malloc,在最后退出的时候,也要free掉开辟的空间,我用了一个DestoryContact函数来实现,。
  • void DestoryContact(Contact* ps)
    {
    	free(ps->date);
    	ps->date = NULL;
    	ps->capacity = ps->size = 0;
    }


  • 最后一个函数:是否想过这样的一个问题,在我们输入信息的时候,如果我们退出,信息在下一次打开时还保存着呢???经过学习,我发现文件操作就可以来实现它 。
  • 我是用了fopen来打开一个二进制的文件Contact.dat,用到了for循环fwrite来把已经存在的信息保存在文件流中(也就是这个文件中),又因为一开始初始化的时候,要把文件的信息录进去,这样,我们就实现了信息的保存。
void SaveContact(Contact* ps)
{
	FILE* pf = fopen("Contact.dat", "wb");
	if (pf == NULL)
	{
		perror("SaveContact");
		return;
	}
	for (int i =0; i<ps->size; i++)
	{
		fwrite(ps->date+ i, sizeof(PeoInfo), 1, pf);
	}

	fclose(pf);
	pf = NULL;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Nautilus Chain 更换全新测试网,主网即将在不久上线

目前&#xff0c;Nautilus Chain 正在为主网上线前的最后阶段做准备&#xff0c;据悉该链更新了全新的测试网&#xff0c;在此前版本的测试网的基础上进行了全新的技术升级&#xff0c;最新测试网版本与生态发展的技术规划更为贴近。本次测试网升级将会是最后一次测试网版本的迭…

基于深度学习的高精度80类动物目标检测系统(PyTorch+Pyside6+YOLOv5模型)

摘要&#xff1a;基于深度学习的高精度80类动物目标检测识别系统可用于日常生活中或野外来检测与定位80类动物目标&#xff0c;利用深度学习算法可实现图片、视频、摄像头等方式的80类动物目标检测识别&#xff0c;另外支持结果可视化与图片或视频检测结果的导出。本系统采用YO…

Ubuntu下配置Redis哨兵集群

目录 准备实例和配置 启动哨兵集群 测试配置 搭建一个三节点形成的Sentinel集群&#xff0c;来监管Redis主从集群。 三个sentinel哨兵实例信息如下&#xff1a; 节点IPPORTs1192.168.22.13527001s2192.168.22.13527002s3192.168.22.13527003 准备实例和配置 要在同一台虚…

week2

第2周 目录 第2周 四、多变量线性回归(Linear Regression with Multiple Variables) 4.1 多维特征4.2 多变量梯度下降4.3 梯度下降法实践1-特征缩放4.4 梯度下降法实践2-学习率4.5 特征和多项式回归4.6 正规方程4.7 正规方程及不可逆性(可选) 四、多变量线性回归(Linear Regres…

opencv 基础图像操作-彩色图像

opencv 基础图像操作-彩色图像 彩色图像 相比二值图像和灰度图像&#xff0c;彩色图像是更常见的一类图像&#xff0c;它能表现更丰富的细节信息。 神经生理学实验发现&#xff0c;在视网膜上存在三种不同的颜色感受器&#xff0c;能够感受三种不同的颜色&#xff1a;红色、绿色…

SpringBoot【原理分析、YAML文件、SpringBoot注册web组件】(二)-全面详解(学习总结---从入门到深化)

目录 SpringBoot原理分析_核心注解 YAML文件_配置文件介绍 YAML文件_自定义配置简单数据 YAML文件_自定义配置对象数据 YAML文件_自定义配置集合数据 YAML文件_读取配置文件的数据 使用ConfigurationProperties读取 YAML文件_占位符的使用 YAML文件_配置文件存放位置及优…

python调用钉钉发送告警消息

zabbix和prometheus都能做监控告警调用企业微信和钉钉。那么问题来了做为一名合格的运维和开发人员&#xff0c;脚本发现故障告警&#xff0c;自动调用发送详细告警内容&#xff0c;这点儿东西得会。 效果图如下&#xff1a; 普通发送方式 import requestsdef dingding_info(…

学习系统编程No.30【多线程控制实战】

引言&#xff1a; 北京时间&#xff1a;2023/7/7/9:58&#xff0c;耳机正在充电中&#xff0c;所以刚好让我们先把引言写一写&#xff0c;昨天睡觉前听了一会小说&#xff0c;听小说的好处就在于&#xff0c;它可以让你放下手机&#xff0c;快速睡觉&#xff0c;并且还有一定的…

【DBA课程-笔记】第2章:MongoDB数据数据查询与分析

目录 一、课程大纲 二、MongoDB 条件查询 1. 制造 MongoDB 测试数据 2. MongoDB 数据查询 与 SQL对应关系 3. MongoDB 查询运算符 4. MongoDB 数据查询、条件查询、过滤 5. MongoDB 条件查询命令 6. MongoDB 数据查询数组条件 A. 精确匹配数组元素&#xff1a; B. 无…

检测到错误页面web应用服务器版本信息泄露

详细描述 Web服务器未能正确处理异常请求导致Web服务器版本信息泄露&#xff0c;攻击者收集到服务器信息后可进行进一步针对性攻击。 解决办法 临时修复建议如下&#xff1a; 1、关闭web服务器错误提示。 2、关闭运行平台的错误提示。 3、建立错误机制&#xff0c;不要把真实…

校园课表微信小程序全栈项目

遇到的第一个问题关于npm start 首先找到对应的后端项目 crtl shift (esc键下的波浪号) 召唤终端 Error:Cannot find module dotenv 解决问题: npm install dotenv 简写 npm i dotenv 然后遇到第二个问题 Port 3000 is already in use 我的3000接口被占用 那就要去检…

leetcode 74. 搜索二维矩阵(java)

搜索二维矩阵 leetcode 74. 搜索二维矩阵题目描述抽象BST代码演示 抽象BST leetcode 74. 搜索二维矩阵 来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 链接&#xff1a;https://leetcode.cn/problems/search-a-2d-matrix 题目描述 给你一个满足下述两条属性的 m x n 整…

uniapp和uview组件实现下拉触底刷新列表

下面是一个在UniApp中使用uView组件实现下拉触底刷新列表的示例&#xff0c;并使用Axios来请求分页数据列表&#xff1a; 首先&#xff0c;确保你已经在UniApp项目中添加了uView组件库。你可以在项目根目录执行以下命令安装它们&#xff1a; npm install uview-ui或者使用 Hb…

【Vue3】setup参数细讲!computed计算属性和watch监听属性

setup参数细讲&#xff01;computed计算属性和watch监听属性 setup细讲!setup参数&#xff0c;steup&#xff08;props&#xff0c;context&#xff09;参数1.props&#xff0c;负责接收父组件传过来的值参数2.contextcontext.attrscontext.emitcontext.slots&#xff0c; 插槽…

CSS 伪元素: ::marker 自定义列表序号

::marker 伪元素 ::marker&#xff0c;可作用在任何设置了 display: list-item 的元素或伪元素上&#xff0c;例如<li>和<summary>。 /** <ul><li>Peaches</li><li>Apples</li><li>Plums</li> </ul> */ ul li::…

Java 设计模式——迭代器模式

目录 1.概述2.结构3.案例实现3.1.抽象迭代器3.2.具体迭代器3.3.抽象聚合3.4.具体聚合3.5.测试 4.优缺点5.使用场景6.JDK 源码解析——Iterator 1.概述 迭代器模式 (Iterator Pattern) 是一种行为型设计模式&#xff0c;它提供一种顺序访问聚合对象&#xff08;如列表、集合等&…

Hyperledger Fabric测试网络运行官方Java链码[简约版]

文章目录 启动测试网络使用peer CLI测试链码调用链码 启动测试网络 cd fabric-samples/test-networknetwork.sh的脚本语法是&#xff1a;network.sh <mode> [flag] ./network.sh up./network.sh createChannel在java源码路径下 chmod 744 gradlew vim gradlew :set ffu…

「观察者(Observer)」设计模式 Swift实现

这里写目录标题 介绍设计模式介绍举例 iOS 中已有的 观察者设计模式实现Notification什么是通知机制或者说如何实现通知机制&#xff1f; KVOKVO底层实现如何实现手动KVO&#xff1f; 介绍 设计模式介绍 观察者设计模式&#xff08;Observer Pattern&#xff09;是一种行为型…

【ArcGIS Pro二次开发】(49):村规数据入库【福建省】

之前用Arcpy脚本工具做了一个村规数据入库和主要图纸生成工具。 在使用过程中&#xff0c;感觉对电脑环境比较高&#xff0c;换电脑用经常会一些莫名其妙的错误&#xff0c;bug修得很累。近来随着ArcGIS Pro SDK的熟悉&#xff0c;就有了移植的想法。 这里先把村规数据入库工…

TabLayout+ViewPager实现滚动页面

目录 一、TabLayout介绍 二、TabLayout的常用属性和方法 常用属性&#xff1a; 常用方法&#xff1a; 三、适配器介绍 &#xff08;一&#xff09;、PagerAdapter介绍&#xff1a; &#xff08;二&#xff09;、FragmentPagerAdapter介绍&#xff1a; &#xff08;三&am…