【leetcode】二分查找本质

news2024/9/22 9:53:21

标题:【leetcode】二分查找本质

@水墨不写bug


正文开始:(点击题目标题转跳到OJ

目录

(O)前言*

(一) 在排序数组中查找元素的第一个和最后一个位置

思路详解: 

参考代码: 

(二)x的平方根​​​​​​

(三)搜索插入位置

(四)山峰数组的峰顶索引

(五)寻找峰值

(六)寻找旋转排序数组的最小值

(七)点名


(O)前言*

        “二分思想”相信看过我的这篇文章《小白鼠试毒——二分法怎么分》的朋友都对其有深刻的理解,具体来说:当我们可以根据一定的标准将元素分为两部分,并根据标准判断哪一部分留下,哪一部分舍去,这时候就可以使用二分。 

        如果我们加上一些限制:考虑题目中给的数组的元素,那么:如果我们可以判断数组具有“二段性”,那么就可以使用二分。 什么是二段性?其实,就是存在一个标准,根据这个标准,我们可以把数组分为“是”和“不是”的两部分,一般这两部分是连续的,这就表明有一个点将数组分为两部分。这就是使用二分的前提条件。

        好消息是二分拥有特定的模板,我们可以根据题目的要求接合对模板的理解来使用模板。最重要的是要理解模板的原理。不然无法正确使用模板。在本文,我会总结出二分的模板,以及如何推导和使用。

        另外,二分的特征时间复杂度是O(logN),换句话说,如果一道题的时间复杂度要求是O(logN),不要忘了试试二分的思路。


        二分判断点的选取:

        其实看似是选取二分点,即使是你随便选取一个点作为判断点,二分也是可以进行:如果你选取一个随机点作为二分下标,那么二分的效率可能会受到影响。

        最高效的二分下标选取点是数组的接近中间点的位置。根据概率论中有关数学期望的知识可以给出严格证明,这里我们不再证明。


(一) 在排序数组中查找元素的第一个和最后一个位置

给你一个按照非递减顺序排列的整数数组 nums,和一个目标值 target。请你找出给定目标值在数组中的开始位置和结束位置

如果数组中不存在目标值 target,返回 [-1, -1]

你必须设计并实现时间复杂度为 O(log n) 的算法解决此问题。

示例 1:

输入:nums = [5,7,7,8,8,10]
, target = 8
输出:[3,4]

示例 2:

输入:nums = [5,7,7,8,8,10]
, target = 6
输出:[-1,-1]

示例 3:

输入:nums = [], target = 0
输出:[-1,-1]

提示:

  • 0 <= nums.length <= 10^5
  • -10^9 <= nums[i] <= 10^9
  • nums 是一个非递减数组
  • -10^9 <= target <= 10^9

题目要求查找两个区间端点的位置,也就是左端点和右端点的位置。分两个步骤依次进行

思路详解: 

第一道题细致讲解,之后的题目只给出重要的部分思路


        记查找的值为key


        当我们查找左端点时,根据 “<key” 和 “>=key” 这个标准将区间分为两部分,左边的部分区间都是小于key的,右边的区间都是大于等于key的:

        如果区间中点mid对应的nums[mid] < key,则mid位于左半部分区间,此时更新left下标left = mid +1;(因为左边区间内的值都小于key,可以确定需要查找的目标值一定在右区间中,所以left可以向右移动)

        如果mid对应的nums[mid] >= key,则可以确定mid位于目标值右侧,此时更新查找区域右端点为right = mid;(因为区间右边的值是大于等于key的,可以确定要查找的目标位于左区间内 + mid对应的位置——因为nums[mid]此时等于key,由于无法确定目标值是否在mid位置,如果此时冒然right = mid-1,那么如果mid位置就是目标值的这种情况就被忽略了!)

        当我们查找区间右端点时,根据“<=key” “>key”的二段性将区间分为两部分,左部分的值小于key,右部分的值大于等于key。

        如果区间中点mid对应值nums[mid] <= key 则mid位于则mid在key的左边+mid这个位置,更新left = mid;

        如果区间中点mid对应值nums[mid] > key ,则mid位于key的右边,更新right = right-1;

循环的终止条件:

left < right  

为什么?

对于要查找的数组分三种情况考虑:

        1.有目标值

        

        2.都大于目标值

        3.都小于目标值

在讲解之前,先引入mid的选取规则:

        1.偏左选取——mid = left + (right - left)/2;

        2.偏右选取——mid = left + (right - left + 1)/2;

        他们的区别就是当区间内有偶数个值时,mid选取的是中间偏左的一个还是中间偏右的一个。

        结论:

        在更新区间的时候,无法越过mid的一侧需要选取偏另一侧的mid选取规则。

        为什么?

当区间仅有两个元素的时候,区间端点的选取无非就只有两个情况——偏左的元素或者偏右的元素。在查找区间左端点时,right无法越过mid,正确的mid选取是选取偏左的元素。如果我们这时mid选取偏右的元素,那么由于nums[mid] >= key还是更新 right = mid,这样下去会死循环。

接下来讲解循环终止条件的选择原因:

        对于要查找的数组分三种情况考虑:

        1.有目标值

        left和right都在向目标值靠拢,当left==right时,此时就是目标值,不需要进入循环。

        2.都大于目标值

        right向左移动,当left==right时,此时不需要进入循环,跳出循环判断一下相遇时的值的与目标值的大小就可以了。

        3.都小于目标值

        left向右移动,当left==right时,此时不需要进入循环,跳出循环判断一下相遇时的值的与目标值的大小就可以了。

参考代码: 

class Solution {
public:
    vector<int> searchRange(vector<int>& nums, int target) 
    {
        if(nums.size()==0) return {-1,-1};
        int left = 0,right = nums.size()-1;
        //查找左区间
        int ret_l = -1,ret_r = -1;
        while(left<right)
        {
            int mid = left+(right-left)/2;
            if(nums[mid]<target) left = mid+1;
            else right = mid;
        }
        if(nums[left] == target) ret_l = left;
        left = 0,right = nums.size()-1;
        //查找右区间
        while(left<right)
        {
            int mid = left+(right-left+1)/2;
            if(nums[mid]<=target) left = mid;
            else right = mid-1;
        }
        if(nums[right] == target) ret_r = right;
        return {ret_l,ret_r};
    }
};

二分查找模板:

        查找右区间端点:

int left = ..., right = ...;
while (left < right)
{
	int mid = left + (right - left) / 2;
	if (...)
		left = mid + 1;
	else
		right = mid;
}

        查找左区间端点:

int left = ..., right = ...;
while (left < right)
{
	int mid = left + (right - right + 1) / 2;
	if (...)
		left = mid;
	else
		right = mid - 1;
}

        通过观察两个模板,可以发现规律:       

        循环的终止条件相同;

        if条件中有“-1”,mid有“+1”;

有这两个规律就足够了。


(二)x的平方根​​​​​​

给你一个非负整数 x ,计算并返回 x 的 算术平方根 。

由于返回类型是整数,结果只保留 整数部分 ,小数部分将被 舍去 。

注意:不允许使用任何内置指数函数和算符,例如 pow(x, 0.5) 或者 x ** 0.5 。

示例 1:

输入:x = 4
输出:2

示例 2:

输入:x = 8
输出:2
解释:8 的算术平方根是 2.82842..., 由于返回类型是整数,小数部分将被舍去。

提示:

  • 0 <= x <= 2^31 - 1

         根据题意:

        x < 1 时,根下x小于1,去除小数部分,返回0;

        x > 1 时1 < 根下x  < x   使用二分,首先要寻找二段性

                对于从1开始向后数,每一个数的平方是这样的一组数:

        1   4   9   16   25   36   49   64   81...

        假设x 是26,由于需要将小数部分舍去,得到的结果是5,但是(这里用数学的区间表示)对于[25,36)内的任何一个数,得到的结果都是5,这就相当于一个程序输入[25,36),输出同一个值,于是这就转化为第一道题查找区间左端点的问题了。

        二段性:

        先不考虑mid的左偏还是右偏,mid是一个大概的区间中点的位置。(对于mid的位置左偏和右偏问题可以通过模板来解决

        我们首先看二段性:

        根据  mid*mid <= x 或者 mid*mid > x 分类; 

                mid*mid <= x,说明target在mid右侧+mid现在的位置,更新left = mid;

                mid*mid > x ,说明target在target在mid左侧,更新right = mid -1;

        根据模板

        下面有 “-1”,则mid有“+1”:mid = left + (right - left + 1)/2;

        或者由于left无法越过mid,所以要mid向右偏,需要“+1”。

class Solution {
public:
    int mySqrt(int x) {
        if(x < 1) 
            return 0;
        int left = 1,right = x;
        while(left < right)
        {
            long long mid = left + ( right - left + 1 ) / 2;
            if(mid*mid<=x) 
                left = mid;
            else 
                right = mid-1;
        }
        return left;
    }
};


(三)搜索插入位置

        给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。

        请必须使用时间复杂度为 O(log n) 的算法。

示例 1:

输入: nums = [1,3,5,6], target = 5
输出: 2

示例 2:

输入: nums = [1,3,5,6], target = 2
输出: 1

示例 3:

输入: nums = [1,3,5,6], target = 7
输出: 4

提示:

  • 1 <= nums.length <= 10^4
  • -10^4 <= nums[i] <= 10^4
  • nums 为 无重复元素 的 升序 排列数组
  • -10^4 <= target <= 10^4

二段性:

        数组分为< target  和 >= target 两部分,这就类似于查找区间右端点了。

        需要处理的特殊情况:

        当我们要插入的值在数组的最后时,由于left和right会最终停在最后一个位置上,但是我们要返回的下标是停下的下一个位置,所以需要特殊处理。

参考代码:

class Solution {
public:
    int searchInsert(vector<int>& nums, int target) 
    {
        int left = 0,right = nums.size()-1;
        while(left<right)
        {
            int mid = left+(right-left)/2;
            if(nums[mid]>=target) right = mid;
            else left = mid+1;
        }
        if(nums[left] < target) return left+1;//说明插入在最后一个数据后面
        return left;//left的位置即为插入位置
    }
};

什么?你问我为什么不是分为<=  和 >?

        首先,通过测试,将区间分为<= 和 > 是可以 凑巧 通过oj的:


int searchInsert(vector<int>& nums, int target)
{
    int left = 0, right = nums.size() - 1;
    while (left < right)
    {
        int mid = left + (right - left + 1) / 2;
        if (nums[mid] > target) right = mid - 1;
        else left = mid;
    }
    if (nums[left] < target) return left + 1;//说明插入在最后一个数据后面
    return left;//left的位置即为插入位置
}
int searchInsert_(vector<int>& nums, int target)
{
    int left = 0, right = nums.size() - 1;
    while (left < right)
    {
        int mid = left + (right - left) / 2;
        if (nums[mid] >= target) right = mid;
        else left = mid + 1;
    }
    if (nums[left] < target) return left + 1;//说明插入在最后一个数据后面
    return left;//left的位置即为插入位置
}

int main()
{
    vector<int> v = { 1,2,4,5,6 };
    int ret =  searchInsert(v, 3);
    int ret_ =  searchInsert_(v, 3);
    cout << ret << endl;
    cout << ret_ << endl;
	return 0;
}

结果:

2
2

具体为什么,我们分析一下: 

        我们审题,如果target存在,能够找到target,返回target下标即可;(两种分法都是一样的)

        如果target不存在,因为我们在一个位置插入是在当前位置插入,<=  和 >分法  查找的是我们目标插入位置的前一个位置,只不过最后一句对特殊情况的处理发挥了本来它不该发挥的作用,导致结果凑巧正确。(这是不该发生的,是意料之外的情况)

        如果把“<=  和 >分法”的最后的对特殊情况的处理删掉,结果:

1
2

        所以“<=  和 >分法”是错误的。


(四)山峰数组的峰顶索引

        符合下列属性的数组 arr 称为 山脉数组 :

  • arr.length >= 3
  • 存在 i0 < i < arr.length - 1)使得:
    • arr[0] < arr[1] < ... arr[i-1] < arr[i]
    • arr[i] > arr[i+1] > ... > arr[arr.length - 1]

        给你由整数组成的山脉数组 arr ,返回满足 arr[0] < arr[1] < ... arr[i - 1] < arr[i] > arr[i + 1] > ... > arr[arr.length - 1] 的下标 i 。

        你必须设计并实现时间复杂度为 O(log(n)) 的解决方案。

示例 1:

输入:arr = [0,1,0]
输出:1

示例 2:

输入:arr = [0,2,1,0]
输出:1

示例 3:

输入:arr = [0,10,5,2]
输出:1

提示:

  • 3 <= arr.length <= 10^5
  • 0 <= arr[i] <= 10^6
  • 题目数据保证 arr 是一个山脉数组

 二段性:

        根据arr[mid]    arr[mid +1] 的大小关系 作为二段性判断依据。

class Solution {
public:
    int peakIndexInMountainArray(vector<int>& arr) {
        int left = 1,right = arr.size()-2;
        while(left<right)
        {
            int mid = left +(right-left)/2;
            if(arr[mid] < arr[mid+1]) //在上升段,目标在右侧
                left = mid+1;
            else right = mid;
        }
        return left;
    }
};

(五)寻找峰值

        峰值元素是指其值严格大于左右相邻值的元素。

        给你一个整数数组 nums,找到峰值元素并返回其索引。数组可能包含多个峰值,在这种情况下,返回 任何一个峰值 所在位置即可。

        你可以假设 nums[-1] = nums[n] = -∞ 。

        你必须实现时间复杂度为 O(log n) 的算法来解决此问题。

示例 1:

输入:nums =[1,2,3,1]
输出:2
解释:3 是峰值元素,你的函数应该返回其索引 2。

示例 2:

输入:nums =[1,2,1,3,5,6,4]
输出:1 或 5 
解释:你的函数可以返回索引 1,其峰值元素为 2;
     或者返回索引 5, 其峰值元素为 6。

提示:

  • 1 <= nums.length <= 1000
  • -2^31 <= nums[i] <= 2^31 - 1
  • 对于所有有效的 i 都有 nums[i] != nums[i + 1]

 二段性:

        根据nums[mid]和nums[mid+1]的大小关系为二段性标准。

class Solution {
public:
    int findPeakElement(vector<int>& nums) {
        int left = 0,right = nums.size()-1;
        while(left<right)
        {
            int mid = left+(right-left)/2;
            if(nums[mid]<nums[mid+1]) left = mid+1;
            else right = mid;//不存在等于的情况
        }
        return left;
    }
};


(六)寻找旋转排序数组的最小值

 

        已知一个长度为 n 的数组,预先按照升序排列,经由 1 到 n 次 旋转 后,得到输入数组。例如,原数组 nums = [0,1,2,4,5,6,7] 在变化后可能得到:

  • 若旋转 4 次,则可以得到 [4,5,6,7,0,1,2]
  • 若旋转 7 次,则可以得到 [0,1,2,4,5,6,7]

        注意,数组 [a[0], a[1], a[2], ..., a[n-1]] 旋转一次 的结果为数组 [a[n-1], a[0], a[1], a[2], ..., a[n-2]] 。

        给你一个元素值 互不相同 的数组 nums ,它原来是一个升序排列的数组,并按上述情形进行了多次旋转。请你找出并返回数组中的 最小元素 。

        你必须设计一个时间复杂度为 O(log n) 的算法解决此问题。

示例 1:

输入:nums = [3,4,5,1,2]
输出:1
解释:原数组为 [1,2,3,4,5] ,旋转 3 次得到输入数组。

示例 2:

输入:nums = [4,5,6,7,0,1,2]
输出:0
解释:原数组为 [0,1,2,4,5,6,7] ,旋转 3 次得到输入数组。

示例 3:

输入:nums = [11,13,15,17]
输出:11
解释:原数组为 [11,13,15,17] ,旋转 4 次得到输入数组。

提示:

  • n == nums.length
  • 1 <= n <= 5000
  • -5000 <= nums[i] <= 5000
  • nums 中的所有整数 互不相同
  • nums 原来是一个升序排序的数组,并进行了 1 至 n 次旋转

 根据:

        nums[mid]和nums[n-1]大小关系

class Solution {
public:
    int findMin(vector<int>& nums) 
    {
        int n = nums.size(),left = 0,right = n-1;
        while(left<right)
        {
            int mid = left+(right-left)/2;
            if(nums[mid]<nums[n-1]) right = mid;
            else left = mid+1;
        }
        return nums[left];
    }
};


(七)点名

        某班级 n 位同学的学号为 0 ~ n-1。点名结果记录于升序数组 records。假定仅有一位同学缺席,请返回他的学号。

示例 1:

输入: records = [0,1,2,3,5]
输出: 4

示例 2:

输入: records = [0, 1, 2, 3, 4, 5, 6, 8]
输出: 7

提示:

1 <= records.length <= 10000

 二段性:

        根据是否有偏移来作为二段性依据。

class Solution {
public:
    int takeAttendance(vector<int>& records) 
    {
        
        int left = 0,right = records.size()-1;
        while(left<right)
        {
            int mid = left+(right-left)/2;
            if(records[mid]==mid) left = mid+1;
            else right = mid;
        }
        if(records[left]==left) return left+1;
        return left;
    }
};

完~

未经作者同意禁止转载

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

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

相关文章

cocos2d-x安装和项目

首先多方查找资料发现教程很简洁&#xff0c;发现对自己的操作方面没多大帮助&#xff0c;后来干脆去官网&#xff0c;好像也很简洁。基于这样一个原因&#xff0c;加上我首次碰cocos2d-x&#xff0c;决定记录一下整个流程&#xff0c;解决实际操作上的疑惑。 涉及的方面&…

数模·时间序列

时间序列思维导图 时间序列分析的方法 聚焦于趋势和季节变化 时间序列预测模型 时间序列平稳性 时间序列的平稳性是时间序列分析中的一个重要概念。一个时间序列如果是平稳的&#xff0c;意味着其统计性质&#xff08;如均值、方差、自相关等&#xff09;不随时间变化。 平稳性…

[计网04] 传输层和应用层 笔记 总结 万字详解

目录 传输层概述和功能 URL 和URI 端口号划分 套接字Socket UDP&#xff08;User Datagram Protocol&#xff09; UDP首部 UDP伪首部 TCP 三报文握手和四报文挥手 TCP&#xff08;Transmission Control Protocol&#xff09; TCP首部报文格式 TCP流量控制 cwnd&…

如何在 C# ASP.NET MVC 项目中实现 Memcached?

一.介绍 在现代 Web 应用程序中&#xff0c;性能和可扩展性至关重要。提高性能的有效方法之一是使用缓存。Memcached 是一种开源、高性能、分布式内存缓存系统&#xff0c;被广泛使用。本文将引导您在 C# ASP.NET MVC 项目中实现 Memcached。 二.什么是 Memcached&#xff1f;…

SpringBoot面试高频总结01

1. 什么是SpringBoot&#xff1f; SpringBoot是一个基于Spring框架的快速开发框架&#xff0c;它采用约定大于配置&#xff0c;自动装配的方式&#xff0c;可以快速地创建独立的&#xff0c;生产级别的&#xff0c;基于Spring的应用程序。 相比于传统的Spring框架&#xff0c;S…

真实测评,霍尼韦尔、希喂、352宠物空气净化器性能对比

在快节奏的社会生活中&#xff0c;人们越来越注重精神需要&#xff0c;许多年轻人纷纷选择拥抱宠物&#xff0c;作为生活中的温馨伴侣。宠物们治愈心灵的同时也要付出一定“代价”&#xff0c;日常养护&#xff0c;如清理猫毛、管理气味以及保持宠物环境的清洁&#xff0c;都是…

Agent类型解析:AIGC在不同领域的应用与影响

目录 引言 垂直智能体&#xff08;Vertical Agent&#xff09; 水平智能体&#xff08;Horizontal Agent&#xff09; 混合智能体&#xff08;Hybrid Agent&#xff09; 结论 引言 在人工智能&#xff08;AI&#xff09;领域&#xff0c;智能体&#xff08;Agent&#xf…

更加深入Mysql-04-MySQL 多表查询与事务的操作

文章目录 多表查询内连接隐式内连接显示内连接 外连接左外连接右外连接 子查询 事务事务隔离级别 多表查询 有时我们不仅需要一个表的数据&#xff0c;数据可能关联到俩个表或者三个表&#xff0c;这时我们就要进行夺标查询了。 数据准备&#xff1a; 创建一个部门表并且插入…

JavaWeb day01-HTML入门

Web前端 课程安排 HTML、CSS简介 HTML快速入门 实现标题排版 新闻标题样式

Docker核心技术:Docker的基本使用

云原生学习路线导航页&#xff08;持续更新中&#xff09; 本文是 Docker核心技术 系列文章&#xff1a;Docker的基本使用&#xff0c;其他文章快捷链接如下&#xff1a; 应用架构演进容器技术要解决哪些问题Docker的基本使用&#xff08;本文&#xff09;Docker是如何实现的 3…

Spark调优特殊case- Task倾斜

首先我们观察下上面的stage5, Task MaxTime2.4Min, 但是stage5的整体耗时竟然可以达到55Min, 其实分区1000&#xff0c; 300个executor&#xff0c; 按照最大的TaskTime2.4Min来估算所有Task运行完成时间, 那么时间应该是- 2.4Min * 3 2.4Min 9.6Min 也就是最慢也就跑10分钟就…

【时时三省】(C语言基础)函数和数组

山不在高&#xff0c;有仙则名。水不在深&#xff0c;有龙则灵。 ——csdn时时三省 函数 跟数学里面的函数很相似 数组 一组相同类型的元素的集合 比如把5个整形1-5存起来 int arr&#xff3b;10&#xff3d;&#xff1d;&#xff5b;1&#xff0c;2&#xff0c;3&#x…

StyleGAN——定制人脸生成思路

定制人脸生成思路 控制生成码 ( Z ) 的分布范围&#xff1a;适合粗略控制生成图像的主要特征&#xff08;如性别、人种、年龄&#xff09;&#xff0c;方法简单但精度较低。特定维度特征的替换&#xff1a;适用于细致控制图像的特征&#xff08;如皮肤颜色、发型&#xff09;&…

贪吃蛇超精讲(C语言)

前言 如果你还是个萌新小白&#xff0c;那么该项目的攻克过程一定会十分艰难。虽然作者已经将文章尽可能写的逻辑清晰&#xff0c;内容详细。但所谓“纸上得来终觉浅”&#xff0c;在讲到陌生结构和函数时&#xff0c;大家请一定自己动手去敲一遍代码&#xff0c;这很重要&…

Opencv学习项目3——人脸识别

之前我们获取了一张图像的人脸信息&#xff0c;现在我们来使用特征点分析来匹配两张lyf照片的相似度 获取两张图片的人脸信息 import cv2 import face_recognition# 加载图像文件 img1 face_recognition.load_image_file(lyf1.png) img2 face_recognition.load_image_file(l…

昇思25天学习打卡营第13天 | 模型训练

深入理解模型训练的关键步骤 在深入学习深度学习模型训练的过程中&#xff0c;我获得了宝贵的经验和理解&#xff0c;尤其是在构建数据集、定义模型、调整超参数以及实际的训练和评估过程中。以下是我对这些关键步骤的一些心得体会。 1. 构建数据集 数据集的构建是模型训练成…

(11)Python引领金融前沿:投资组合优化实战案例

1. 前言 本篇文章为 Python 对金融的投资组合优化的示例。投资组合优化是从一组可用的投资组合中选择最佳投资组合的过程&#xff0c;目的是最大限度地提高回报和降低风险。 投资组合优化是从一组可用的投资组合中选择最佳投资组合的过程&#xff0c;目的是最大限度地提高回报…

第四周:机器学习笔记

第四周学习周报 摘要Abstract机器学习任务攻略1.loss on training data1.1 training data的loss过大怎么办&#xff1f;1.2 training data的loss小&#xff0c;但是testing data loss大怎么办&#xff1f; 2. 如何选择一个中最好的模型&#xff1f;2.1 Cross Validation&#x…

前端调试技巧:动态高亮渲染区域

效果&#xff1a; 前端界面的渲染过程、次数&#xff0c;会通过高亮变化来显示&#xff0c;通过这种效果排除一些BUG 高亮 打开方式 F12进入后点击ESC&#xff0c;进入rendering&#xff0c;选择前三个即可&#xff08;如果没有rendering&#xff0c;点击橘色部分勾选上&…

docker compose 容器 编排分组

遇到问题&#xff1a;执行docker compose up -d 后docker compose 创建的容器们 在desktop-docker 中都在docker下一堆 搜索想着能不能把这个docker名字改一下&#xff0c;但是都没有找到这样的一个方案&#xff1b; 最后发现&#xff0c;我执行docker compose up -d 命令所在…