十大排序算法Java实现及时间复杂度

文章目录

  • 十大排序算法
    • 选择排序
    • 冒泡排序
    • 插入排序
    • 希尔排序
    • 快速排序
    • 归并排序
    • 堆排序
    • 计数排序
    • 基数排序
    • 桶排序
    • 时间复杂度
  • 参考资料

十大排序算法

选择排序

  • 原理
    从待排序的数据元素中找出最小或最大的一个元素,存放在序列的起始位置,
    然后再从剩余的未排序元素中寻找最小/最大元素,放在已排序的序列的末尾,
    以此类推,直到全部待排序的数据元素的个数为零。

  • 实现方法

  1. 设置下标指针i和j,i从数组的第一个元素开始,j从(i+1)个元素开始
  2. j遍历到lens,选出最小的值min,将nums[i]与min交换;如果没有找到一个nums[j]<nums[i],说明自己本身就是最值,不交换;
  3. i++开始选取下一个元素,重复2,直到i到达lens-1出
    以数据{12,8,6,45,18}为例
  • 图示
    在这里插入图片描述

  • 代码实现

public class Sort {public static void main(String[] args) {int[] nums = {12,8,6,45,18};//选择排序selectSort(nums);}public static void selectSort(int[] nums){int lens = nums.length;int temp;//优化,排序之前先遍历boolean isSort = true;for(int i=0; i < lens-1; ++i){if(nums[i] > nums[i+1]){//有无序的isSort = false;break;}}if(isSort){return;//直接结束}//优化结束System.out.println("开始选择排序");for(int i=0; i< lens-1; ++i){for(int j=i+1;j< lens;++j){if(nums[j] < nums[i]){temp = nums[j];nums[j] = nums[i];nums[i] = temp;}}}for(int i =0; i < lens; ++i){System.out.print(nums[i] + " ");}}}

冒泡排序

  • 原理
    通过对排序序列从前向后(从下标较小的元素开始)依次比较相邻元素的值,若发现逆序则交换,
    使得值比较大的元素逐渐从前向后移动,就像水底下的气泡一样逐渐向上冒。

  • 实现方法

  1. 设置下标指针i和j,i用于统计外循环的次数,j用来表示当前轮次需要遍历的元素范围
  2. j的范围是0~lens-1-i,因为我们这里是每次将最大的值放在尾部,因此到第i轮的时候,最后i个值已经排完序了,不需要再判断了;
  3. 如果nums[j] > nums[j+1],则进行交换
  4. 重复上述步骤,直到lens轮排序完毕
  • 图示
    在这里插入图片描述

  • 代码

public class Sort {public static void main(String[] args) {int[] nums = {12,8,6,45,18};//冒泡排序bubbleSort(nums);}public static void bubbleSort(int[] nums){int lens = nums.length;int temp;System.out.println("开始冒泡排序");for(int i=0; i< lens - 1; ++i){for(int j = 0; j < lens - 1 - i; ++j){if(nums[j] > nums[j + 1]){temp  = nums[j];nums[j] = nums[j + 1];nums[j + 1] = temp;}}}for(int i=0;i<lens;++i){System.out.print(nums[i] + " ");}}
}

插入排序

  • 原理
    将一个记录插入到有序表中,从而形成一个新的有序表;
    每一步将一个待排序的元素,按照排序码的大小,插入到前面已经排好序的一组元素的适当位置上去,直到元素全部插入为主。

  • 实现过程

  1. 每次从待排序数组中选取元素value,将其插入到有序表中
  2. 设置下标指针i和j,i指向待排序元素,j指向已排序元素尾部,并不断左移
  3. j=i-1,当j不越界并且value小于nums[j]的时候,我们要将nums[j]及其后面的数组往右边移一位,直到value大于等于nums[j]
  4. 此时j+1的位置是value应该插入的位置,将其插入进去即可
  • 图示
    在这里插入图片描述

  • 代码

public class Sort {public static void main(String[] args) {int[] nums = {12, 8, 6, 45, 18};insertSort(nums);}public static void insertSort(int[] nums) {int lens = nums.length;System.out.println("开始插入排序");for (int i = 1; i < lens; ++i) {int value = nums[i];int j;for (j = i - 1; j >= 0 && value < nums[j]; j--) {nums[j + 1] = nums[j];//挪空位}nums[j + 1] = value;}for (int i = 0; i < lens; ++i) {System.out.print(nums[i] + " ");}}
}

希尔排序

  • 原理
    先将整个待排序的记录序列分组,对若干子序列分别进行直接插入排序,
    随着增量逐渐减少即整个序列中的记录“基本有序”时,再对全体记录进行依次直接插入排序。

  • 实现过程
    参考:java希尔排序

  • 代码

public class Sort {public static void main(String[] args) {int[] nums = {12, 8, 6, 45, 18};shellSort(nums);}public static void shellSort(int[] nums) {for (int gap = nums.length / 2; gap > 0; gap /= 2) {for (int i = gap; i < nums.length; ++i) {for (int j = i - gap; j >= 0; j -= gap) {if (nums[j] > nums[j + gap]) {int temp = nums[j];nums[j] = nums[j + gap];nums[j + gap] = temp;}}}}System.out.println(Arrays.toString(nums));}
}

快速排序

  • 原理
    通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,
    然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。

  • 图示
    在这里插入图片描述

  • 代码

public class Sort {static int[] nums = {12, 8, 6, 45, 18};public static void main(String[] args) {quickSort(nums, 0, nums.length - 1);System.out.println("快速排序: " + Arrays.toString(nums));}public static void quickSort(int[] nums, int low, int high) {int i, j, pivot;//结束条件if (low >= high) {return;}i = low;j = high;//选择的节点,默认选择第一位数pivot = nums[low];while (i < j) {//从右到左找到第一个比pivot小的数while (nums[j] >= pivot && i < j) {j--;}//从左到右找到比节点大的数while (nums[i] <= pivot && i < j) {i++;}if (i < j) {//循环找到后,交换int temp = nums[i];nums[i] = nums[j];nums[j] = temp;}}//一轮结束后,交换节点的数和ij相遇点的数nums[low] = nums[i];nums[i] = pivot;//对pivot左边和右边的数进行快速排序quickSort(nums, low, i - 1);quickSort(nums, i + 1, high);}
}

归并排序

  • 原理
    基于分治思想,先将待排序的数组不断拆分,直到拆分到区间里只剩下一个元素的时候。
    不断合并两个有序的子区间,直到所有区间都合并在一起,此时数组有序。

  • 实现过程

  1. 编写递归函数sortMerge(int[] nums,int left,int right);
  2. 参数nums表示要排序的数组,left和right表示当前排序的范围;
  3. 每进入一个子函数,计算mid,将待排序数组再一分为二,函数sortMerge的终止条件是left==right,即无法再拆分
  4. 回溯时要合并刚刚自己拆分的两个数组,合并的范围同样是left到right,用k表示合并后数组元素对应的下标
  5. 此时,两个子区间的合并,就说合并两个有序数组,借助临时数组temp存储还未合并的两个子数组原始的内容
  6. 有序数组1的下标用i表示,范围是[left,mid],有序数组2的下标用j表示,范围是[mid+1,right]
  7. 在i,j都未越界的情况下,选择小的存到nums[k],并将对应的指针往右移;
  8. 若i/j越界,则将j/i剩下的数据修改到nums中
  • 图示
    从数组中间拆分,每次拆成两个子区间
    在这里插入图片描述

函数的指向过程就是构造一个二叉树,红色箭头是当递归到left==right时,进行回溯
此时指向函数体里面的合成操作
在这里插入图片描述

  • 代码
public class Sort {public static void main(String[] args) {int[] nums = {6, 2, 7, 1, 9, 4, 8, 5, 12, 10};          //给定一个数组int len = nums.length;int[] temp = new int[len];mergeSort(nums, 0, len - 1, temp);System.out.println(Arrays.toString(nums)); //打印输出得到数组}private static void mergeSort(int[] nums, int left, int right, int[] temp) {if (left == right) {//当拆分到数组当中只要一个值的时候,结束递归return;}int mid = (left + right) / 2;   //找到下次要拆分的中间值mergeSort(nums, left, mid, temp);//记录树左边的mergeSort(nums, mid + 1, right, temp);//记录树右边的//合并两个区间for (int i = left; i <= right; i++) {temp[i] = nums[i];//temp就是辅助列表,新列表的需要排序的值就是从辅助列表中拿到的}int i = left;       //左子数组起点int j = mid + 1;  //右子数组起点//合并两个有序数组,成为一个新的有序数组for (int k = left; k <= right; k++) {//k 就为当前要插入的位置if (i == mid + 1) { //i到了右子数组起点,证明左子数字已经比较完毕nums[k] = temp[j]; //右子数字剩余的全部值赋给原数组j++;} else if (j == right + 1) { //当j超过当前的数组范围,证明右区间的数组已经遍历完毕了nums[k] = temp[i];//如果k还没有走完,证明左区间数据还有剩余,直接全部复制上去i++;}//用来比较,寻找小的哪一位插入else if (temp[i] <= temp[j]) { //如果左子数组最小值小于右子数组最小值nums[k] = temp[i]; //将两个数组中的最小值赋值给原数组i++;} else {nums[k] = temp[j];j++;}}}
}

堆排序

  • 原理
    堆是一种完全二叉树的数据结构,可以分为大根堆,小根堆。
    大根堆:每个结点的值都大于或者等于他的左右孩子结点的值
    小根堆:每个结点的值都小于或等于其左右孩子的结点值
    以大根堆为例,首先把待排序的元素按照大小在二叉树位置上排列,且要满足堆的特性。
    根据特性把根节点拿出来,然后再堆化下,即用父节点和他的孩子节点进行比较,取最大的孩子节点和其进行交换,
    再把根节点拿出来,一直循环到最后一个节点,就排序好了。

  • 实现过程

  1. 给定的待排序序列作为二叉树的层序遍历结果,构建二叉树
  2. 将这个二叉树构造成一个大顶堆(从最后一个非叶子结点开始,比较它的左右孩子是否比自己大,比自己大就交换,逐层往上找,最后根节点是最大值)
  3. 将堆顶元素与末尾元素进行交换,此时末尾为最大值;
  4. 将剩余n-1个元素重新构成一个堆,这样会得到n个元素的次小值。如此反复执行,便得到有序序列;
  • 图示
    在这里插入图片描述

  • 代码

public class Sort {static int[] nums = {4, 6, 1, 8, 9, 3, 5, 7, 11};public static void main(String[] args) {//给定一个数组heapSort(nums);System.out.println(Arrays.toString(nums)); //打印输出得到数组}public static void heapSort(int[] nums) {System.out.println("开始堆排序");//1.构建堆,使得nums[0]成为最大值buildMaxHeap(nums);for (int i = nums.length - 1; i >= 1; i--) {swap(nums, 0, i);//将当前的最大堆顶放在最后一位adjustHeap(nums, 0, i);//寻找次大值}}public static void buildMaxHeap(int[] nums) {for (int i = (nums.length - 1 - 1) / 2; i >= 0; i--) {adjustHeap(nums, i, nums.length);}}public static void adjustHeap(int[] nums, int i, int length) {int left = 2 * i + 1;int right = 2 * i + 2;int largest = i;if (left < length && nums[left] > nums[i]) {//左结点大,修改largest下标largest = left;}if (right < length && nums[right] > nums[largest]) {//看右节点的值是否会比largest的大largest = right;}if (largest != i) {//需要交换swap(nums, i, largest);adjustHeap(nums, largest, length);//继续调整}}public static void swap(int[] nums, int i, int largest) {int temp = nums[i];nums[i] = nums[largest];nums[largest] = temp;}
}

计数排序

  • 原理
    将待排序元素值转换为键存储在额外开辟的数组空间中,其要求输入的数据必须是有确定范围的整数。

  • 实现过程
    以待排序元素为0~9以内整数为例
    我们创建一个长度为10的整数ans,ans[i]用于统计数字i在待排序元素中出现的次数
    之后,根据ans[i]的值,输出ans[i]次i,直到遍历完成。

  • 图示
    在这里插入图片描述

  • 代码

public class Sort {static int[] nums = {6, 2, 7, 1, 9, 4, 8, 5, 2, 1, 3, 2, 4, 4, 5, 6, 7};public static void main(String[] args) {int len = nums.length;countSort(nums);System.out.println(Arrays.toString(nums)); //打印输出得到数组}public static void countSort(int[] nums) {System.out.println("开始计数排序");int len = nums.length;int[] a = new int[10];//下标0~9for (int i = 0; i < len; ++i) {a[nums[i]]++;}int k = 0;for (int i = 0; i < 10; ++i) {for (int j = 1; j <= a[i]; j++) {nums[k++] = i;}}}
}

基数排序

  • 原理
    通过键值的部分资讯,将要排序的元素分配至某些桶中;对于一个整数数组,先按个位数从低到高进行排序,相同的放在同一个桶中;
    之后按十位数排序,再按百位数排序,直到所有数的第k位数都是0(K取决于数组中最大的元素)。
  • 实现过程
  1. 找出数组中的最大值maxNum,遍历轮次与其有关
  2. 指针div表示的是当前按哪一个键值进行排序,1,10,100,1000分别表示键值为个位,十位,百位,千位。
  3. 每一轮计算元素对应的键值,做法是 nums[i] / div % 10; 如nums[i] = 248,div = 10; nums[i]/div = 248 / 10 = 24,24 %10得到4,
  4. 将元素依次装入对应的桶中,每一轮分配完之后,将桶中的数据按顺序依次传回原数组nums中,因为下一轮遍历需要根据此顺序。
  • 图示在这里插入图片描述

  • 代码

public class Sort {static int[] nums = {4, 6, 1, 8, 9, 3, 5, 7, 11};public static void main(String[] args) {//给定一个数组radixSort(nums);System.out.println(Arrays.toString(nums)); //打印输出得到数组}public static void radixSort(int[] nums) {System.out.println("开始基数排序");//先找到最大值,知道要排序几轮int maxNum = Integer.MIN_VALUE;for (int i = 0; i < nums.length; ++i) {if (nums[i] > maxNum) {maxNum = nums[i];}}//创建10个桶,因为桶里面装的数据个数未知,所以用数组+list优于二维数组LinkedList<Integer>[] lists = new LinkedList[10];for (int i = 0; i < 10; ++i) {lists[i] = new LinkedList<>();}//开始分桶,div表示当前排序的位数,1为个位,10为十位for (int div = 1; div <= maxNum; div *= 10) {for (int i = 0; i < nums.length; ++i) {int num = nums[i] / div % 10;//计算其位数的值lists[num].offer(nums[i]);}//把桶中的数据传回nums数组int index = 0;for (LinkedList<Integer> list : lists) {while (!list.isEmpty()) {nums[index++] = list.poll();}}}}
}

桶排序

  • 原理
    将序列中的元素分布到一定数量的桶内,然后分别对桶内的元素进行排序与,最后再将各个桶内的有序子序列放回原始序列中。
    对于桶内的元素,可以使用别的排序算法,也可以递归使用桶排序;
    一般桶内元素使用插入算法进行排序。
  • 实现过程
  1. 找出待排序的数组中的最大元素max和最小元素min
  2. 根据指定的桶数创建桶,本文使用的桶是List结构,桶里面的数据也采用List结构存储
  3. 根据公式遍历数组元素:桶编号=(数组元素-最小值)*(桶个数-1)/(最大值-最小值),把数据放到相同的桶中
  4. 从小到大遍历每一个桶,同时对也桶里的元素进行排序
  5. 把排好序的元素从索引为0开始放入,完成排序
  • 代码
public class Sort {static int[] nums = {4, 6, 1, 8, 9, 3, 5, 7, 11};public static void main(String[] args) {//给定一个数组bucketSort(nums, 3);System.out.println(Arrays.toString(nums)); //打印输出得到数组}public static void bucketSort(int[] nums, int bucketSize) {System.out.println("开始桶排序");int max = Integer.MIN_VALUE;int min = Integer.MAX_VALUE;for (int num : nums) {max = Math.max(num, max);min = Math.min(num, min);}//创建bucketSize个桶List<List<Integer>> bucketList = new ArrayList<>();for (int i = 0; i < bucketSize; i++) {bucketList.add(new ArrayList<>());}//将数据放入桶中for (int num : nums) {//确定桶号:桶编号=(数组元素-最小值)*(桶个数-1)/(最大值-最小值)int bucketIndex = (num - min) * (bucketSize - 1) / (max - min);List<Integer> list = bucketList.get(bucketIndex);list.add(num);}//对每一个桶进行排序for (int i = 0, index = 0; i < bucketList.size(); ++i) {List<Integer> list = bucketList.get(i);list.sort(null);for (int value : list) {nums[index++] = value;}}}
}

时间复杂度

排序方法时间复杂度(平均)时间复杂度(最坏)时间复杂度(最好)空间复杂度稳定性
插入排序O(N2)O(N2)O(N)O(1)稳定
希尔排序O(N1.3)O(N2)O(N)O(1)不稳定
选择排序O(N2)O(N2)O(N2)O(1)不稳定
堆排序O(N log2 N)O(N log2 N)O(N log2 N)O(1)不稳定
冒泡排序O(N2)O(N2)O(N)O(1)稳定
快速排序O(N log2 N)O(N2)O(N log2 N)O(log2 N)不稳定
归并排序O(N log2 N)O(N log2 N)O(N log2 N)O(N)稳定
计数排序O(N+k)O(N+k)O(N+k)O(N+k)稳定
桶排序O(N+k)O(N2)O(N)O(N+k)稳定
基数排序O(N*k)O(N*k)O(N*k)O(N+k)稳定

参考资料

资料1
十大经典排序
快速排序
堆排序
堆排序(Java)
桶排序

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

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

相关文章

时间序列分析基础篇

**时间序列分析&#xff08;time series analysis&#xff09;是量化投资中的一门基本技术。时间序列是指在一定时间内按时间顺序测量的某个变量的取值序列。**比如变量是股票价格&#xff0c;那么它随时间的变化就是一个时间序列&#xff1b;同样的&#xff0c;如果变量是股票…

HelloKitty 代码 Python

话不多说直接上代码,绘制速度慢,录屏之后调倍速 import math import turtle as t# 计算长度、角度 t1:画笔对象 r:半径 angle:扇形(圆形)的角度 def myarc(t1, r, angle):arc_length = 2 * math.pi * r * angle

asp.net会议预约管理系统VS开发sqlserver数据库web结构c#编程Microsoft Visual Studio

一、源码特点 asp.net 会议预约管理系统 是一套完善的web设计管理系统&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用B/S模式开发。开发环境为vs2010&#xff0c;数据库为sqlserver2008&#xff0c;使用c#语 言开发 asp.net 会议预约管理系统 二、…

【Java学习之道】日期与时间处理类

引言 在前面的章节中&#xff0c;我们介绍了Java语言的基础知识和核心技能&#xff0c;现在我们将进一步探讨Java中的常用类库和工具。这些工具和类库将帮助我们更高效地进行Java程序开发。在本节中&#xff0c;我们将一起学习日期与时间处理类的使用。 一、为什么需要日期和…

Java从resources文件下载文档,文档没有后缀名

业务场景&#xff1a;因为公司会对excel文档加密&#xff0c;通过svn或者git上传代码也会对文档进行加密&#xff0c;所以这里将文档后缀去了&#xff0c;这样避免文档加密。 实现思路&#xff1a;将文档去掉后缀&#xff0c;放入resources下&#xff0c;获取输入流&#xff0…

深度学习验证码项目

项目代码&#xff1a; GitHub - kerlomz/captcha_trainer: [验证码识别-训练] This project is based on CNN/ResNet/DenseNetGRU/LSTMCTC/CrossEntropy to realize verification code identification. This project is only for training the model. GitHub - Python3WebSpi…

win10搭建gtest测试环境+vs2019

首先是下载gtest&#xff0c;这个我已经放在了博客上方资源绑定处&#xff0c;这个适用于win10vs版本&#xff0c;关于liunx版本的不能用这个。 或者百度网盘链接&#xff1a; 链接&#xff1a;https://pan.baidu.com/s/15m62KAJ29vNe1mrmAcmehA 提取码&#xff1a;vfxz 下…

ATF(TF-A)之UBSAN动态代码分析

安全之安全(security)博客目录导读 目录 一、UBSAN简介 二、TF-A中UBSAN配置选项 一、UBSAN简介 未定义行为消毒器(Undefined Behavior Sanitizer&#xff0c;UBSAN)是Linux内核的未定义行为动态检测器。 详细信息可参考&#xff1a;https://github.com/google/kernel-sanit…

js实现日历 完整版

<template><div id"calendar"><!-- 年份 月份 --><div class"title"><div class"label">活动日历</div><div class"total">当前活动 {{ list.length }} 场</div></div><div…

基于Qt C++的工具箱项目源码,含命令行工具、桌面宠物、文献翻译、文件处理工具、医学图像浏览器、插件市场、设置扩展等工具

一、介绍 1. 基本信息 完整代码下载地址&#xff1a;基于Qt C的工具箱项目源码 TBox是一款基于Qt C的工具箱。用户可以自行选择安装所需的工具&#xff08;以插件的形式&#xff09;&#xff0c;将TBox打造成专属于自己的效率软件。TBox基本界面展示如下&#xff1a; 2. 使用…

小程序首页如何进行装修设置

小程序首页是展示给用户的第一屏&#xff0c;它的装修直接影响到用户对小程序的第一印象。小程序首页的设置在小程序管理员后台->页面设置->首页&#xff0c;下图是小程序首页默认的设置。 下图&#xff0c;是小程序首页的具体表现形式。下面具体解释小程序首页各个设置项…

【PCIE720】基于PCIe总线架构的高性能计算(HPC)硬件加速卡

PCIE720是一款基于PCI Express总线架构的高性能计算&#xff08;HPC&#xff09;硬件加速卡&#xff0c;板卡采用Xilinx的高性能28nm 7系列FPGA作为运算节点&#xff0c;在资源、接口以及时钟的优化&#xff0c;为高性能计算提供卓越的硬件加速性能。板卡一共具有5个FPGA处理节…

树和二叉树 | 一些遇到的小问题

1. TreeNode<T> &a TreeNode<T> &a是一个引用&#xff0c;指向类型为T的TreeNode节点。这个引用可以用来修改或访问该节点的值或属性。 2. *BiTree是什么意思&#xff1a; typedef struct BiTNode{ char data;struct BiTNode* lchild, * rchild; }BiT…

标定板生成网址,可以直接打印,matlab标定工具箱

Camera Calibration Pattern Generator – calib.io matlab 打开标定的成像 cameraCalibrator 点击完成之后 命令行中输入 cameraParams.IntrinsicMatrix

修改ubuntu服务器fs文件最大打开数

起因 在对项目进行压测的时候&#xff0c;请求异常 java.net.SocketException: socket closed&#xff0c;查看nginx代理服务器的日志。tail -f -n500 /var/log/nginx/error.log 显示 文件打开数太多socket() failed (24: Too many open files) while connecting to upstream …

Kubernetes核心组件Services

1. Kubernetes Service概念 Service是kubernetes最核心的概念&#xff0c;通过创建Service&#xff0c;可以为一组具有相同功能的POD&#xff08;容器&#xff09;应用提供统一的访问入口&#xff0c;并且将请求进行负载分发到后端的各个容器应用上。 在Kubernetes中&#xf…

C++ opencv实现letterbox

代码&#xff1a; #include <iostream> #include "string" #include "opencv2/opencv.hpp"cv::Mat preprocess_img(cv::Mat& img, int input_w,int input_h) {int w,h,x,y;float r_winput_w/(img.cols*1.0);float r_hinput_h/(img.rows*1.0);if…

凉鞋的 Godot 笔记 108. 第二个通识:增删改查

在这一篇&#xff0c;我们来学习此教程的第二个通识&#xff0c;即&#xff1a;增删改查。 增删改查我们不只是一次接触到了。 在最先接触的场景窗口中&#xff0c;我们是对 Node 进行增删改查。 在文件系统窗口中&#xff0c;我们是对文件&文件夹进行增删改查&#xff1…

leetCode 583.两个字符串的删除操作 动态规划 + 优化空间复杂度(二维dp、一维dp)

583. 两个字符串的删除操作 - 力扣&#xff08;LeetCode&#xff09; 给定两个单词 word1 和 word2 &#xff0c;返回使得 word1 和 word2 相同所需的最小步数。 每步 可以删除任意一个字符串中的一个字符。 示例 1&#xff1a; 输入: word1 "sea", word2 &qu…

基于springboot实现校园闲置物品交易平台系统项目【项目源码+论文说明】

基于springboot实现校园闲置物品交易平台系统演示 摘要 社会的发展和科学技术的进步&#xff0c;互联网技术越来越受欢迎。网络计算机的交易方式逐渐受到广大人民群众的喜爱&#xff0c;也逐渐进入了每个用户的使用。互联网具有便利性&#xff0c;速度快&#xff0c;效率高&am…