java学习之数据结构:三、八大排序
主要介绍学过的各种排序算法
目录
1.插入排序
1.1直接插入排序
1.2希尔排序
2.选择排序
2.1直接选择排序
2.2堆排序
3.交换排序
3.1冒泡排序
3.2快速排序
4.归并排序
5.基数排序
1.插入排序
1.1直接插入排序
基本思想:就是将待排序的数据按照其元素值的大小注意插入到一个已经排好序的有序序列中,直到所有数据插完为止。
基本流程:
- 初始化:从数组的第二个元素开始,将其当作第一个待插入的元素。
- 遍历未排序元素:对数组从第二个元素到最后一个元素进行遍历,每次取一个未排序的元素作为待插入元素。
- 寻找插入位置:把待插入元素存储在临时变量
temp
中,然后从已排序序列的末尾开始向前查找,只要已排序序列中的元素大于待插入元素,就将该元素向后移动一位。 - 插入元素:当找到一个不大于待插入元素的位置或者已经遍历完已排序序列时,将待插入元素插入到该位置之后。
- 重复步骤 2 - 4:持续重复上述操作,直至所有元素都被插入到合适的位置,此时数组完成排序。
代码如下:
public static void insertSort(int[] arr) {for (int i = 1; i < arr.length; i++) {int temp = arr[i];int j = i - 1;while (j >= 0 && arr[j] > temp) {arr[j + 1] = arr[j];j--;}arr[j + 1] = temp;}}
时间复杂度:
最好的情况:最好的情况就是数组已经升序有序,只有最外面一层for循环遍历一次数组,里面的while循环每一次都是直接退出,因此最好的情况时间复杂度为O(N)
最坏的情况:最坏的情况就是数组是降序排序,里面while循环的时间复杂度为O(N),因此最坏情况下,时间复杂度为O(N^2)
综上,直接插入法的时间复杂度为O(N^2)
1.2希尔排序
基本思想:对直接插入的优化。先选定一个整数gap,把待排序文件中所有记录分成数组,所有距离为gap的记录分在同一组内,并对每一组内的记录进行排序。然后缩小gap,重复上述步骤,当gap == 1时,所有记录在统一组内已经排好序。
基本流程:
- 初始化间隔:把间隔
gap
设定为数组长度的一半。这个间隔用于将数组划分成多个子序列。 - 分组排序:在
gap
大于 0 的条件下,开展如下操作:- 从
gap
位置开始遍历数组,对每个元素执行插入排序操作。 - 把当前元素保存到临时变量
temp
里。 - 从当前元素的前一个间隔位置开始,只要该位置的元素大于
temp
,就将其向后移动间隔个位置。 - 找到合适位置后,把
temp
插入进去。
- 从
- 缩小间隔:每完成一轮分组排序后,将间隔
gap
缩小,一般是除以 2。 - 重复操作:持续重复步骤 2 和 3,直至
gap
变为 0,此时数组排序完成。
代码如下:
public static void shellSort(int[] arr) {int gap = arr.length / 2;while (gap > 0) {for (int i = gap; i < arr.length; i++) {int temp = arr[i];int j = i - gap;while (j >= 0 && arr[j] > temp) {arr[j + gap] = arr[j];j -= gap;}arr[j + gap] =temp;}}}
时间复杂度:希尔排序的时间复杂度为O(NLogN)
2.选择排序
2.1直接选择排序
基本思想:每一轮从待排序的数据元素中选出最小(或最大)的一个元素,存放在序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。以此类推,直到全部待排序的数据元素排完
基本流程:
- 外层循环遍历:使用变量
i
从数组的第一个元素开始,一直到倒数第二个元素(i
范围是0
到arr.length - 2
)进行遍历。每一轮外层循环都对应着将一个未排序元素放到正确的位置。 - 确定最小元素索引:
- 每轮外层循环开始时,先假设当前
i
位置的元素是未排序部分中的最小元素,将其索引i
赋值给minIndex
。 - 开启内层循环,让变量
j
从i + 1
开始,遍历到数组末尾(j
范围是i + 1
到arr.length - 1
)。在内层循环中,逐个比较arr[j]
和arr[minIndex]
,若arr[j]
更小,就把j
的值赋给minIndex
。内层循环结束后,minIndex
就指向了未排序部分里的最小元素。
- 每轮外层循环开始时,先假设当前
- 交换元素:
- 检查
minIndex
是否和i
不相等。若不相等,意味着未排序部分的最小元素不在i
位置,需要交换它们。 - 借助临时变量
temp
来完成交换,先把arr[i]
的值存到temp
中,再把arr[minIndex]
的值赋给arr[i]
,最后把temp
(即原来arr[i]
的值)赋给arr[minIndex]
。
- 检查
- 重复操作:外层循环不断推进,
i
逐步增加,重复上述步骤,直到i
达到arr.length - 2
,此时整个数组排序完成。
代码如下:
public static void selectSort(int[] arr) {for (int i = 0; i < arr.length - 1; i++) {int minIndex = i;for (int j = i + 1; j < arr.length; j++) {if (arr[j] < arr[minIndex]) {minIndex = j;}}if (minIndex != i) {int temp = arr[i];arr[i] = arr[minIndex];arr[minIndex] = temp;}}}
时间复杂度:由于使用了两层嵌套循环,无论数组初始状态如何,都需要进行大约 \(n(n - 1)/2\) 次比较,因此时间复杂度为O(n^2).
2.2堆排序
堆的逻辑结构是一棵完全二叉树
堆的物理结构是一个数组
即我们可以将堆看成是一棵完全二叉树的顺序存储
父子节点的关系:
我们可以通过数组的下标来确定父子节点的关系,结论如下
leftchild = parent * 2 + 1
rightchild = parent * 2 + 2
parent = (child - 1) / 2
堆的两个特性:
结构性:用数组表示的完全二叉树
有序性:任意节点的关键值是其子树所有节点的最大值(或最小值)
”最大堆(MaxHeap)“,也叫”大顶堆“:最大值,即每棵子树的根节点的值都大等于于其子节点的值
”最小堆(MaxHeap)“,也叫”小顶堆“:最小值,即每棵子树的根节点的值都小于等于其子节点的值
基本思想:将待排序的序列构造成一个堆,使得堆顶元素(大根堆为最大值,小根堆为最小值)是序列中的最大(或最小)元素。然后将堆顶元素与堆的最后一个元素交换位置,此时堆顶元素就是已排序好的最大(或最小)元素,将其从堆中移除,剩下的元素重新调整为一个堆,重复上述过程,直到整个序列都被排序。
算法流程:
1. 构建初始大顶堆
- 大顶堆的特点是每个节点的值都大于或等于其子节点的值。从数组的最后一个非叶子节点开始调整,逐步向前,直到根节点。最后一个非叶子节点的索引为
arr.length / 2 - 1
。 - 对于每个非叶子节点,调用
adjustHeap
方法进行调整,确保以该节点为根的子树满足大顶堆的性质。
2. 交换堆顶元素与末尾元素并调整堆
- 将堆顶元素(即最大值)与数组的最后一个元素交换位置。此时,最大值就被放到了数组的末尾,完成了一次排序。
- 缩小堆的范围(排除已经排好序的元素),并对新的堆顶元素调用
adjustHeap
方法,重新调整堆,使其满足大顶堆的性质。 - 重复这个过程,直到堆中只剩下一个元素,此时整个数组就完成了排序。
3. 输出排序结果
- 遍历排序好的数组并输出。
代码如下:
public class HeapSort {// 堆排序主方法public static void heapSort(int[] arr) {// 第一步:构建初始大顶堆// 从最后一个非叶子节点开始调整,最后一个非叶子节点的索引为 arr.length / 2 - 1for (int i = arr.length / 2 - 1; i >= 0; i--) {// 调用 adjustHeap 方法调整以 i 为根节点的子树,使其满足大顶堆的性质adjustHeap(arr, i, arr.length);}// 第二步:交换堆顶元素与末尾元素并调整堆// 从数组的最后一个元素开始,依次与堆顶元素交换for (int i = arr.length - 1; i > 0; i--) {// 交换堆顶元素(最大值)与当前末尾元素int temp = arr[i];arr[i] = arr[0];arr[0] = temp;// 交换后,对新的堆顶元素进行调整,此时堆的范围缩小为 iadjustHeap(arr, 0, i);}// 第三步:输出排序结果// 遍历排序好的数组并输出for (int i = 0; i < arr.length; i++) {System.out.print(arr[i] + " ");}}// 调整堆的方法,确保以 i 为根节点的子树满足大顶堆的性质public static void adjustHeap(int[] arr, int i, int length) {// 保存当前根节点的值int temp = arr[i];// 从当前节点的左子节点开始for (int k = i * 2 + 1; k < length; k = k * 2 + 1) {// 如果右子节点存在且右子节点的值大于左子节点的值if (k + 1 < length && arr[k] < arr[k + 1]) {// 则将 k 指向右子节点k++;}// 如果子节点的值大于保存的根节点的值if (arr[k] > temp) {// 将子节点的值赋给当前根节点arr[i] = arr[k];// 更新当前节点的索引为子节点的索引i = k;} else {// 如果子节点的值小于等于根节点的值,说明已经满足大顶堆的性质,退出循环break;}}// 将保存的根节点的值赋给最终的位置arr[i] = temp;}public static void main(String[] args) {int[] arr = {4, 6, 8, 5, 9};heapSort(arr);}
}
时间复杂度:向下调整算法AdjustDown()
的时间复杂度为O(logN)。自下而上建堆的时间复杂度为O(N)。建堆后排序的时间复杂度为O(NlogN)。因此整个堆排序的时间复杂度为O(NlogN)
3.交换排序
3.1冒泡排序
算法思想:比较相邻元素,如果前面的比后面的元素大,则两元素交换位置。对每一对相邻元素进行比较,大的放后,这样最后的元素将是最大的元素。对越来越少的混乱元素重复上述步骤(最后的元素已经有序,不需比较),直到没有元素需要交换位置。
算法流程:
- 外层循环控制排序轮数:外层循环使用变量
i
从0
到arr.length - 2
进行遍历。i
代表当前进行的排序轮数,总共需要进行arr.length - 1
轮排序。因为经过arr.length - 1
轮比较和交换后,数组就会完成排序。 - 内层循环进行相邻元素比较和交换:
- 内层循环使用变量
j
从0
到arr.length - 1 - i
进行遍历。每一轮排序中,随着i
的增加,需要比较的元素范围逐渐缩小,因为每一轮排序都会将当前未排序部分的最大元素 “冒泡” 到末尾,所以后面已经排好序的元素不需要再参与比较。 - 在内层循环中,比较相邻的两个元素
arr[j]
和arr[j + 1]
。如果arr[j]
大于arr[j + 1]
,说明这两个元素的顺序不符合升序要求,需要进行交换。 - 交换操作通过临时变量
temp
来完成,先将arr[j]
的值赋给temp
,再将arr[j + 1]
的值赋给arr[j]
,最后将temp
(即原来arr[j]
的值)赋给arr[j + 1]
。
- 内层循环使用变量
- 重复操作直至排序完成:
- 不断重复上述外层循环和内层循环的操作,每一轮排序都会使当前未排序部分的最大元素移动到未排序部分的末尾。经过
arr.length - 1
轮排序后,整个数组就会变成升序排列。
- 不断重复上述外层循环和内层循环的操作,每一轮排序都会使当前未排序部分的最大元素移动到未排序部分的末尾。经过
代码如下:
public static void bubbleSort(int[] arr) {for (int i = 0; i < arr.length - 1; i++) {for (int j = 0; j < arr.length - 1 - i; j++) {if (arr[j] > arr[j + 1]) {int temp = arr[j];arr[j] = arr[j + 1];arr[j + 1] = temp;}}}}
时间复杂度:冒泡排序的时间复杂度为O(N^2)
3.2快速排序
基本思想:任取待排序元素序列中的某个元素作为基准值,按照该排序码将待排序集合分割成两个子序列,左子序列中所有元素均小于基准值,右子序列中所有元素均大于基准值,然后左右子序列重复该过程,直到所有元素都排列在相应位置上为止。
算法流程:
- 基准条件判断:若
left
小于right
,说明当前子数组元素数量多于 1,需要进行排序;反之则无需排序。 - 初始化变量:
i
初始化为left
,作为左指针。j
初始化为right
,作为右指针。- 选取
arr[left]
作为基准值temp
。
- 分区操作:
- 当
i
小于j
时,开展以下操作:- 从右向左移动右指针
j
,只要arr[j]
大于等于基准值temp
,就将j
减 1,直至找到一个小于基准值的元素,然后把该元素赋值给arr[i]
。 - 从左向右移动左指针
i
,只要arr[i]
小于等于基准值temp
,就将i
加 1,直至找到一个大于基准值的元素,然后把该元素赋值给arr[j]
。
- 从右向左移动右指针
- 当
- 放置基准值:当
i
与j
相遇时,把基准值temp
放到arr[i]
位置。此时,基准值左边的元素都小于等于它,右边的元素都大于等于它。 - 递归排序:
- 对基准值左边的子数组
arr[left]
到arr[i - 1]
递归调用quickSort
方法进行排序。 - 对基准值右边的子数组
arr[i + 1]
到arr[right]
递归调用quickSort
方法进行排序。
- 对基准值左边的子数组
代码如下:
public static void quickSort(int[] arr, int left, int right) {if (left < right) {int i = left;int j = right;int temp = arr[left];while (i < j) {while (i < j && arr[j] >= temp) {j--;}arr[i] = arr[j];while (i < j && arr[i] <= temp) {i++;}arr[j] = arr[i];}arr[i] = temp;quickSort(arr, left, i - 1);quickSort(arr, i + 1, right);}}
时间复杂度:平均情况下为 (O(nlogn)),最坏情况下为 (O(n^2))。
4.归并排序
基本思想:归并排序是建立在归并操作上的一种有效的排序算法,该算法是采用分治法的一个非常典型的应用:将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。
-
归并排序首先要对待排序列不断二分,直到分成不可分割的子序列(即只有一个元素的序列,相当于有序)
-
然后,再对有序的子序列不断进行归并操作,最后得到完全有序的序列。
算法流程:
mergeSort
方法(递归分解)
- 基准条件判断:检查
left
是否小于right
,如果不满足该条件,说明当前子数组只有一个元素或者为空,无需排序,直接返回。 - 计算中间位置:计算中间索引
mid
,将数组分为左右两部分,即[left, mid]
和[mid + 1, right]
。 - 递归分解:
- 对左半部分数组
[left, mid]
递归调用mergeSort
方法进行排序。 - 对右半部分数组
[mid + 1, right]
递归调用mergeSort
方法进行排序。
- 对左半部分数组
- 合并操作:当左右两部分子数组都递归排序完成后,调用
merge
方法将这两个已排序的子数组合并成一个有序数组。
merge
方法(合并子数组)
- 创建临时数组:创建一个长度为
right - left + 1
的临时数组temp
,用于存储合并后的结果。 - 初始化索引:
i
初始化为left
,表示左子数组的起始索引。j
初始化为mid + 1
,表示右子数组的起始索引。k
初始化为0
,表示临时数组的索引。
- 比较并合并元素:
- 使用
while
循环,只要i
不超过mid
且j
不超过right
,就比较arr[i]
和arr[j]
的大小。 - 如果
arr[i]
小于等于arr[j]
,将arr[i]
放入临时数组temp
中,并将i
和k
分别加 1;否则,将arr[j]
放入临时数组temp
中,并将j
和k
分别加 1。
- 使用
- 处理剩余元素:
- 当左子数组还有剩余元素时,使用
while
循环将剩余元素依次放入临时数组temp
中。 - 当右子数组还有剩余元素时,使用
while
循环将剩余元素依次放入临时数组temp
中。
- 当左子数组还有剩余元素时,使用
- 复制回原数组:将临时数组
temp
中的元素复制回原数组arr
中,从索引left
开始。
代码如下:
// 归并排序主方法,用于递归分解数组public static void mergeSort(int[] arr, int left, int right) {if (left < right) {// 计算中间位置int mid = (left + right) / 2;// 递归对左半部分数组进行排序mergeSort(arr, left, mid);// 递归对右半部分数组进行排序mergeSort(arr, mid + 1, right);// 合并左右两部分已排序的数组merge(arr, left, mid, right);}}// 合并两个已排序的子数组public static void merge(int[] arr, int left, int mid, int right) {// 创建一个临时数组,用于存储合并后的结果int[] temp = new int[right - left + 1];// 左子数组的起始索引int i = left;// 右子数组的起始索引int j = mid + 1;// 临时数组的索引int k = 0;// 比较左右子数组的元素,将较小的元素依次放入临时数组while (i <= mid && j <= right) {if (arr[i] <= arr[j]) {temp[k++] = arr[i++];} else {temp[k++] = arr[j++];}}// 将左子数组中剩余的元素放入临时数组while (i <= mid) {temp[k++] = arr[i++];}// 将右子数组中剩余的元素放入临时数组while (j <= right) {temp[k++] = arr[j++];}// 将临时数组中的元素复制回原数组for (int m = 0; m < temp.length; m++) {arr[left + m] = temp[m];}}
时间复杂度:归并排序的时间复杂度为O(NlogN)
5.基数排序
基本思想:是对于给定的输入序列中的每一个元素x,确定该序列中值小于x的元素的个数(此处并非比较各元素的大小,而是通过对元素值的计数和计数值的累加来确定)。一旦有了这个信息,就可以将x直接存放到最终的输出序列的正确位置上。
算法流程:
1. 找出最大值:遍历数组,找出数组中的最大值 max
。这一步是为了确定排序需要进行的轮数,因为需要根据最大值的位数来决定要对每一位进行排序的次数。
2. 确定排序轮数:根据最大值 max
的位数确定排序的轮数。可以通过不断除以 10 来计算位数,例如,如果最大值是 123,那么需要进行 3 轮排序,分别对个位、十位、百位进行排序。
3. 按位排序:从最低位(个位)开始,依次对每一位进行排序,直到最高位。
每一轮排序的具体步骤如下:
初始化计数数组:创建一个长度为 10 的计数数组 count
,用于统计每个数字(0 - 9)在当前位上出现的次数。
统计当前位数字的出现次数:遍历数组,对于每个元素,取出其当前位上的数字,将计数数组中对应位置的计数加 1。
计算前缀和:对计数数组进行前缀和计算,即 count[i] = count[i] + count[i - 1]
。这样做的目的是为了确定每个数字在排序后数组中的最终位置。
排序元素:创建一个与原数组长度相同的临时数组 result
,从后往前遍历原数组,根据当前位上的数字和计数数组,将元素放入临时数组的正确位置,并更新计数数组。
更新原数组:将临时数组 result
中的元素复制回原数组 arr
,完成当前位的排序。
代码如下:
public static void radixSort(int[] arr) {// 找出数组中的最大值int max = arr[0];for (int i = 1; i < arr.length; i++) {if (arr[i] > max) {max = arr[i];}}// 对每一位进行排序for (int exp = 1; max / exp > 0; exp *= 10) {countingSort(arr, exp);}}private static void countingSort(int[] arr, int exp) {int[] count = new int[10];int[] result = new int[arr.length];// 统计当前位数字的出现次数for (int i = 0; i < arr.length; i++) {int digit = (arr[i] / exp) % 10;count[digit]++;}// 计算前缀和for (int i = 1; i < 10; i++) {count[i] += count[i - 1];}// 排序元素for (int i = arr.length - 1; i >= 0; i--) {int digit = (arr[i] / exp) % 10;result[count[digit] - 1] = arr[i];count[digit]--;}// 更新原数组for (int i = 0; i < arr.length; i++) {arr[i] = result[i];}}
时间复杂度: 基数排序的时间复杂度为 (O(d * (n + k))),其中 d 是最大值的位数,n 是数组的长度,k 是基数10.
相关文章:
java学习之数据结构:三、八大排序
主要介绍学过的各种排序算法 目录 1.插入排序 1.1直接插入排序 1.2希尔排序 2.选择排序 2.1直接选择排序 2.2堆排序 3.交换排序 3.1冒泡排序 3.2快速排序 4.归并排序 5.基数排序 1.插入排序 1.1直接插入排序 基本思想:就是将待排序的数据按照其元素值的…...
Docker Compose:服务编排:批量管理多个容器
通过docker compose进行容器批量管理:一次性启动四个容器(nginx,tomcat,redis,mysql) (1) 创建docker-compose目录 mkdir ~/docker-compose cd ~/docker-compose (2&…...
微服务设计约束
相较于单体应用,微服务架构在提升开发、部署等环节灵活性的同时,也提升了在运维、监控环节的复杂性。结合实践总结,微服务架构的设计有以下四条设计约束遵循: (1)微服务个体约束 一个设计良好的微服务应用,所完成的功…...
C语言中的自定义类型 —— 结构体.位段.联合体和枚举
自定义类型 1. 前言2. 结构体2.1 结构体的声明2.2 结构体变量的定义和初始化2.3 结构体的特殊声明2.4 结构体的自引用2.5 结构体的内存对齐2.6 修改默认对齐数2.7 结构体传参 3. 位段4. 联合体5. 枚举6. 结言 1. 前言 在C语言中已经为用过户提供了内置类型,如&…...
【序列贪心】摆动序列 / 最长递增子序列 / 递增的三元子序列 / 最长连续递增序列
⭐️个人主页:小羊 ⭐️所属专栏:贪心算法 很荣幸您能阅读我的文章,诚请评论指点,欢迎欢迎 ~ 目录 摆动序列最长递增子序列递增的三元子序列最长连续递增序列 摆动序列 摆动序列 贪心策略:统计出所有的极大值和极小…...
从零开发一个B站视频数据统计Chrome插件
从零开发一个B站视频数据统计Chrome插件 前言 B站(哔哩哔哩)作为国内最大的弹幕视频网站之一,视频的播放量、点赞、投币、收藏等数据对于内容创作者和数据分析者来说非常重要。本文将带你一步步实现一个Chrome插件,自动统计并展…...
【Python实战】飞机大战
开发一个飞机大战游戏是Python学习的经典实战项目,尤其适合结合面向对象编程和游戏框架(如Pygame)进行实践。以下是游戏设计的核心考虑因素和模块划分建议: 一、游戏设计核心考虑因素 性能优化 Python游戏需注意帧率控制ÿ…...
WebAPI项目从Newtonsoft.Json迁移到System.Text.Json踩坑备忘
1.控制器层方法返回类型不能为元组 控制器层方法返回类型为元组时,序列化结果为空。 因为元组没有属性只有field,除非使用IncludeFields参数专门指定,否则使用System.Text.Json进行序列化时不会序列化field var options new JsonSerializ…...
人工智能助力工业制造:迈向智能制造的未来
在当今数字化转型的浪潮中,人工智能(AI)技术正逐渐成为推动工业制造领域变革的核心力量。智能制造作为工业 4.0 的重要组成部分,通过将 AI 技术与传统制造工艺深度融合,正在重塑整个生产流程,提高生产效率、…...
影楼精修-露齿笑算法解析
注意,为避免侵权,本文图片均为AIGC生成或网络公开数据; 像素蛋糕-露齿笑 在介绍本文之前,先说一下,其实露齿笑特效,并非像素蛋糕首创,早在几年前,face app就率先推出了这个效果&am…...
【iview】es6变量结构赋值(对象赋值)
变量的解构赋值 以iview的src/index.js中Vue.prototype.$IVIEW改造为例练习下怎么使用变量的解构赋值 原来的写法: const install function(Vue, opts {}) {if (install.installed) return;locale.use(opts.locale);locale.i18n(opts.i18n);Object.keys(iview).fo…...
在Windows系统中使用Docker发布镜像到镜像仓库
在Windows系统中使用Docker发布镜像到镜像仓库的步骤如下: 步骤 1:安装并配置Docker 安装Docker Desktop • 下载Docker Desktop for Windows并安装。 • 确保启用WSL 2或Hyper-V后端(根据系统版本选择)。 验证Docker运行状态 打…...
糖尿病筛查常识---秋浦四郎
糖尿病筛查可以早期发现糖尿病或糖尿病前期(血糖异常但未达到糖尿病标准),以利于及时干预,预防并发症。因为许多人患上糖尿病时没有明显症状,但已经开始对身体造成损害,有了明显糖尿病症状才检查发现糖尿病…...
CSS 预处理器 Sass
目录 Sass 一、Sass 是什么? 二、核心功能详解 1. 变量(Variables) 2. 嵌套(Nesting) 3. 混合宏(Mixins) 4. 继承(Inheritance) 5. 运算(Operations&…...
Mybatisplus:一些常用功能
自动驼峰 mybatis-plus:configuration:# 开启驼峰命名规则,默认true开启map-underscore-to-camel-case: true# 控制台日志打印,便于查看SQLlog-impl: org.apache.ibatis.logging.stdout.StdOutImpl TableName 作用:表名注解,标识…...
Golang WaitGroup 用法 源码阅读笔记
使用 sync.WaitGroup可以用来阻塞等待一组并发任务完成 下面是如何使用sync.WaitGroup的使用 最重要的就是不能并发调用Add()和Wait() var wg sync.WaitGroupfor ... {wg.Add(1) // 不能和wg.Wait()并发执行go func() {// 不能在启动的函数里面执行wg.Add(), 否则会panicde…...
第二章:一致性基础 A Primer on Memory Consistency and Cache Coherence - 2nd Edition
在本章中,我们将介绍足够多的缓存一致性知识,以便理解一致性模型是如何与缓存相互作用的。我们在 2.1 节首先给出在本入门教程中所考虑的系统模型。为了简化本章以及后续章节的阐述,我们选择了尽可能简单的系统模型,该模型足以说明…...
C++类_移动构造函数
std::move 的主要用途是在对象所有权转移时,触发移动构造函数或移动赋值运算符,避免不必要的深拷贝,提升性能。 移动构造函数 和 移动赋值运算符, std::move转换为右值,匹配到移动构造函数和移动赋值运算符。…...
Spring AI 实战:第一章、Spring AI入门之DeepSeek调用
引言:当Spring遇上AI,会擦出怎样的火花? 作为一名Java开发者,是否曾经眼红Python阵营那些花里胡哨的AI应用?是否在对接各种大模型API时,被五花八门的接口规范搞得头大?好消息是,Spr…...
fastapi+vue中的用户权限管理设计
数据库设计:RBAC数据模型 这是一个典型的基于SQLAlchemy的RBAC权限系统数据模型实现,各模型分工明确,共同构成完整的权限管理系统。 图解说明: 实体关系: 用户(USER)和角色(ROLE)通过 USER_ROLE 中间表实现多对多关系…...
Space Engineers 太空工程师 [DLC 解锁] [Steam] [Windows]
Space Engineers 太空工程师 [DLC 解锁] [Steam] [Windows] 需要有游戏正版基础本体,安装路径不能带有中文,或其它非常规拉丁字符; DLC 版本 至最新全部 DLC 后续可能无法及时更新文章,具体最新版本见下载文件说明 DLC 解锁列表&…...
随机变量数字特征
主要介绍一维随机变量期望和方差、二维随机变量期望和方差、以及协方差相关公式,及推导。 一维随机变量 以一个抛硬币的场景作为例子,如下: 抛掷两枚均匀硬币,如果两枚都是正面向上,则赢得2元,否则就输掉…...
C++总结01-类型相关
一、数据存储 1.程序数据段 • 静态(全局)数据区:全局变量、静态变量 • 堆内存:程序员手动分配、手动释放 • 栈内存:编译器自动分配、自动释放 • 常量区:编译时大小、值确定不可修改 2.程序代码段 •…...
【多线程】七、POSIX信号量 环形队列的生产者消费者模型
文章目录 Ⅰ. 信号量一、POSIX 信号量的概念二、POSIX 信号量的类型区别三、POSIX 信号量与 SystemV 信号量的区别Ⅱ. 线程信号量基本原理一、为什么要引入信号量❓二、PV 操作三、POSIX 信号量的实现原理四、CAS操作介绍Ⅲ. POSIX未命名信号量接口一、初始化无名信号量二、销毁…...
二维码批量识别—混乱多张二维码识别-物品分拣—-未来之窗-仙盟创梦IDE
仙盟模型 用途 精准分拣:快速准确识别物品上复杂或多个二维码,依据码中信息(如目的地、品类等)实现物品自动化分拣,提高分拣效率与准确性。库存管理:识别入库、出库物品二维码,更新库存数据&am…...
《TensorFlow 与 TensorFlow Lite:协同驱动 AI 应用全景》
《TensorFlow 与 TensorFlow Lite:协同驱动 AI 应用全景》 摘要 :在机器学习技术浪潮中,TensorFlow 与 TensorFlow Lite 作为 Google 技术栈的核心组件,分别占据云端训练与端侧部署的关键位置。本文将系统梳理二者架构特性、功能…...
Spring AI 实战:第三章、Spring AI结构化输出之告别杂乱无章
引言:当程序员遇上剧荒 “周末看什么?” 这个看似简单的问题,往往能让我们在各大影视平台间反复横跳半小时,最后无奈选择重刷《老友记》。本期让我们用技术解决这个"世纪难题":让大模型成为你的私人影视推荐…...
ros2 humble 控制真实机械臂(以lerobot为例)
基础版 0.确保串口访问权限 sudo chmod 666 /dev/ttyARM0 # 确保串口访问权限 1.下载 lerobot 驱动功能包 git clone https://gitee.com/kong-yue1/lerobot_devices.git 2.编写控制节点(完整代码) 主要功能是与 Feetech 电机总线进行通信&#…...
REINFORCE蒙特卡罗策略梯度算法详解:python从零实现
🧠 向所有学习者致敬! “学习不是装满一桶水,而是点燃一把火。” —— 叶芝 我的博客主页: https://lizheng.blog.csdn.net 🌐 欢迎点击加入AI人工智能社区! 🚀 让我们一起努力,共创…...
模拟SIP终端向Freeswitch注册用户
1、简介 使用go语言编写一个程序,模拟SIP-T58终端在Freeswitch上注册用户 2、思路 以客户端向服务端Freeswitch发起REGISTER请求,告知服务器当前的联系地址构造SIP REGISTER请求 创建UDP连接,连接到Freeswitch的5060端口发送初始的REGISTER请…...
Elasticsearch 中的索引模板:如何使用可组合模板
作者:来自 Elastic Kofi Bartlett 探索可组合模板以及如何创建它们。 更多阅读: Elasticsearch:可组合的 Index templates - 7.8 版本之后 想获得 Elastic 认证吗?查看下一期 Elasticsearch Engineer 培训的时间! El…...
一篇文章看懂时间同步服务
Linux 系统时间与时区管理 一、时间与时钟类型 时钟类型说明管理工具系统时钟由 Linux 内核维护的软件时钟,基于时区配置显示时间timedatectl硬件时钟 (RTC)主板上的物理时钟,通常以 UTC 或本地时间存储,用于系统启动时初始化时间hwclock …...
Mysql常用语句汇总
Mysql语句分类 DDL: 数据定义语言,用来定义数据库对象(数据库、表、字段)DML: 数据操作语言,用来对数据库表中的数据进行增删改DQL: 数据查询语言,用来查询数据库中表的记录DCL: 数据控制语言,用来创建数据…...
迭代器的思想和实现细节
1. 迭代器的本质 迭代器是一种行为类似指针的对象,它可能是指针(如 std::vector 的迭代器),也可能是封装了指针的类(如 std::list 的迭代器)。如果是指针那天然就可以用下面的运算,如果是类&am…...
[Vue]编程式导航
在 Vue 中,编程式导航是通过 JavaScript 代码(而非 <router-link> 标签)动态控制路由跳转的核心方式。这个方法依赖于 Vue Router 提供的 API,能更灵活地处理复杂场景(如异步操作、条件跳转等)。 一、…...
使用Node.js搭建https服务器
一、引言 https是是http的安全版本,在http的基础上通过传输加密和身份认证保证了传输过程中的安全性。可以认为:https http tls/ssl。本文讲述使用Node.js搭建https服务器的方法。 二、编译OpenSSL 按照《Openssl在Linux下编译/交叉编译》࿰…...
网络安全:sql注入练习靶场——sqli_labs安装保姆级教程
网络安全:sql注入练习靶场——sqli_labs安装保姆级教程 前言 sqli-labs靶场是一个开源的sql注入练习的综合靶场,包含大部分sql注入漏洞以及注入方式 网络安全学习者可以通过在sqli-labs靶场练习提升对sql注入的理解,以及学习各种绕过姿势。…...
rfsoc petalinux适配调试记录
1。安装虚拟机 2.设置共享文件夹 https://xinzhi.wenda.so.com/a/1668239544201149先设置文件夹路径 vmware 12 下安装 ubuntu 16.04 后,按往常的惯例安装 vmware-tools,安装时提示建议使用 open-vm-tools,于是放弃 vmware-tools 的安装&am…...
Windows下编译WebRTC源码
一、开发环境要求 准备一台64位的win10或win11(我用的是win11)电脑。最好是一台纯净的、没有安装过其它软件的Windows主机,避免已安装的软件和库对编译造成影响。 然后最好预留超过100G的硬盘空间。因为编译WebRTC时会产生大量的临时文件需…...
【vscode】.dart文件没有错误波浪线
解决方法: 新建一个文件夹,在vscode里打开这个文件夹 在这个文件夹里新建.dart文件后打开即可出现错误波浪线...
OpenharmonyOS+RK3568,【编译烧录】
文章目录 1. 摘要 ✨2. 代码下载 📩3. 编译 🖥️4. 修改&适配 ✂️4.1 编译框架基本概念4.2 vendor & device 目录4.3 内核编译4.3.1 如何修改、适配自己的开发板? 4.4 修改外设驱动 5. 烧录&验证 📋参考 1. 摘要 ✨ …...
从零开始理解 C++ 后端编程中的分布式系统
一、什么是“分布式”? 简单来说,分布式系统就是由“多个计算机(或服务器)”组成的一个大系统,它们通过网络协作完成某个任务,就像一个“团队合作”一样。 想象你开了一家餐馆,最初只有 一个厨房 和 一个服务员,所有订单都在这里处理。随着生意变好,你需要: 开分店…...
基于SpringBoot的篮球竞赛预约平台设计与实现
1.1 研究背景 科学技术日新月异的如今,计算机在生活各个领域都占有重要的作用,尤其在信息管理方面,在这样的大背景下,学习计算机知识不仅仅是为了掌握一种技能,更重要的是能够让它真正地使用到实践中去,以…...
具身系列——PPO算法实现CartPole游戏(强化学习)
完整代码参考: https://gitee.com/chencib/ailib/blob/master/rl/ppo_cartpole.py 执行结果: 部分训练得分: (sd) D:\Dev\traditional_nn\feiai\test\rl>python ppo_cartpole_v2_succeed.py Ep: 0 | Reward: 23.0 | Running: 2…...
小程序与快应用:中国移动互联网的渐进式革命——卓伊凡的技术演进观
小程序与快应用:中国移动互联网的渐进式革命——卓伊凡的技术演进观 在知乎看到很多:“懂王”发布的要把内行笑疯了的评论,卓伊凡必须怼一下,真印证那句话,无知者无畏 一、Web与小程序的技术本质差异 1.1 浏览器渲染…...
Socket 编程 UDP
Socket 编程 UDP UDP 网络编程V1 版本 - echo serverV2 版本 - DictServerV3 版本 - 简单聊天室 补充参考内容地址转换函数关于 inet_ntoa UDP 网络编程 声明:下面代码的验证都是用Windows作为客户端的,如果你有两台云服务器可以直接粘贴我在Linux下的客…...
Lua 基础 API与 辅助库函数 中关于创建的方法用法
目录 基础 API 函数1. lua_len(L, index)2. lua_load(L, reader, data, chunkname, mode)3. lua_newstate(allocator, ud)4. lua_newtable(L)5. lua_newthread(L)6. lua_newuserdata(L, size)7. lua_next(L, index) 辅助库函数(luaL_*)8. luaL_len(L, in…...
YOLOv11改进:利用RT-DETR主干网络PPHGNetV2助力轻量化目标检测
这里写自定义目录标题 YOLOv11改进:利用RT-DETR主干网络PPHGNetV2助力轻量化目标检测1. 介绍2. 引言3. 技术背景3.1 YOLOv11概述3.2 RT-DETR与PPHGNetV23.3 相关工作 4. 应用使用场景5. 详细代码实现5.1 环境准备5.2 PPHGNetV2主干网络实现5.3 YOLOv11与PPHGNetV2集…...
centos7.0无法安装php8.2/8.3
在centos安装php8.2报错 configure: error: *** A compiler with support for C17 language features is required. 配置过程检测到你的系统编译器不支持 C17 语言特性,而 PHP 8.2 的编译需要编译器支持 C17 sudo yum update -y sudo yum install centos-releas…...
工业传动核心部件深度剖析:丝杆升降机与气缸的技术特性及选型指南
在工业自动化技术飞速发展的当下,丝杆升降机与气缸作为关键的直线传动部件,广泛应用于各类机械设备中。对于工程师而言,深入了解它们的技术特性、优缺点及适用场景,是实现高效、精准设备设计的重要前提。本文将从技术原理出发&…...