CSDN每日一题学习训练——Java版(克隆图、最接近的三数之和、求公式的值)

版本说明

当前版本号[20231109]。

版本修改说明
20231109初版

目录

文章目录

  • 版本说明
  • 目录
  • 克隆图
    • 题目
    • 解题思路
    • 代码思路
    • 参考代码
  • 最接近的三数之和
    • 题目
    • 解题思路
    • 代码思路
    • 参考代码
  • 求公式的值
    • 题目
    • 解题思路
    • 代码思路
    • 参考代码

克隆图

题目

给你无向 连通(https://baike.baidu.com/item/连通图/6460995?fr=aladdin) 图中一个节点的引用,请你返回该图的 深拷贝(https://baike.baidu.com/item/深拷贝/22785317?fr=aladdin)(克隆)。

图中的每个节点都包含它的值 val(int) 和其邻居的列表(list[Node])。

class Node {
public int val;
public List neighbors;
}

测试用例格式:

简单起见,每个节点的值都和它的索引相同。例如,第一个节点值为 1(val = 1),第二个节点值为 2(val = 2),以此类推。该图在测试用例中使用邻接列表表示。

邻接列表 是用于表示有限图的无序列表的集合。每个列表都描述了图中节点的邻居集。

给定节点将始终是图中的第一个节点(值为 1)。你必须将 给定节点的拷贝 作为对克隆图的引用返回。

示例 1:

image-20231109230708214

输入:adjList = [[2,4],[1,3],[2,4],[1,3]]
输出:[[2,4],[1,3],[2,4],[1,3]]

解释:
图中有 4 个节点。
节点 1 的值是 1,它有两个邻居:节点 2 和 4 。
节点 2 的值是 2,它有两个邻居:节点 1 和 3 。
节点 3 的值是 3,它有两个邻居:节点 2 和 4 。
节点 4 的值是 4,它有两个邻居:节点 1 和 3 。

示例 2:

image-20231109230957257

输入:adjList = [[]]
输出:[[]]

解释:输入包含一个空列表。该图仅仅只有一个值为 1 的节点,它没有任何邻居。

示例 3:

输入:adjList = []
输出:[]
解释:这个图是空的,它不含任何节点。

示例 4:

image-20231109231042597

输入:adjList = [[2],[1]]
输出:[[2],[1]]

提示:

节点数不超过 100 。
每个节点值 Node.val 都是唯一的,1 <= Node.val <= 100。
无向图是一个简单图(https://baike.baidu.com/item/简单图/1680528?fr=aladdin),这意味着图中没有重复的边,也没有自环。
由于图是无向的,如果节点 p 是节点 q 的邻居,那么节点 q 也必须是节点 p 的邻居。
图是连通图,你可以从给定节点访问到所有节点。

解题思路

  1. 创建一个哈希表,用于存储已访问过的节点及其对应的克隆节点。
  2. 使用深度优先搜索(DFS)遍历原图,对于每个节点: a. 如果该节点已经被访问过,则直接返回其克隆节点。 b. 创建一个新的克隆节点,并将其添加到哈希表中。 c. 遍历该节点的邻居列表,递归调用DFS函数,将邻居节点的克隆节点添加到克隆节点的邻居列表中。
  3. 返回克隆图的任意一个节点即可。

代码思路

  1. 首先定义了一个名为Node的类,表示图中的节点,包含一个整数值val和指向左右子节点的指针leftright,以及指向下一个节点的指针next

    class Node {public int val; // 节点的值public Node left; // 左子节点public Node right; // 右子节点public Node next; // 下一个节点
    }
    
  2. 同时,还定义了一个名为Solution的类,其中包含了两个方法:cloneGraphDFS

  3. cloneGraph方法是主要的入口点,它接受一个Node类型的参数node,表示要克隆的图的起始节点。

     // 克隆图的方法,接收一个节点作为参数public Node cloneGraph(Node node)
    
  4. 该方法首先创建一个空的哈希表visited,用于存储已经访问过的节点及其对应的克隆节点。然后调用DFS方法进行深度优先搜索,从起始节点开始遍历整个图,并返回克隆后的图的根节点。

      // 创建一个哈希表,用于存储已访问过的节点HashMap<Node, Node> visited = new HashMap<>();// 调用深度优先搜索方法进行克隆return DFS(node, visited);
    
  5. DFS方法是一个递归方法,用于实现深度优先搜索。它接受两个参数:当前节点node和一个哈希表visited,用于记录已经访问过的节点及其对应的克隆节点。

      // 深度优先搜索方法,接收一个节点和一个哈希表作为参数public Node DFS(Node node, HashMap<Node, Node> visited) 
    
  6. 在方法内部,首先判断当前节点是否为空,如果为空则直接返回。接着判断当前节点是否已经被访问过,如果已经访问过则直接返回其对应的克隆节点。然后创建一个新的克隆节点clone,并将其添加到visited哈希表中。

     // 如果节点为空,返回空节点if (node == null) {return node;}// 如果哈希表中已经存在该节点的克隆,直接返回克隆节点if (visited.containsKey(node)) {return visited.get(node);}// 创建一个新的克隆节点,值为原节点的值,邻居列表为空Node clone = new Node(node.val, new ArrayList<>());// 将原节点和克隆节点添加到哈希表中visited.put(node, clone);
    
  7. 最后,遍历当前节点的所有邻居节点,对每个邻居节点递归调用DFS方法,并将返回的克隆节点添加到当前节点的克隆节点的邻居列表中。最终返回当前节点的克隆节点。

             // 遍历原节点的邻居节点,递归调用DFS方法进行克隆,并将克隆后的邻居节点添加到克隆节点的邻居列表中for (Node n : node.neighbors) {clone.neighbors.add(DFS(n, visited));}// 返回克隆节点return clone;
    
  8. 这段代码实现了一个简单的图的深度优先搜索和克隆过程,可以用于处理一些需要复制或克隆图结构的场景。

参考代码

class Node {public int val;public Node left;public Node right;public Node next;public Node() {}public Node(int _val) {val = _val;}public Node(int _val, Node _left, Node _right, Node _next) {val = _val;left = _left;right = _right;next = _next;}
};
class Solution {public Node cloneGraph(Node node) {HashMap<Node, Node> visited = new HashMap<>();return DFS(node, visited);}public Node DFS(Node node, HashMap<Node, Node> visited) {if (node == null) {return node;}if (visited.containsKey(node)) {return visited.get(node);}Node clone = new Node(node.val, new ArrayList<>());visited.put(node, clone);for (Node n : node.neighbors) {clone.neighbors.add(DFS(n, visited));}return clone;}
}

最接近的三数之和

题目

给定一个包括 n 个整数的数组 nums 和 一个目标值 target。找出 nums 中的三个整数,使得它们的和与 target 最接近。返回这三个数的和。假定每组输入只存在唯一答案。

示例:

输入:nums = [-1,2,1,-4], target = 1
输出:2
解释:与 target 最接近的和是 2 (-1 + 2 + 1 = 2) 。

提示:

3 <= nums.length <= 10^3
-10^3 <= nums[i] <= 10^3
-10^4 <= target <= 10^4

解题思路

  1. 首先对数组进行排序,这样可以方便后续的查找操作。
  2. 使用两层循环遍历数组中的每两个数的组合,计算它们的和,记为twoSum
  3. 对于每个twoSum,我们需要找到一个与它最接近的数,使得这三个数的和与目标值target最接近。我们可以使用递归的方法来实现这个功能。具体来说,我们可以定义一个函数oneSumCloset,它接受五个参数:数组nums、两个整数ij表示当前考虑的两个数在数组中的位置、一个整数startend表示在数组中查找的范围、以及目标值target
  4. oneSumCloset函数中,我们首先处理边界情况。如果start等于ij,则将start加1;如果end等于ij,则将end减1。接下来,我们判断startend的关系。如果它们相等,说明已经找到了与target最接近的数,直接返回该数即可。如果它们相差为1,说明只有一个数与target最接近,比较这两个数与target的差值,返回差值较小的那个数。否则,我们计算中间位置middle,并根据nums[middle]target的大小关系来更新查找范围。然后递归调用oneSumCloset函数,继续查找。
  5. 在主函数threeSumClosest中,我们初始化最小值为0,并设置一个布尔变量hasMin为false,用于标记是否已经找到了与目标值最接近的三个数。然后使用两层循环遍历数组中的每两个数的组合,计算它们的和,记为twoSum。接着,我们调用oneSumCloset函数,找到与twoSum最接近的数,记为restClost。最后,我们计算这三个数的和newValue,并与当前最小值进行比较。如果找到了更接近目标值的三个数,更新最小值和hasMin
  6. 当两层循环结束后,返回最小值minValue即可。

代码思路

它包含两个方法:oneSumCloset和threeSumClosest。

  1. oneSumCloset方法:这个方法用于在一个有序数组中找到与给定目标值最接近的数。它接受6个参数:一个整数数组nums,两个整数i和j,表示要搜索的范围,以及三个整数start、end和target,分别表示搜索的起始位置、结束位置和目标值。

  2. 方法首先检查start和end是否等于i或j,如果是,则将它们分别加1或减1。

     // 在数组nums中,从i到j范围内查找与target最接近的数int oneSumCloset(int[] nums, int i, int j, int start, int end, int target) {if (start == i || start == j) // 如果start等于i或j,则将start加1start = start + 1;
    
  3. 然后,根据start和end的值进行不同的处理。如果start和end相等,说明已经找到了与目标值最接近的数,直接返回该数。如果start和end相差1,说明只有一个数与目标值最接近,比较这两个数与目标值的差值,返回差值较小的那个数。否则,计算start和end的中间位置middle,并根据nums[middle]与目标值的大小关系,更新start或end的值。

    if (end == i || end == j) // 如果end等于i或j,则将end减1end = end - 1;if (start == end) { // 如果start等于end,说明已经找到了与target最接近的数,直接返回该数return nums[start];} else if (end == start + 1 || end == start - 1) { // 如果end等于start+1或start-1,说明只有一个数与target最接近,比较这两个数与target的差值,返回差值较小的那个数if (Math.abs(nums[end] - target) > Math.abs(nums[start] - target)) {return nums[start];} else {return nums[end];}} else { // 如果以上条件都不满足,说明需要继续查找int middle = (int) Math.floor((start + end) / 2); // 计算中间位置if (nums[middle] > target) { // 如果中间位置的数大于target,说明目标数在左半部分,将end更新为middleend = middle;} else { // 如果中间位置的数小于等于target,说明目标数在右半部分,将start更新为middlestart = middle;}
    
  4. 最后,递归调用oneSumCloset方法,继续在更新后的范围内查找与目标值最接近的数。

       return oneSumCloset(nums, i, j, start, end, target); // 递归调用oneSumCloset方法,继续查找
    
  5. threeSumClosest方法:这个方法用于在一个有序数组中找到三个数之和最接近给定目标值的组合。它接受两个参数:一个整数数组nums和一个整数target,分别表示输入的数组和目标值。

  6. 方法首先对数组nums进行排序。然后,使用两层循环遍历数组中的每对数,计算它们的和作为twoSum,并计算剩余的目标值rest。接着,调用oneSumCloset方法,在剩余的数组元素中查找与rest最接近的数作为restClost。

     // 在数组nums中,查找三个数之和最接近target的组合public int threeSumClosest(int[] nums, int target) {Arrays.sort(nums); // 对数组进行排序int minValue = 0; // 初始化最小值为0boolean hasMin = false; // 初始化hasMin为false,表示还没有找到最小值for (int i = 0; i < nums.length - 2; i++) { // 遍历数组中的前n-2个数for (int j = i + 1; j < nums.length - 1; j++) { // 遍历数组中的第i+1个数到倒数第二个数int twoSum = nums[i] + nums[j]; // 计算两个数之和int rest = target - twoSum; // 计算剩余的目标值int restClost = oneSumCloset(nums, i, j, j + 1, nums.length - 1, rest); // 在剩余的数组元素中查找与rest最接近的数
    
  7. 将restClost与twoSum相加得到新的值newValue。接下来,判断是否已经找到了最小值minValue,如果没有找到,则将newValue赋值给minValue,并将hasMin设置为true。如果已经找到了最小值minValue,则比较newValue与minValue与目标值的差值d1和d2,如果d1大于d2,则将newValue赋值给minValue。

      int newValue = restClost + twoSum; // 计算新的值if (!hasMin) { // 如果还没有找到最小值minValue = newValue; // 更新最小值hasMin = true; // 将hasMin设置为true,表示已经找到了最小值} else { // 如果已经找到了最小值int d1 = Math.abs(minValue - target); // 计算当前最小值与目标值的差值int d2 = Math.abs(newValue - target); // 计算新值与目标值的差值if (d1 > d2) { // 如果当前最小值的差值大于新值的差值minValue = newValue; // 更新最小值}}
    
  8. 最后,返回minValue作为结果。

return minValue; // 返回最小值

参考代码

class Solution {int oneSumCloset(int[] nums, int i, int j, int start, int end, int target) {if (start == i || start == j)start = start + 1;if (end == i || end == j)end = end - 1;if (start == end) {return nums[start];} else if (end == start + 1 || end == start - 1) {if (Math.abs(nums[end] - target) > Math.abs(nums[start] - target)) {return nums[start];} else {return nums[end];}} else {int middle = (int) Math.floor((start + end) / 2);if (nums[middle] > target) {end = middle;} else {start = middle;}return oneSumCloset(nums, i, j, start, end, target);}}public int threeSumClosest(int[] nums, int target) {Arrays.sort(nums);int minValue = 0;boolean hasMin = false;for (int i = 0; i < nums.length - 2; i++) {for (int j = i + 1; j < nums.length - 1; j++) {int twoSum = nums[i] + nums[j];int rest = target - twoSum;int restClost = oneSumCloset(nums, i, j, j + 1, nums.length - 1, rest);int newValue = restClost + twoSum;;if (!hasMin) {minValue = newValue;hasMin = true;} else {int d1 = Math.abs(minValue - target);int d2 = Math.abs(newValue - target);if (d1 > d2) {minValue = newValue;}}}}return minValue;}
}

求公式的值

题目

求 1-1/2!-1/3! -… -1/10!

解题思路

  1. 首先,我们需要计算阶乘。在给定的代码中,已经有一个名为jiecheng的方法用于计算阶乘。我们可以使用这个方法来计算1-1/2!-1/3! -… -1/10!的值。
  2. sum方法中,我们需要遍历从1到n的所有整数。对于每个整数i,我们需要计算1/i!并将其累加到总和中。同时,我们需要更新变量s的值,使其在每次迭代时取反。
  3. 最后,在main方法中,我们调用sum方法并传入参数n=10,然后输出结果。

代码思路

这段代码是一个Java程序,用于计算一个数列的和。该数列由以下公式生成:

sum = 1/n! + (-1)^n / (n-1)! + ... + 1/1!
  1. 程序中定义了两个方法:jiechengsum

  2. jiecheng方法用于计算阶乘(factorial),即从1到给定整数n的所有正整数的乘积。它使用了一个循环来逐个乘以从1到n的整数,并将结果存储在变量s中。最后,该方法返回计算得到的阶乘值。

    // 定义一个静态方法jiecheng,接收一个整数n作为参数,返回n的阶乘public static double jiecheng(int n) {double s = 1; // 初始化变量s为1// 使用for循环计算n的阶乘for (int i = 1; i <= n; i++) {s *= i; // 将s乘以i}return s; // 返回计算结果
    
  3. sum方法用于计算数列的和。它使用了一个循环来迭代从1到n的每个整数。在每次迭代中,它将当前项的值累加到变量sum中。每一项的值由公式1/n! + (-1)^n / (n-1)!计算得出。其中,(-1)^n表示交替的符号,(n-1)!表示n-1的阶乘。最后,该方法返回计算得到的数列和。

    // 定义一个静态方法sum,接收一个整数n作为参数,返回数列的和public static double sum(int n) {double sum = 0.0; // 初始化变量sum为0.0int s = 1; // 初始化变量s为1// 使用for循环计算数列的和for (int i = 1; i <= n; i++) {sum += s / jiecheng(i); // 将s除以i的阶乘加到sum上s = -s; // 将s取反}return sum; // 返回计算结果}
    
  4. main方法中,程序将整数n设置为10,并调用sum方法来计算数列的和。然后,它将结果打印到控制台上。

// 定义一个主方法main,程序的入口点public static void main(String[] args) throws Exception {int n = 10; // 定义变量n为10double ss = sum(n); // 调用sum方法计算数列的和,并将结果赋值给变量ssSystem.out.println(ss); // 输出数列的和}

参考代码

public class TEST {public static double jiecheng(int n) {double s = 1;for (int i = 1; i <= n; i++)s *= i;return s;}public static double sum(int n) {double sum = 0.0;int s = 1;for (int i = 1; i <= n; i++) {sum += s / jiecheng(i);s = -s;}return sum;}public static void main(String[] args) throws Exception {int n = 10;double ss = sum(n);System.out.println(ss);}
}

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

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

相关文章

Python算法:动态规划解决0-1背包问题

动态规划&#xff08;Dynamic Programming&#xff0c;简称DP&#xff09;是一种在数学、计算机科学和经济学中使用的&#xff0c;通过把原问题分解为相对简单的子问题的方式求解复杂问题的方法。动态规划常常适用于有重叠子问题和最优子结构性质的问题&#xff0c;它能够将问题…

Spark 读取ES采坑系列

目录 一、使用的插件 二、ES集群和Elasticsearch-hadoop版本问题 三、Elasticsearch-hadoop 和Scala版本以及Spark版本&#xff08;版本不匹配会有各种异常信息 一、使用的插件 <dependency><groupId>org.elasticsearch</groupId><artifactId>elas…

java入坑之类加载器

一、类加载机制 1.1类加载过程 类加载是Java虚拟机将类的字节码数据从磁盘或网络中读入内存&#xff0c;并转换成在JVM中可以被执行的Java类型的过程。类加载器是Java虚拟机的重要组成部分&#xff0c;负责加载和解析类的字节码&#xff0c;将其转换成Java虚拟机中的类对象&am…

nav2 调节纯追踪算法

纯追踪算法 纯追踪基础 The core idea is to find a point on the path in front of the robot and find the linear and angular velocity to help drive towards it. 核心思想是在机器人前方的路径上找到一个点&#xff0c;并找到一个合适的线速度和角速度&#xff0c;以驱…

[量化投资-学习笔记007]Python+TDengine从零开始搭建量化分析平台-布林带

布林带&#xff08;Bollinger Bands&#xff09;也称为布林通道、保力加通道&#xff0c;是由约翰布林格&#xff08;John Bollinger&#xff09;发明的技术分析指标。布林通道通常被用来确认资产价格波动范围。 布林通道是由三条平滑的曲线组成的趋势线图表&#xff0c;中线为…

leetcode刷题 - SQL - 中等

1. 176. 第二高的薪水 筛选出第二大 查询并返回 Employee 表中第二高的薪水 。如果不存在第二高的薪水&#xff0c;查询应该返回 null(Pandas 则返回 None) 。查询结果如下例所示。 666中等的第一题就上强度 强行解法 select max(salary) as SecondHighestSalary from Emp…

深入解析 Redis 分布式锁原理

一、实现原理 1.1 基本原理 JDK 原生的锁可以让不同线程之间以互斥的方式来访问共享资源&#xff0c;但如果想要在不同进程之间以互斥的方式来访问共享资源&#xff0c;JDK 原生的锁就无能为力了。此时可以使用 Redis 来实现分布式锁。 Redis 实现分布式锁的核心命令如下&am…

【Git】如何安装git,项目中使用git上传到远程仓库,使用git中对多人使用出现的版本问题的解决

前言&#xff1a; 一&#xff0c;Git的介绍&#xff0c;安装&#xff0c;与SVN的对比 1.1Git的介绍 Git 是一个开源的分布式版本控制系统&#xff0c;用于敏捷高效地处理任何或小或大的项目。 Git 是 Linus Torvalds 为了帮助管理 Linux 内核开发而开发的一个开放源码的版本控…

ubuntu下Anaconda环境安装GPU的pytorch(docker镜像)

实验室需要给每个人分配docker的container环境&#xff0c;为了节省系统的空间&#xff0c;打算把anaconda和深度学习的开发环境配置好拉取镜像以省时间。 基础环境配置 apt更新了清华源 安装了基础环境 gcc vim Linux文本编辑库 openssh-server ssh远程连接库 net-tools 包含…

关于Android Studio中开发Flutter配置

配置系统环境变量&#xff1a;path下 &#xff0c;flutter的bin目录下 File->Settings->Languages&Frameworks->FlutterFile->Settings->Languages&Frameworks->DartFile->Settings->Languages&Frameworks->Android SDK 确认是…

接口开发之使用C#插件Quartz.Net定时执行CMD任务工具

C#制作定时任务工具执行CMD命令 概要准备知识点实现原理thinkphp配置winform执行CMD命令读取ini配置文件定时任务Quartz.Net 完整代码Job.csIniFunc.csForm1.csconfig.ini简易定时任务工具雏形 概要 很多时候写接口上线后还会遇到很多修改&#xff0c;类似JAVA,C#,delphi制作的…

面试字节测开岗失败后,被面试官在朋友圈吐槽了......(心累)

在和朋友吃夜宵的时候&#xff0c;朋友向我吐槽说自己在参加某大厂测试面试的时候被面试官怼得哑口无言&#xff0c;场面让他一度十分尴尬 印象最深的就是下面几个问题&#xff1a; 根据你以前的工作经验和学习到的测试技术&#xff0c;说说你对质量保证的理解&#xff1f; 非…

动态规划-构建乘积数组

** 描述 给定一个数组 A[0,1,…,n-1] ,请构建一个数组 B[0,1,…,n-1] ,其中 B 的元素 B[i]A[0]A[1]…*A[i-1]A[i1]…*A[n-1]&#xff08;除 A[i] 以外的全部元素的的乘积&#xff09;。程序中不能使用除法。&#xff08;注意&#xff1a;规定 B[0] A[1] * A[2] * … * A[n-1…

GeoGebra:数学动画制作工具重磅来袭

【线性代数】线性代数可视化工具&#xff1a;manim manim是之前我跟大家分享的一个线性代数动画制作工具。 但我之前的描述有些许偏差&#xff0c;这里要更正一下&#xff0c;manim不仅限于制作线性代数动画&#xff0c;也可以制作数学其他学科的动画&#xff0c;例如微积分&…

[极客大挑战 2019]BuyFlag 1(两种解法)

题目环境&#xff1a; FLAG NEED YOUR 100000000 MONEY flag需要你的100000000元 F12瞅瞅源代码&#xff1a; if (isset($_POST[password])){ $password $_POST[password]; if (is_numeric($password)) { echo "password cant be number" } elseif ($pas…

理解MySQL的日志 Redo、Undo

理解MySQL的Redo日志和Undo日志 1、MySQL 日志文件解决的问题2、redo 日志2.1、redo log 的组成2.2、redo log 刷盘策略2.3、MySQL 的 redo log解决了哪些问题 3、undo 日志3.1、undo 日志作用3.2、undo log 的类型3.3、undo log 的生命周期3.4、事务回滚相关的几个隐藏字段 1、…

使用easyui前端框架快速构建一个crud应用

本篇文章将会详细介绍jquery easyui前端框架的使用&#xff0c;通过创建一个crud应用来带大家快速掌握easyui的使用。 easyui是博主最喜欢的前端框架&#xff0c;没有之一&#xff0c;因为它提供了多种主题&#xff0c;而且有圆润的各种组件。 目录 一、快速开始 二、准备工作…

JS逆向爬虫---请求参数加密③【比特币交易爬虫】

查询参数确定 t无加密 请求头参数加密 X-Apikey参数加密确定 X-Apikey逆向 const API_KEY "a2c903cc-b31e-4547-9299-b6d07b7631ab" function encryptApiKey(){ var t API_KEY, e t.split(""), n e.splice(0, 8);return t e.concat(n).join("&…

利用Ansible实现批量Linux服务器安全配置

1.摘要 在上一篇<<初步利用Ansible实现批量服务器自动化管理>>文章中, 我初步实现了通过编写清单和剧本来实现多台服务器的自动化管理,在本章节中, 我将利用Ansible的剧本来实现更实用、更复杂一点的功能, 主要功能包括三个:1.同时在三台服务器中增加IP访问控制,只…

万宾科技智能井盖,实现对井盖的监测

随着人工智能和物联网技术的不断变化&#xff0c;各种适用于市政府提高管理能力和公共服务水平的高科技产品不断更新。在道路基础设施建设过程中&#xff0c;智能井盖传感器的出现时刻保护着城市地下生命线&#xff0c;而且可以对地下水道井盖进行实时的监测并完成数据上传等工…