java - 七大比较排序 - 详解

news2025/1/23 21:32:39

前言

本篇介绍了七大比较排序,直接插入排序,希尔排序,冒泡排序,堆排序,选择排序,快速排序,归并排序,一些简单思想+代码实现,如有错误,请在评论区指正,让我们一起交流,共同进步!


文章目录

  • 前言
  • 1. 直接插入排序
  • 2. 希尔排序
  • 3. 冒泡排序
  • 4. 堆排序 - 基于完全二叉树
  • 5. 选择排序
  • 6 快速排序:
    • 6.2 非递归快排
  • 7. 归并排序
  • 总结

本文开始

1. 直接插入排序

思路 : 一组数据, 当插入下标为i时的数据, 比较前面i-1个数据, 如果前面i-1个数据中,有数据大于插入为i位置下标的数据, i-1位置数据, 向后移动, 直至i-1个数据中没有小于i下标的值, 再将i下标的值插入j+1位置;
时间复杂度: O(N^2)
空间复杂度: O(1)
稳定性: 稳定
直接插入代码实现:

 public static void insertSort(int[] array) {
        //默认第一个有序不用排序, 从第二个开始排序
        for (int i = 1; i < array.length; i++) {
            int tmp = array[i];//作为判断标准,作为插入元素
            int j = i-1; //循环完,还需要使用到j下标, 所以拿出来定义
            for (; j >= 0; j--) {
                //判断tmp 是否 大于下标为j的元素
                if(array[j] > tmp) {
                    //小于就前移一位
                    array[ j+1] = array[ j ];
                }else {
                    //插入值大于插入值前面的元素. 直接跳出
                    //array[ j+1 ] = tmp;
                    break;
                }
            }
            //比较完后,需要将插入值放到j+1位置
            array[j+1] = tmp;
        }
    }

在这里插入图片描述

2. 希尔排序

思想: 希尔排序, 直接插入排序的优化版本; 每组分组, 组数任意,这再进行排序(插入排序); 这里可以将数据除2依次分组;
数据分组不一定是相邻的,假设10个元素分5组,每组两个元素,在5位置的元素组数减5就是0位置,就是跟5一组的另一个元素;

以上可得到, 一个位置加减组数,可以得到一组的另外元素位置
平均时间复杂度: O(n^1.3)
空间复杂度: O(1)

希尔排序代码实现:

public static void shellSort(int[] array) {
        int gap = array.length;//初始化组数, 每个元素为一组,在除2分组
        while (gap > 1) {
            //shell排序
            shell(array,gap);
            //分组
            gap /= 2;
        }
        //第一次gap越界没有排, 此时以整体为一组,进行排序
        shell(array,gap);
    }
    
    public static void shell(int[] array,int gap) {
        //gap => 分几组的组数
        // 从第gap个开始排序
        for (int i = gap; i < array.length; i++) {
            int tmp = array[i];//作为判断标准,作为插入元素
            int j = i-gap; //循环完,还需要使用到j下标, 所以拿出来定义
            for (; j >= 0; j--) {
                //判断tmp 是否 大于下标为j的元素
                if(array[j] > tmp) {
                    //小于就前移gap位
                    array[j+gap] = array[j];
                }else {
                    //插入值大于插入值前面的元素. 直接跳出
                    break;
                }
            }
            //比较完后,需要将插入值放到j+1位置
            array[j+gap] = tmp;
        }
    }

3. 冒泡排序

思路: 升序为例,遍历数组, 比较两个相邻的值, 如果左位置大于右位置,就交换两个的位置, 一直比较数组结束;
时间复杂度: O(n^2)
空间复杂度: O(1)
稳定性: 稳定

冒泡排序优化实现 :
定义一个标志,某趟排序标志没有改变, 证明数组已经有序,不需要继续排序了,直接跳出即可;
发现每趟冒泡排序都会确定一个位置, 让每趟排序少1次比较;

public static void bubbleSort(int[] array) {
        //arrat.length个元素, 跑arrat.length-1趟
        //定义一个标志,某趟排序,标志没有改变, 证明数组已经有序,不需要继续排序了,直接跳出即可
        boolean flag = false; 
        for (int i = 0; i < array.length - 1; i++) {
            for (int j = 0; j < array.length - 1 - i; j++) {
                if(array[j] > array[j+1]) {
                    //交换
                    swap(array,j,j+1);
                    flag = true;
                }
            }
            if(flag == false) {
                break;
            }
        }
    }

4. 堆排序 - 基于完全二叉树

思路: 创建一个堆, 以大根堆为例, 要获取升序数组,只知道堆顶是最大,不知道左右孩子谁大, 需要堆排;
将堆顶与尾元素交换, 再向下调整为大根堆, 使尾元素下标-1, 循环执行此过程直至尾元素减为0;

时间复杂度: O(n*logn)
空间复杂度: O(logn)
稳定性: 不稳定

堆排序代码实现:

 public static void heapSort(int[] array) {
        //堆排先有堆,建立一个大根堆
        creatHeap(array);
        //大根堆的左右不能确定谁大谁小,所以需要堆排
        int end = array.length-1;
        while (end > 0) {
            //首尾元素交换
            swap(array,0,end);
            //此时保持大根堆,需要重新向下调整
            shiftDown(array,0,end);
            //排好最后一个位置, 最后一个位置有序,数组减1排下一个,第二大的就可以放到倒数第二,依次类推直到end结束
            //最后层序遍历,可以获取升序数组
            end--;
        }
    }
    //建堆(基于完全二叉树): 从最后一颗子树开始,使用向下调整(父节点位置向下移动到子节点位置)
    private static void creatHeap(int[] array) {
    	//父节点下标 = ( 孩子节点下标 - 1 ) / 2
        for (int parent = (array.length - 1 - 1) / 2; parent >= 0; parent--) {
            shiftDown(array,parent,array.length);
        }
    }
    //向下调整需要获取子节点位置, 所以给了父节点, 根据二叉树规则得到子节点下标,
    // 得到子节点下标,需要保证它不越界,
    // 通过观察每个子节点最大都不会超过数组长度, 所以给一个通用的数组长度即可
    private static void shiftDown(int[] array, int parent, int end) {
        //确定孩子节点
        int child = 2 * parent + 1;
        //必须有左孩子,保证孩子节点不越界
        while (child < end) {
            //左右孩子谁大,右孩子大则更新右孩子节点下标,否则不更新
            if(child + 1 < end && array[child] < array[child + 1]) {
                child++;//下标+1表示到了右孩子下标
            }
            //比较父子节点大小
            if(array[child] > array[parent]) {
                swap(array,child,parent);
                //更新父子节点位置,向下进行调整的,父子节点都是向下的
                parent = child;
                child = 2 * parent + 1;
            }else {
                break;//孩子小于父亲这棵树就不用换
            }
        }
    }

5. 选择排序

**思路①: 遍历数组, 第一个 i 位置有序,从i+1位置开始到数组结束(待排序位置), 寻找最小值下标, 提前定义变量记录最小值下标位置, 最小值不是一次就可以找到的, 可能需要不断更新需要记录; 找到之后交换最小值位置与 i 位置元素, 之后一次i++,重复上述操作即可;
思路②:
定义两个下标left, right表示每次遍历的范围,使用 i 遍历数组(数组范围[left+1,right]);
循环找最大最小值下标maxIndex,minIndex,当left < right 时,找到两个下标, 分别进行交换(交换最小值与left位置,最大值与right交换);
特殊情况:交换最小值与左下标, 交换后会遇到特殊情况最大值刚好位于left位置, 交换最小值后,最大值正好跑到最小值位置, 此时需要更新最大值位置; **
特殊情况如图: left位置与maxIndex位置一致时产生的问题

在这里插入图片描述

时间复杂度: O(n^2)
空间复杂度: O(1)
稳定性: 不稳定

选择排序代码1实现:

public static void selectSort(int[] array) {
        //遍历数组
        for (int i = 0; i < array.length; i++) {
            int minIndex = i;
            //在剩余待排序序列中找到最小的, 如果找不到直接下一个
            for (int j = i + 1; j < array.length; j++) {
                if(array[j] <  array[minIndex]) {
                    //更改最小值下标
                    minIndex = j;
                }
            }
            //找到最小值与起始位置交换, 每次i位置就为起始位置, 在剩余length-i位置中找比i位置小的元素,找到交换
            //直到数组找完 i > length
            int tmp = array[i];
            array[i] = array[minIndex];
            array[minIndex] = tmp;
        }
    }

选择排序代码2实现:

    public static void selectSort2(int[] array) {
        int left = 0;
        int right = array.length - 1;
        while (left < right) {
            //定义两个存储下标
            int minIndex = left;
            int maxIndex = left;
            //从待排序序列中找, 第一个认为有序, 找到最大最小值位置
            for (int i = left + 1; i < array.length; i++) {
                if(array[i] < array[minIndex]) {
                    minIndex = i;
                }
                if(array[i] > array[maxIndex]) {
                    maxIndex = i;
                }
            }
            //交换, 最小值换到左边, 最大值换到右边
            swap(array,left,minIndex);
            //存在特殊情况最大值刚好位于left位置, 交换最小值后,最大值正好跑到最小值位置
            // 此时需要更新最大值位置
            if(maxIndex == left) {
                maxIndex = minIndex;
            }
            //在交换最大值位置
            swap(array,right,maxIndex);
            left++;
            right--;
        }
    }
    private static void swap(int[] array, int i, int j) {
        int tmp = array[i];
        array[i] = array[j];
        array[j] = tmp;
    }

6 快速排序:

思路: 快排递归是一样的,只是使用的找基准值下标的方法不同;
①挖坑法找基准值下标思路: 每次以第一个为基准定义为tmp(left位置), 先从数组右边开始找到大于tmp的数就将它放入left位置, 再从左边找到小于tmp的数,将他放到right位置, 直到left == right相遇时, tmp再放到left位置, 此时它的位置就是基准值位置;
② hoare法: 先记录最左边下标位置t, 循环结束后还需要交换t位置与新的left位置;
以第一个为基准值, 先从右边找到比基准值小的下标, 从左边找到比基准值大的下标, 再交换二者元素,依次交换直至left > right 循环结束;

好的情况:
时间复杂度: O(n*logn)
空间复杂度: O(logn)
坏蛋情况:
时间复杂度: O(n^2)
空间复杂度: O(n)
稳定性: 不稳定

快排代码: 基准值左边全部小于基准值, 基准值右边全部大于基准值,
patition方法能返回基准值下标, 再返还基准值下标前,代码已经完成排序操作了;

public void quickSort(int[] array) {
        quick(array,0,array.length-1);
    }
    private void quick(int[] array, int start, int end) {
        //判断越界
        if(start >= end) {
            return;
        }
        //基准值: 它的左边小于基准值,右边大于基准值;
        //获取下一次基准值下标,依次为界,将数组分为两个子序列
        //左右两边重复次过程
        int index = patition(array,start,end);
        //左右两边快排
        quick(array,start,index - 1);
        quick(array,index + 1,end);
    }
  //挖坑法 -
    private int patition(int[] array, int left, int right) {
        //获取第一个为基准值,一个数组的最左边
        int tmp = array[left];
        //遍历数组
        while (left < right) {
            //先遍历右边找到比基准值小的
            //从最右边往左边找, 如果都是小于第一个的值, 会越界需要判断
            while (left < right && array[right] > tmp) {
                right--;
            }
            //找到小的就放到左边
            array[left] = array[right];
            while (left < right && array[left] < tmp) {
                left++;
            }
            //左边找到大的就放到右边
            array[right] = array[left];
        }
        array[left] = tmp;
        //左右相遇时,此下标就是下一次分割数组的位置
        return left;
    }

**快排方法2:

    private int patition2(int[] array, int left, int right) {
        //获取第一个为基准值,一个数组的最左边
        int tmp = array[left];
        int t = left;
        //遍历数组
        while (left < right) {
            //先遍历右边找到比基准值小的
            //从最右边往左边找, 如果都是小于第一个的值, 会越界需要判断 => 逆序情况
            while (left < right && array[right] > tmp) {
                right--;
            }
            //从最左边往右边找, 如果都是大于第一个的值, 会越界需要判断 => 升序情况
            while (left < right && array[left] < tmp) {
                left++;
            }
            //交换左右下标值
            swap(array,left,right);
        }
        swap(array,t,left);
        //左右相遇时,此下标就是下一次分割数组的位置
        return left;
    }

快排代码优化:
优化1: 三数取中法, 在数组最左边,中间,最右边,比较获取中间的值下标, 与数组第一个元素交换;
patition每次会返回基准值下标, 为了让基准值在数组中间,均分数组
基准值分割数组,不一定是均等分割,这样排序速度就会减慢
优化2:减少递归次数, 排序到最后会趋向大部分有序, 对于大部分有序直接插入排序是最快的

		//优化2:
  		 if(end - start + 1 < 10) {
            insertSort(array,start,end);
            return;
        }
        //优化1: 尽量让基准值为于每次数组中间,均分数组能加快排序速度
        int midTree = minTree(array,start,end);
        //在最左边,中间,最右边,获取中间的值下标, 与数组第一个元素交换
        swap(array,midTree,start);

插入排序:

private void insertSort(int[] array, int start, int end) {
        for (int i = start + 1; i <= end; i++) {
            //从第二个开始,记录此时值
            int tmp = array[i];
            int j = i-1;
            for (; j >= start ; j--) {
                //数组中值大于 tmp j位置值前移1
                if(array[j] > tmp) {
                    array[j+1] = array[j];
                }else {
                    //前面的已经排行序了, 如果小于tmp直接跳出就行,不要看前面的了
                    break;
                }
            }
            array[j] = tmp;
        }
    }

6.2 非递归快排

非递归思想:
第一次数组前后下标已知(0,array.length-1),从第二次开始使用栈;
求的基准值下标,根据基准值分割数组获取子数组;
判断获得的子数组元素是否大于1,大于1才让数组或子数组左右下标,放入栈中; 栈不为空,弹出栈中元素,进行快排patition方法(跟递归方法中的patition一样,这里就不写了)直到栈中元素为空

 public void quickSort2(int[] array) {
        //Deque实现的顺序栈
        Deque<Integer> stack = new ArrayDeque<>();
        //获取数组的前后下标
        int left = 0;
        int right = array.length - 1;
        //找基准值
        int pivot = patition(array,left,right);
        //保证基准值左边至少有一位数
        if(pivot > left + 1) {
            stack.push(left);
            stack.push(pivot - 1);
        }
        //保证基准值右边至少有1个元素
        if(pivot < right - 1) {
            stack.push(pivot + 1);
            stack.push(right);
        }
        while (!stack.isEmpty()) {
            //先弹出右下标,再弹出左下标
            right = stack.pop();
            left = stack.pop();
            pivot = patition(array,left,right);//求新的基准值下标
            //保证基准值左边至少有一位数
            if(pivot > left + 1) {
                stack.push(left);
                stack.push(pivot - 1);
            }
            //保证基准值右边至少有1个元素
            if(pivot < right - 1) {
                stack.push(pivot + 1);
                stack.push(right);
            }
        }
        //跳出循环, 再也没有子数组时,快排完毕;
    }

7. 归并排序

7.1 归并排序递归实现:
归并思想:
分治法: 先分,使用递归,将数组分为最小单位1, 再合并看自己要求,这里按升序为例, 合并两个数组,需定义一个新数组,数组1与数组2比较大小, 谁小谁先放入新数组, 可能两数组长度不一样, 长的数组的剩余元素直接放入新数组即可, 最后再将新数组组放到旧数组即可;

归并时间复杂度: O(n*logn)
空间复杂度: O(n)
稳定性: 稳定
归并排序实现代码:

public void mergeSort(int[] array) {
        mergeSortChild(array,0,array.length-1);
    }
    public void mergeSortChild(int[] array,int left,int right) {
        //防止越界,需要判断
        //当left == right: 说明递归结束
        if(left >= right) {
            return;
        }
        //找到中间下标,从而拆分数组
        int mid = (left + right) / 2;
        //拆分左边
        mergeSortChild(array,left,mid);
        //拆分右边
        mergeSortChild(array,mid + 1,right);
        //合并
        merge(array,left,right,mid);
    }
    //升序为例
    public void merge(int[] array, int left, int right, int mid) {
        //比较数组,需要遍历下标
        int s1 = left;
        int s2 = mid + 1;//数组2的开始下标
        //创建一个新数组,存放排好序的数组
        int[] tmp = new int[right - left + 1];
        int k = 0;//记录新数组元素个数
        //数组1与数组2都有元素进行比较
        while (s1 <= mid && s2 <= right) {
            //因为是升序,谁小谁先进新数组tmp
            if(array[s1] < array[s2]) {
                tmp[k++] = array[s1++];
            }else {
                tmp[k++] = tmp[s2++];
            }
        }
        //如果比较的s1,s2数组不等长(两数组中元素个数不一致)
        //再次遍历剩余数组,将剩余元素放入新数组中
        while (s1 <= mid) {
            tmp[k++] = array[s1++];
        }
        while (s2 <= right) {
            tmp[k++] = array[s2++];
        }
        //新数组在放到旧数组中
        for (int i = 0; i < tmp.length; i++) {
            array[i + left] = array[i];
        }
    }

归并排序非递归实现:
思想: 分组排序,一组一个一个,在一组两个两个,再四个四个, 分完一组再归并;

代码实现:

 public void mergeSort2(int[] array) {
        //定义组数
        int gap = 1;
        while (gap < array.length) {
            for (int i = 0; i < array.length; i += gap*2) {
                //i += gap*2 => 排完一组排一组,使最左下标更换以至于遍历全部数组元素
                int left = i;
                int mid = left + gap - 1;
                //中间下标越界
                if(mid > array.length) {
                    mid = array.length - 1;
                }
                //右下标越界
                int right = mid + gap;
                if(right > array.length) {
                    right = array.length - 1;
                }
                //排序
                merge(array,left,right,mid);
            }
            //增加每组的个数1->2,2->4等等
            gap *= 2;
        }
    }

总结

✨✨✨各位读友,本篇分享到内容如果对你有帮助给个👍赞鼓励一下吧!!
感谢每一位一起走到这的伙伴,我们可以一起交流进步!!!一起加油吧!!!

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

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

相关文章

代码生活处处是代理--静态代理+动态代理的实现

你买房了吗&#xff1f;你买车了吗&#xff1f;你买啥啥啥了吗&#xff1f;现在这个社会除了物理物质我们离不开&#xff0c;还有一种角色也在潜移默化中渗透我们的生活。---中介 代理模式的思想是为了提供额外的处理或者不同的操作而在实际对象与调用者之间插入一个代理对象。…

【深度学习实验】卷积神经网络(四):自定义二维汇聚层:最大汇聚(max pooling)和平均汇聚(average pooling)

目录 一、实验介绍 二、实验环境 1. 配置虚拟环境 2. 库版本介绍 三、实验内容 0. 导入必要的工具包 1. Conv2D&#xff08;二维卷积层&#xff09; 2. Pool2D&#xff08;二维汇聚层&#xff09; 理论知识 a. 初始化 b. 前向传播(最大汇聚层) c. 前向传播(平均汇聚…

GLTF编辑器也可以转换GLB模型

1、GLB模型介绍 GLB&#xff08;GLTF Binary&#xff09;是一种用于表示三维模型和场景的文件格式。GLTF是"GL Transmission Format"的缩写&#xff0c;是一种开放的、跨平台的标准&#xff0c;旨在在各种3D图形应用程序和引擎之间进行交换和共享。 GLB文件是GLTF文件…

ubuntu22.04怎么安装微信3.2.1

下载微信和wine的安装包 https://archive.ubuntukylin.com/software/pool/partner/ 使用root权限安装wine apt-get install -f -y ./ukylin-wine_70.6.3.25_amd64.deb安装微信 apt-get install -f -y ./ukylin-wechat_3.0.0_amd64.deb安装成功会出来一个页面

高仿互站网站源码 后台手机端两套模板 电脑端二十套模版

高仿互站网 后台手机端两套模板 电脑端二十套模版&#xff0c;简单介绍几个功能&#xff0c; 支持用户注册开店 开店申请&#xff0c;支持用户发布自己商品 支持卡密形式或实物形式&#xff0c; 支持用户自己发布求助 任务大厅功能&#xff0c;源码完整 更多功能自己去发现吧…

低代码平台如何助力国内企业数字化转型?

数字化是什么 数字化&#xff08;Digitalization&#xff09;是将许多复杂多变的信息转变为可以度量的数字、数据&#xff0c;再以这些数字、数据建立起适当的数字化模型&#xff0c;把它们转变为一系列二进制代码&#xff0c;引入计算机内部&#xff0c;进行统一处理&#xf…

【Linux】生产者和消费者模型

生产者和消费者概念基于BlockingQueue的生产者消费者模型全部代码 生产者和消费者概念 生产者消费者模式就是通过一个容器来解决生产者和消费者的强耦合问题。 生产者和消费者彼此之间不直接通讯&#xff0c;而通过这个容器来通讯&#xff0c;所以生产者生产完数据之后不用等待…

网安工具 | Windows便携式渗透测试环境PentestBox入门到进阶使用指南

微信改版了&#xff0c;现在看到我们全凭缘分&#xff0c;为了不错过【全栈工程师修炼指南】重要内容及福利&#xff0c;大家记得按照上方步骤设置「接收文章推送」哦~ 关注【公众号】回复【学习交流群】加入【SecDevOps】学习交流群! 文章目录&#xff1a; 本文为作者原创文章…

Kafka Log存储解析以及索引机制

1.概述 在Kafka架构&#xff0c;不管是生产者Producer还是消费者Consumer面向的都是Topic。Topic是逻辑上的概念&#xff0c;而Partition是物理上的概念。每个Partition逻辑上对应一个log文件&#xff0c;该log文件存储是Producer生产的数据。Producer生产的数据被不断追加到该…

CSP-J第二轮试题-2021/2/3年-注意事项

参考: https://blog.csdn.net/Keven_11/article/details/126388791 https://www.cnblogs.com/-Wallace-/p/cpp-stl.html 2021年真题要求 2022年真题要求 考试注意事项 优先注意&#xff1a; 1.使用freopen。 #include <bits/stdc.h> #include<cstdio>//必须包含…

晨控CK-FR08系列读写器与LS可编程逻辑控制器MODBUSRTU连接手册

晨控CK-FR08系列读写器与LS可编程逻辑控制器MODBUSRTU连接手册 晨控CK-FR08是一款基于射频识别技术的高频RFID标签读卡器&#xff0c;读卡器工作频率为13.56MHZ&#xff0c;支持对I-CODE 2、I-CODE SLI等符合ISO15693国际标准协议格式标签的读取。读卡器内部集成了射频部分通信…

spring-boot入门之如何利用idea创建一个spring-boot项目

1.创建流程&#xff01;&#xff01;&#xff01; 选择新建项目&#xff0c;这里我们需要注意是基于maven建立的和java版本和jdk版本要对应 这里我们是基于web项目创建的记得选择这个框架。 2.测试程序 编写hello测试类 我们需要通过程序的入口进行启动程序。idea已经为我们自…

【算法练习Day7】反转字符串替换空格反转字符串中的单词左旋转字符串

​ ​&#x1f4dd;个人主页&#xff1a;Sherry的成长之路 &#x1f3e0;学习社区&#xff1a;Sherry的成长之路&#xff08;个人社区&#xff09; &#x1f4d6;专栏链接&#xff1a;练题 &#x1f3af;长路漫漫浩浩&#xff0c;万事皆有期待 文章目录 反转字符串反转字符串 I…

【洛谷 P1644】跳马问题 题解(深度优先搜索)

跳马问题 题目背景 在爱与愁的故事第一弹第三章出来前先练练四道基本的回溯/搜索题吧…… 题目描述 中国象棋半张棋盘如图 1 1 1 所示。马自左下角 ( 0 , 0 ) (0,0) (0,0) 向右上角 ( m , n ) (m,n) (m,n) 跳。规定只能往右跳&#xff0c;不准往左跳。比如图 1 1 1 中所…

修改switch Nand无线区码 以支持高频5G 信道

环境&#xff1a;NS switch 问题&#xff1a;日版&#xff0c;港版无法连接大于44信道的5G WIFI 解决办法&#xff1a;修改PRODINFO.dec的WIFI 区域码 背景&#xff1a;我的switch是最早买的港版的一批&#xff0c;WIFI 只能连接日本的信道&#xff0c;家里的路由器是大陆的&am…

CentOS7安装Oracle XE记录

本文仅是CentOS7安装Oracle XE记录&#xff0c;供参考 1、下载安装包 oracle-xe-11.2.0-1.0.x86_64.rpm.zip 2、安装 &#xff08;1&#xff09;第一次安装 [rootnode1 opt]# cd oracle-xe/ [rootnode1 oracle-xe]# ll 总用量 309884 -rw-r--r-- 1 root root 317320273 6月…

软件设计模式系列之十七——解释器模式

1 模式的定义 解释器模式是一种行为型设计模式&#xff0c;它用于将一种语言或表达式解释为对象。该模式通过定义语言的文法规则&#xff0c;并使用解释器来解释和执行这些规则&#xff0c;将复杂的语言转换为对象的操作。 在软件开发中&#xff0c;解释器模式常用于处理类似…

初识ebpf

介绍eBPF技术 当代计算机系统中&#xff0c;性能、安全性和可观察性是至关重要的关键因素。为了应对这些挑战&#xff0c;Linux 内核引入了一种名为eBPF&#xff08;extended Berkeley Packet Filter&#xff09;的强大技术。eBPF 不仅仅是一种网络数据包过滤器&#xff0c;它…

轻松拿下Offer!20个Salesforce管理员顾问的基础面试问题

先是Salesforce第二财季业绩远超预期&#xff0c;股价大涨&#xff0c;后是首席执行官Marc Benioff表示将在各部门招聘3300名员工&#xff0c;生态系统呈现欣欣向荣的态势&#xff0c;Salesforce也成为越来越多人的职业选择。 管理员作为入门级的岗位&#xff0c;是小白和新手…

黑豹程序员-再不怕猪队友把配置文件上传git暴露数据库密码了

问题&#xff1a; 项目中含有配置文件&#xff0c;而配置文件中含有数据库的用户名和密码。而团队猪队友不论三七二十一玩git时全都上传git。git上开放给外部用户。外部用户获得数据库ip地址&#xff0c;用户名&#xff0c;密码。如果运维猪队友数据库为方便直接对外网开放。那…