ArcGIS arcpy代码工具——关于标识码的那些事(查找最大标识码、唯一性检查、重排序、空值赋值)

系列文章目录

ArcGIS arcpy代码工具——批量对MXD文件的页面布局设置修改
ArcGIS arcpy代码工具——数据驱动工具批量导出MXD文档并同步导出图片
ArcGIS arcpy代码工具——将要素属性表字段及要素截图插入word模板
ArcGIS arcpy代码工具——定制属性表字段输出表格
ArcGIS arcpy代码工具——批量栅格转点文件导出属性表


文章目录

  • 系列文章目录
  • 功能说明
  • 准备工作
  • 1 查找最大标识码
      • (1) 首先,判定字段是否存在
      • (2) 之后,判定图层是否为空
      • (3) 查找最大值
      • (4) 结果输出txt文档
      • (5) 运行效果
  • 2 标识码唯一性检查
      • (1) 首先,判定字段是否存在、图层是否为空
      • (2) 判定标识码值是否重复
      • (3) 结果输出txt文档
      • (4) 运行效果
  • 3 标识码重排序
      • (1) 首先,对数据库进行备份
      • (2) 判定字段是否存在
      • (3) 默认起始值
      • (4) 重排标识码
      • (5) 记录最大值
      • (6) 运行效果
  • 4 更新补全字段、空值赋值
      • (1) 判定字段是否存在
      • (2) 起始值(仅可以填写整数)
      • (3) 选择递增值,工具判定统一赋值操作还是补全字段操作
      • (4) 根据两种不同操作,输出不同结果语句
      • (5) 运行效果
  • 5 后记


功能说明

关于标识码的那些事(查找最大标识码、唯一性检查、重排序、空值赋值)。

首先,认识一下标识码,标识码通常是指用于唯一标识某个实体(如人、物、组织、事件等)的一串字符,它可以是数字、字母或者是数字与字母的组合。在不同的领域中,名称标识码有着不同的应用和含义。类如身份证号码,唯一标识人的特征码。

在数据库中,标识码是一种用于唯一标识数据库对象的代码或值。它通常被用作主键或索引的基础。标识码可以是数字、字符或二进制数据,具体取决于数据库管理系统的实现。标识码可以通过自动生成、手动输入或使用特定算法来创建。在使用标识码时,需要注意确保其唯一性以避免数据冲突。

一般在地理数据库中,给图层的字段命名为BSM、标识码,字段类型有数值型或者字符型,是标识和定位地理要素的索引,故一般要求具有唯一性,不能重复,不能空值,维护起来较为复杂。

本文章针对维护标识码工作的痛点,介绍一些维护工具目标为:

  • 1 查找数据库中最大标识码
  • 2 标识码唯一性检查
  • 3 重排序标识码
  • 4 标识码空值赋值、更新补全

准备工作

实验工具,随便找一个地理数据库,可以是mdb或gdb格式,包含数据集图层和数据集外图层,每个图层含有BSM字段,字段类型有数值型,有字符型,个别图层缺失BSM字段,个别图层要素为空,个别图层BSM字段为空值null或者空格,个别图层的BSM值是重复值。

下面是样式数据的结构图:
1

1 查找最大标识码

一个合格的地理数据库,每个图层的每个要素都应该有一个唯一的标识码,那么这个地理数据库也应该有一个最大标识码,获取到最大标识码,针对地理数据库后续增加的要素就可以准确和方便的进行标识码的赋值工作。

如何查找地理数据库中的最大标识码?

思路如下:
遍历地理数据库中的要素集和单独图层,再遍历每个图层的中的每个要素,将标识码的值放入列表 [ ] 中,然后对列表取最大值,得到最大标识码。

再复杂一点,可以建立字典 { },针对每个图层进行查询,图层名作为字典的键,该图层的最大标识码作为字典的键值。这样就获知了每个图层的最大标识码,最后再对字典的键值取最大值,得到地理数据库的最大标识码值。

(1) 首先,判定字段是否存在

查找字段BSM的最大值,首先要判定字段是否缺失,如果缺失BSM字段,则后续的查找最大值也就可以跳过了。

关键代码 :
首先获取图层字段的集合,再使用 if 语句进行判定BSM字段是否在集合内,判定缺失则输出警告语句。

field_names = [f.name for f in arcpy.ListFields(feature_class)]
if field_name not in field_names:arcpy.AddWarning(xxxxx)

(2) 之后,判定图层是否为空

图层有BSM的字段,如果没有要素,空图层,后续查找最大值也可以跳过了。

关键代码 :
如何判定是否是空图层?可以这样理解,该图层的最大标识码为空值,即 None。如果为空,输出警告语句。

max_id = max(long(row[0]) for row in cursor)
max_id = None
arcpy.AddWarning(xxxxx)

(3) 查找最大值

图层有BSM字段,要素不为空,则可以查找最大值了。

关键代码 :
地理数据库,有要素数据集,有直接位于工作空间内的要素,在遍历的时候,要素数据集内的图层比外面的图层要多一层目录(即数据集的名称),这是成功遍历地理数据库所有要素层的关键。

//遍历所有要素集内的要素,如样例数据 T2021、T2022、T2025.
for fds in arcpy.ListDatasets("*", "Feature"):for fc in arcpy.ListFeatureClasses("*", "", fds):......//遍历直接位于工作空间内的要素,如样例数据的 T2023、T2024.
for fc in arcpy.ListFeatureClasses("*", "", ""):......

(4) 结果输出txt文档

地理数据库的图层可能很多,如果都使用arcpy.AddWarning()、arcpy.AddMessage()语句直接在运行窗口显示结果,无法仔细查看详细记录,故最好把结果输出到一个txt结果文档中保存,方便随时查阅。

关键代码 :
可以创建一个输出 txt文档的 函数,将查询过程中产生的 结果语句,调用函数写入txt文档。

def log_to_file(message, outtxt):with open(outtxt, "a") as txtfile:txtfile.write(message.encode('utf-8') + "\n")def find_max_ids_in_workspace(a, b, output_file):......log_to_file(unicode(message, "utf-8"), output_file)

(5) 运行效果

我制作了arcgis工具箱工具 快速查找最大标识码BSM工具 ,有需要的可以去下载使用,下图是运行界面:

2

参数1:输入mdb和gdb地理数据库,支持中文目录;
参数2:指定查询的字段名称,可以是数值型字段和文本型字段,支持中文字段名;
参数3:输出结果txt,点击确定即可运行查找最大标识码。

我设置的地理数据库是某地类库,字段名称为BSM,运行结果如下图:
3
输出的 output.txt 保存的内容与运行窗口显示的一致,将运行结果完整的保存了下来。

2 标识码唯一性检查

标识码的作用就是标识和定位要素的唯一特征码,如果标识码重复了,那么地理数据库就是不合格的。所以,检查标识码的唯一性这项工作,就显得非常必要。

如何判定地理数据库中的标识码是否存在重复值?

思路如下:
遍历地理数据库中的要素集和单独图层,再遍历每个图层的中的每个要素,获取每个要素的标识码,在将 当前标识码 放入总列表 [ ] 前,判断该标识码值是否存在于 总列表 [ ] 中,如果在总列表中已经存在,则说明该标识码值存在重复,将该重复值存入单独的 重复值列表 [ ]。最后将重复的标识码结果输出txt文档进行记录。

再复杂一点,可以建立字典 { },标识码作为字典的键(因为重复的标识码可能存在于不同的图层中,而标识码值作为不同数字是唯一的),图层名字作为字典键值,当存入 重复值字典 { } 中,可以知道 重复的标识码 在 哪个图层中出现。

(1) 首先,判定字段是否存在、图层是否为空

首先要判定字段是否缺失,图层是否为空,满足条件后,开始后续的查找重复值。

(2) 判定标识码值是否重复

判定标识码是否重复,因为要挂上图层名称,处理起来要复杂一些。

首先建立总的标识码库 字典 all_values {},每个标识码值和图层名称组成字典键值对,存入总库;

再建立一个 比较过程的字典 temp_value {},因为要遍历每个图层中的每个要素,重复标识码可能存在于同一个图层内,也可能存在跨图层重复,过程字典 temp_value {} 就是存储单个图层的重复值情况;

每个图层的重复值 temp_value {} 再跟 总库 all_values {} 进行比较,找出跨图层的重复值情况,并记录图层名称,存入 最终的重复标识码库 duplicates {}

最后,使用 format 语句读取 重复标识码库 duplicates {} 输出重复标识记录,输出txt记录文档。

关键代码 :
情况1: 重复标识码存在于同一个图层内。

all_values = {} # 用于存储所有图层中该字段的所有值及其所在图层
temp_value = {} # 存储 同图层  重复值情况
......
for row in cursor:value = row[0]# 如果值已经在 all_values 中,说明重复if value in all_values:# 更新temp_value,记录重复的图层名if value not in temp_value:temp_value[value] = [all_values[value], feature_class]else:temp_value[value].append(feature_class)

情况2: 重复标识码存在于跨图层。

all_values = {} # 用于存储所有图层中该字段的所有值及其所在图层
duplicates = {}  # 存储最终发现的重复值及其所在图层列表
feature_class = 图层名称
......
for row in cursor:value = row[0]   if value in all_values:# 如果值已存在,则记录为重复duplicates.setdefault(value, set()).add(feature_class)                                duplicates[value].add(all_values[value])else:all_values[value] = feature_class

关键语句: duplicates.setdefault(value, set()).add(feature_class)

该函数首先通过 setdefault 方法,检查 value 是否存在于 duplicates字典 中,
如果存在,则直接返回对应的集合;
如果不存在,则在 duplicates字典 中创建一个以value为键的空集合,并将该集合返回。
接下来,通过调用集合的add方法,将feature_class图层名称添加到集合中。
这样可以确保同一个value所对应的集合中,不会出现重复的feature_class图层名称。
最后,该函数没有返回值,只是更新了 duplicates字典 中相应键值对应的集合。

(3) 结果输出txt文档

把结果输出到一个txt结果文档中保存,方便随时查阅。

(4) 运行效果

我制作了arcgis工具箱工具 标识码BSM唯一性检查工具,有需要的可以去下载使用,下图是运行界面:
4
参数1:输入mdb和gdb地理数据库,支持中文目录;
参数2:指定查询的字段名称,可以是数值型字段和文本型字段,支持中文字段名;
参数3:输出结果txt,点击确定即可运行查找最大标识码。

我设置的地理数据库是准备好的 test.mdb,其中存在重复的标识码,字段名称为BSM,运行结果如下图:
5
输出的 标识码唯一性检查结果.txt 保存的内容与运行窗口显示的一致,将运行结果完整的保存了下来。

3 标识码重排序

一个新建的地理数据库,在完成图形要素的编辑后,一般都是在最后一个步骤进行标识码重排序工作,即对地理数据库中的所有图层每个要素,安排一个唯一的不重复的标识码。一般要求同一个图层内要素的标识码是从小到大按增序排列的。

arcgis软件在保存地理要素的时候,要素图层有个默认字段 OBJECT ID字段,即常说的ID字段,通常由数据库系统自动分配,它的值是一个递增的整数序列,从1开始,每插入一条记录,其值就会自动加1,它是一种唯一标识符,用于标识数据库表中的每一行。这个字段可以作为空间数据表的主键,用于建立与其他表的关联。

arcgis软件默认会给每个要素一个不重复的 ID,但是如果删除了某个要素,则该要素的原始ID号在这个地理数据库中将消失不再重现,后续新建立的要素 ID会按照本图层最大ID续排。如果手工设置标识码,可以参考 这个ID字段值,但是难免发生中间断号的情况。

如何完成对整库标识码的重排序工作?

思路如下:
遍历地理数据库中的要素集,遍历要素集中的第1个图层,打开要素属性表,给每行属性表的BSM字段填写数值,从1开始往后增加,到该图层的最后一个要素结束,记录最后的标识码值,作为要素集第2个图层属性表中填写标识码的起始值。依此类推,遍历完要素集后,再遍历外部的图层,遍历完所有的图层后,完成标识码的完整重排序工作。

注意:思路中使用的方法是,调用要素属性表,按照要素属性表的顺序从上到下安排标识码,而属性表的顺序就是 ID字段 生成的顺序,所有本质上还是利用了 ID字段。

更高级的要素排序方法,还有按照地理要素的坐标值,进行从上到下,从左到右的方式进行排序,这种排序方式的好处是,临近的要素 顺序号较为接近或者连续,方便查找定位。

而我的文章中使用的方法,调用ID字段,虽然系统ID值也是按照要素产生的顺序先后生成的,但是不能保证周边的要素ID值是连续的。

(1) 首先,对数据库进行备份

涉及到对整库操作,对某个字段进行重新填写值,原有标识码值将被清除,为了数据安全,在标识码重排序前还是建立一个备份数据库为好。数据备份工作也是 一个成熟GISer的良好习惯。

备份库命名就以时间戳命名,因为地理数据库有文件地理数据库(.gdb)和个人地理数据库(.mdb)
两种格式,后缀名获取自 参数1数据库名称即可。

关键代码 :

def create_backup(workspace, backup_path):timestamp = datetime.now().strftime("%Y%m%d%H%M")  # 获取当前时间戳mdborgdb = os.path.splitext(workspace)[1]  # 更加可靠地获取数据库扩展名backup_name = "备份库_{}{}".format(timestamp, mdborgdb)  # 创建备份文件名backup_full_path = os.path.join(backup_path, backup_name)  # 构建完整备份路径arcpy.Copy_management(workspace, backup_full_path, data_type="Workspace")

(2) 判定字段是否存在

首先要判定字段是否缺失,字段缺失将跳过该图层。满足条件后,开始后续的查找重复值。

(3) 默认起始值

这里增加了一个参数:起始值,默认起始值为1,即从1开始续排。
如果有特殊要求,比如说为了区分数据库,第一个数据库要求标识码从1开始最大999,第二个数据库要求标识码从1000开始最大1999,这个起始值可以设定。

(4) 重排标识码

从设定的起始值start_value开始重排标识码,第一个图层结束后,会产生一个临时最大值,作为下一个图层的起始值,所以 start_value值是一直在变化的。

遍历并增加过程值,最好的方式是 枚举,使用start_value作为计数器的起始值。
关键代码 :

with arcpy.da.UpdateCursor(feature_layer, bsm_field_name) as cursor:for i, row in enumerate(cursor, start=start_value):row[0] = icursor.updateRow(row)......i += 1......

(5) 记录最大值

在重排序函数的外面,建立一个全局参数 max_id = [],将过程最大标识码加入 列表中。最后对列表求取最大值,使用输出语句进行汇报:重排序后最大值为xxx。

关键代码 :

 global max_id = []  # 初始化列表用于存储每个要素类的最大值......max_id.append(i)  # 假设i是当前要素类的最大排序值......
max_value = max(max_id)
arcpy.AddMessage("重排序完成,最大值为 {} .".format(max_value))

(6) 运行效果

我制作了arcgis工具箱工具 数据库标识码BSM重排序工具、重构标识码工具,有需要的可以去下载使用,下图是运行界面:

6
参数1:输入mdb和gdb地理数据库,支持中文目录;
参数2:指定查询的字段名称,可以是数值型字段和文本型字段,支持中文字段名;
参数3:重排序起始值,默认从1开始重排,也可以修改重排的起始值如1000、2000等,以2000、2001、2002排序;
参数4:备份库保存路径,这里选择路径文件夹即可,工具会按照时间戳给备份库命名并复制备份库。

我设置的地理数据库是某地类库,重排字段名称为BSM,重排起始值为1,备份库保存路径为 “D:\临时” ,运行结果如下图:
7

8

4 更新补全字段、空值赋值

在编辑地理数据库的时候,新增的要素往往来不及给标识码赋值,结果产生一些要素的BSM字段为空值,或者对某些选中的要素进行标识清空操作。或者是某些字段需要统一赋相同值,如要素代码。

如何实现这两个功能?

思路如下:
工具设定的是给空值字段赋值,可以是空值null,也可以是填充的空格,所以在使用工具前,需要思考自己想要实现的操作。
工具仅对空值或者空格值的要素进行操作,所以数据安全有保障。

情况1: 统一赋值,即对某个字段填写相同的值,前提条件是该字段为空,否则只填写那些字段为空值的要素。可以使用字段计算器 “字段 = null” 对字段提前清空,经过这种深思熟虑思考的操作后,也就免除了数据备份工作,代表着这一步骤并不是误操作。

情况2: 更新补全字段,针对空值的要素,有一个起始值,从起始值开始续排字段。这就要提前知道该地理数据库中现有的最大值标识码,可以使用我编写的工具 快速查找最大标识码BSM工具 查询出最大标识码,作为补全操作的起始值。

(1) 判定字段是否存在

首先要判定字段是否缺失,字段缺失将跳过该图层。满足条件后,开始后续的补全操作。

(2) 起始值(仅可以填写整数)

对空值字段操作的起始值,如果是统一赋值则表示选定字段都填写 该起始值。
如果是补全字段,则从起始值(含)开始每次+1续排。
因为要进行 加法计算,所以 起始值只能填写整数,即全都是数字组成的,纯数字组成的字符型也可以。

(3) 选择递增值,工具判定统一赋值操作还是补全字段操作

每次递增值,有 0 、 1 两个选项:
选择递增值为 0,表示进行统一赋值操作,对字段统一赋值为起始值;
选择递增值为 1,表示补全字段操作,从起始值(含)进行 +1 续排。

关键代码 :

my_workspace = 数据库
field_name = 字段名称
starting_value = 起始值
incremental_value = 递增值
countvalue = 0  # 更新要素的计数,更新了多少个def updpdateFc(my_workspace, field_name, starting_value, incremental_value, countvalue=0):
......for row in cursor:if row[0] is None or row[0] == "" or row[0] == " " or row[0] == "  " or row[0] == "   ": #找到空值 或 空格值row[0] = starting_valuecursor.updateRow(row)if incremental_value != 0:    # 判定,如果递增值 不是 0,表示补全更新,starting_value +1 续排。starting_value = starting_value + incremental_value   # 过程最大值countvalue += 1                              # 更新要素的计数......

(4) 根据两种不同操作,输出不同结果语句

使用 if 判定,输出两种不同的结果语句。

关键代码 :

if incrementalvalue != 0:starting_value, countvalue = updpdateFc(myworkspace, fieldname, int(startingvalue), int(incrementalvalue))arcpy.AddMessage("更新起始值 " + str(startingvalue) + ",终了最大标识码 " + str(starting_value-1) + ",更新数量 " + str(countvalue) + "。")
else:starting_value, countvalue = updpdateFc(myworkspace, fieldname, startingvalue, incrementalvalue)arcpy.AddMessage("空值字段统一赋值成功,赋值数量 " + str(countvalue) + "。")

(5) 运行效果

我制作了arcgis工具箱工具 更新补全字段BSM工具,空值字段赋值工具,有需要的可以去下载使用,下图是运行界面:

9

参数1:输入mdb和gdb地理数据库,支持中文目录;
参数2:指定查询的字段名称,可以是数值型字段和文本型字段,支持中文字段名;
参数3:起始值,仅可以输入整数;
参数4:每次递增值,选择 0 表示统一赋值为 起始值; 选择 1 表示 从起始值(含) +1 续排。

我设置的地理数据库是准备好的 test.mdb,字段名称为BSM,起始值为2144477777,递增值选择 0 ,即 统一赋值操作,运行结果如下图:
10
递增值选择 1 ,即 更细补全字段操作,运行结果如下图:

11

5 后记

地理数据库的标识码BSM,是日常工作中经常需要维护的一个工作,尤其有些数据库更新工作,去年的旧库作为今年的基础库,这种库的标识码一般情况下是不允许重排的。这种情况下,就需要掌握 旧库的标识码信息,如最大值等等,方便后续工作。网络上也有类似的标识码更新工具,只有一个运行操作,结果不透明,不知其所以然。

我就自己编写了这几个工具,主要是能够输出记录,方便历史查阅。这些工具也可以用于给字段赋值,兼容数值型字段和字符型字段(纯数字组成),支持中文路径,支持mdb和gdb格式地理数据库。

编写arcgis工具的过程,也练习了arcpy的编程,这也是一个学习的过程,大家多多支持,多收藏多点赞,一起共勉之。

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

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

相关文章

复制即用!纯htmlcss写的炫酷input输入框

一般我们写css样式都要用样式库,但是嫌麻烦,如果能找到现成的内容复制上去就很香了,下文是笔者觉得好看的纯html&css写的样式,可以直接复制到Vue等内,十分方便。 input组件 1) 下面这个很推荐&#…

C++的数据结构(九): 笛卡尔树

笛卡尔树(Cartesian Tree)是一种特殊的二叉树,其每个节点的键值(key)满足二叉搜索树的性质,即左子树上所有节点的键值小于根节点的键值,右子树上所有节点的键值大于根节点的键值。与此同时&…

C 深入指针(4)

目录 一、字符指针变量 1 初始化 2 与字符串数组的区别 二、数组指针变量 1 初始化 2 二维数组传参本质 三、函数指针变量 1 初始化 2 用法 四、typedef关键字 五、函数指针数组 一、字符指针变量 1 初始化 //VS2022 x64 #include <stdio.h> int main() {…

InnoDB 事务处理机制

文章目录 前言1. 事务处理挑战1.1 事务机制处理的问题1.2 并发事务带来的问题 2. InnodDB 和 ACID 模型2.1 Innodb Buffer Pool2.2 Redo log2.3 Undo log2.4 应用案例 3. 隔离级别和锁机制3.1 事务隔离级别3.1.1 READ UNCOMMITTED3.1.2 READ COMMITTED3.1.3 REPEATABLE READ3.1…

Linux下redis源码编译安装

华子目录 Redis介绍什么是RedisRedis能干什么Redis的特点Redis与memcached对比 redis源码编译安装下载源码包准备安装环境开始编译开始安装 前台启动后台启动redis开启systemctl启动redis测试redis相关知识 Redis介绍 什么是Redis 2008年&#xff0c;意大利的一家创业公司Mer…

xlrd.biffh.XLRDError: Excel xlsx file; not supported报错原因

xlrd库读取xlsx文件时报错 xlrd.biffh.XLRDError: Excel xlsx file; not supported报错原因&#xff1a; xlrd版本为2.1版本&#xff0c;需要读取xlsx文件需要安装xlrd低一些版本1.2.0版本&#xff0c;重新安装重试即可 更换xlrd版本 重新运行

数仓架构之为什么要进行数仓分层

数仓分层这个概念想必大家都很熟悉&#xff0c;不管是在实际的开发工作当中会用的&#xff0c;还是在面试官面试你的时候会问到&#xff1a;你之前的项目是按照什么分层的&#xff0c;分哪几层&#xff0c;数仓分层有什么好处&#xff0c;举个栗子说说。 简而言之&#xff0c;…

element-ui dialog form 弹框表单组件封装

在使用 element-ui 进行后端管理系统开发时&#xff0c;在封装弹框表单时&#xff0c;遇到两个问题&#xff0c;这里进行简单记录&#xff1a; 1、问题一&#xff1a;点击关闭按钮及遮罩层关闭弹框时&#xff0c;页面报错&#xff0c;如下&#xff1a; 子组件封装&#xff1a;…

上海初中生古诗文大会倒计时4个月:单选题真题示例和独家解析

现在距离2024年初中生古诗文大会还有4个多月时间&#xff0c;备考要趁早&#xff0c;因为知识点还是相对比较多的。这些知识点对于初中语文的学习也是很有帮助的。 今天我们继续来看10道选择题真题和详细解析&#xff0c;以下题目截取自我独家制作的在线真题集&#xff0c;都是…

普通人也能创业!轻资产短视频带货项目,引领普通人实现创业梦想

在这个信息爆炸的时代&#xff0c;创业似乎成为了越来越多人的梦想。然而&#xff0c;传统的创业模式 keJ0277 往往伴随着高昂的资金投入和复杂的管理流程&#xff0c;让许多普通人望而却步。然而&#xff0c;现在有一种轻资产短视频带货项目正在悄然兴起&#xff0c;它以其低…

【CSND博客纪念】“创作纪念日:从灵感迸发到小有成就——我的CSND博客创作之旅”

&#x1f3a9; 欢迎来到技术探索的奇幻世界&#x1f468;‍&#x1f4bb; &#x1f4dc; 个人主页&#xff1a;一伦明悦-CSDN博客 ✍&#x1f3fb; 作者简介&#xff1a; C软件开发、Python机器学习爱好者 &#x1f5e3;️ 互动与支持&#xff1a;&#x1f4ac;评论 &…

Verilog 实现 i2c 协议

在时钟&#xff08;SCL&#xff09;为高电平的时候&#xff0c;数据总线&#xff08;SDA&#xff09;必须保持稳定&#xff0c;所以数据总线&#xff08;SDA&#xff09;在时钟&#xff08;SCL&#xff09;为低电平的时候才能改变。 在时钟&#xff08;SCL&#xff09;为高电平…