当前位置: 首页 > news >正文

【算法】经典排序算法介绍+代码示例

排序算法介绍

    • 1)冒泡排序 (Bubble Sort)
    • 2)选择排序(Selection Sort)
    • 3)插入排序(Insertion Sort)
    • 4)希尔排序(Shell Sort)
    • 5)归并排序(Merge Sort)
    • 6)快速排序(Quick Sort)
    • 7)堆排序(Heap Sort)
    • 8)计数排序(Counting Sort)
    • 9)桶排序(Bucket Sort)

排序算法的目的是将一组数据按照某种顺序(通常是升序或降序)排列。以下内容主要参考:十大经典排序算法(动图演示) ,下面先解释一下相关概念,然后介绍常见的排序算法:

稳定性解释:

  • 稳定:如果a原本在b前面,而a=b,排序之后a仍然在b的前面。
  • 不稳定:如果a原本在b的前面,而a=b,排序之后 a 可能会出现在 b 的后面。

1)冒泡排序 (Bubble Sort)

思想: 通过重复遍历待排序序列,比较相邻元素并根据他们的小大交换他们,小的在前大的最后,最后使较大的元素逐渐“冒泡”到序列的末尾

算法描述:

  • 比较相邻的元素。如果第一个比第二个大,就交换它们两个的位置
  • 对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对,这样在最后的元素应该会是最大的数
  • 此时后面的元素就是从小到大排好序的(有序序列),前的所有元素(无序序列)需要重复以上操作,找到下一个最大的数,加入有序序列的最前面,有序序列元素逐渐增加,无序序列元素逐渐减少
  • 针对无序序列重复以上的步骤,直到无序序列没有元素,排序完成

代码:

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;void BubbleSort(vector<int> nums) {int n=nums.size();cout<<"Original Nums: ";for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;cout<<"Nums Length: "<<n<<endl;cout<<"BubbleSort: "<<endl;for(int len=0;len<n;len++){int swaped=0;for(int i=0;i<n;i++){if(i+1<n-len && nums[i]>nums[i+1]){int tmp=nums[i+1];nums[i+1]=nums[i];nums[i]=tmp;swaped=1;}}cout << "Iteration "<<len+1<<": ";for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;if(!swaped){cout<<"Already sorted,stop earlly!"<<endl;break;}}cout << "Sorted Nums: "<<" ";for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;cout<<"-------------------------------------------------"<<endl;
}int main() {vector<int> nums = {61, 17, 29, 22, 34, 60, 72, 21, 50, 1, 62};BubbleSort(nums);return 0;
}

代码运行结果:

Original Nums: 61 17 29 22 34 60 72 21 50 1 62
Nums Length: 11
BubbleSort:
Iteration 1: 17 29 22 34 60 61 21 50 1 62 72
Iteration 2: 17 22 29 34 60 21 50 1 61 62 72
Iteration 3: 17 22 29 34 21 50 1 60 61 62 72
Iteration 4: 17 22 29 21 34 1 50 60 61 62 72
Iteration 5: 17 22 21 29 1 34 50 60 61 62 72
Iteration 6: 17 21 22 1 29 34 50 60 61 62 72
Iteration 7: 17 21 1 22 29 34 50 60 61 62 72
Iteration 8: 17 1 21 22 29 34 50 60 61 62 72
Iteration 9: 1 17 21 22 29 34 50 60 61 62 72
Iteration 10: 1 17 21 22 29 34 50 60 61 62 72
Already sorted,stop earlly!
Sorted Nums:  1 17 21 22 29 34 50 60 61 62 72
-------------------------------------------------

时间复杂度: O(n²)

空间复杂度: O(1)

稳定性: 稳定

2)选择排序(Selection Sort)

思想: 首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。以此类推,直到所有元素均排序完毕。

算法描述:

  • 初始状态:无序区为R[1…n],有序区为空
  • 第i趟排序(i=1,2,3…n-1)开始时,当前有序区和无序区分别为R[1…i-1]和R(i…n)。该趟排序从当前无序区中选出关键字最小的记录 R[k],将它与无序区的第1个记录交换,使R[1…i]和R[i+1…n)分别变为记录个数增加1个的新有序区和记录个数减少1个的新无序区
  • n-1趟结束,数组有序化了

代码:

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;void SelectionSort(vector<int> nums) {int n=nums.size();cout<<"Original Nums: ";for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;cout<<"Nums Length: "<<n<<endl;cout<<"SelectionSort: "<<endl;for(int len=0;len<n-1;len++){int pos=len;for(int i=len+1;i<n;i++){if(nums[i]<nums[pos]) {pos=i;}}int tmp=nums[len];nums[len]=nums[pos];nums[pos]=tmp;cout << "Iteration "<<len+1<<": ";for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;}cout << "Sorted Nums: "<<" ";for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;cout<<"-------------------------------------------------"<<endl;
}int main() {vector<int> nums = {61, 17, 29, 22, 34, 60, 72, 21, 50, 1, 62};SelectionSort(nums);return 0;
}

代码运行结果:

Original Nums: 61 17 29 22 34 60 72 21 50 1 62
Nums Length: 11
SelectionSort:
Iteration 1: 1 17 29 22 34 60 72 21 50 61 62
Iteration 2: 1 17 29 22 34 60 72 21 50 61 62
Iteration 3: 1 17 21 22 34 60 72 29 50 61 62
Iteration 4: 1 17 21 22 34 60 72 29 50 61 62
Iteration 5: 1 17 21 22 29 60 72 34 50 61 62
Iteration 6: 1 17 21 22 29 34 72 60 50 61 62
Iteration 7: 1 17 21 22 29 34 50 60 72 61 62
Iteration 8: 1 17 21 22 29 34 50 60 72 61 62
Iteration 9: 1 17 21 22 29 34 50 60 61 72 62
Iteration 10: 1 17 21 22 29 34 50 60 61 62 72
Sorted Nums:  1 17 21 22 29 34 50 60 61 62 72
-------------------------------------------------

时间复杂度: O(n²)

空间复杂度: O(1)

稳定性: 不稳定,解释如下:

假设有一个数组 [a, b, 1],其中 a 和 b 相等且大于1
第一轮选择最小元素 1,将其与第一个元素 a 交换,数组变为 [1, b, a]
此时,原本在后面的 b 在排序后跑到了a 的前面,破坏了相等元素的相对顺序
因此,选择排序是不稳定的排序算法

3)插入排序(Insertion Sort)

思想: 将待排序元素插入到已排序序列的适当位置,类似于扑克牌整理

算法描述:

  • 从第一个元素开始,该元素可以认为已经被排序
  • 取出下一个待插入的元素,它与从后向前扫描已经排序的元素序列,与其中的元素比较,直到找到合适的位置(前面的元素小于等于待插入的元素,后面的元素大于待插入的元素)
  • 重复以上步骤

代码:

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;void InsertionSort(vector<int> nums) {int n=nums.size();cout<<"Original Nums: ";for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;cout<<"Nums Length: "<<n<<endl;cout<<"InsertionSort: "<<endl;for(int len=1;len<n;len++){int now_num=nums[len];int i=len;while(i>0 && now_num<nums[i-1]){nums[i]=nums[i-1];i--;}nums[i]=now_num;if(len<n){cout << "Iteration "<<len+1<<": ";}for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;}cout << "Sorted Nums: "<<" ";for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;cout<<"-------------------------------------------------"<<endl;
}
int main() {vector<int> nums = {61, 17, 29, 22, 34, 60, 72, 21, 50, 1, 62};InsertionSort(nums);return 0;
}

代码运行结果:

Original Nums: 61 17 29 22 34 60 72 21 50 1 62
Nums Length: 11
InsertionSort:
Iteration 2: 17 61 29 22 34 60 72 21 50 1 62
Iteration 3: 17 29 61 22 34 60 72 21 50 1 62
Iteration 4: 17 22 29 61 34 60 72 21 50 1 62
Iteration 5: 17 22 29 34 61 60 72 21 50 1 62
Iteration 6: 17 22 29 34 60 61 72 21 50 1 62
Iteration 7: 17 22 29 34 60 61 72 21 50 1 62
Iteration 8: 17 21 22 29 34 60 61 72 50 1 62
Iteration 9: 17 21 22 29 34 50 60 61 72 1 62
Iteration 10: 1 17 21 22 29 34 50 60 61 72 62
Iteration 11: 1 17 21 22 29 34 50 60 61 62 72
Sorted Nums:  1 17 21 22 29 34 50 60 61 62 72
-------------------------------------------------

时间复杂度: O(n²)

空间复杂度: O(1)

稳定性: 稳定

4)希尔排序(Shell Sort)

思想: 希尔排序(Shell Sort)是插入排序的一种,它是针对直接插入排序算法的改进,又称缩小增量法,是第一个突破O(n²)的排序算法,将整个待排序序列按一定间隔分组,分割成若干个子序列,对每个子序列进行插入排序,然后逐步缩小间隔直至1,直至整个序列有序。

为什么先分段再插入排序就可以提升原本的插入排序的效率?

  • 插入排序在处理大规模数据时,如果数据分布不均匀,需要进行大量的元素交换和移动,效率较低。但是插入排序在数据已经部分有序时效率很高,时间复杂度可以接近O(n)
  • 希尔排序通过引入增量序列(gap),将数组分成若干子序列,对每个子序列进行插入排序。这样,元素可以在较大的步长下快速移动到接近最终位置,减少后续插入排序中的移动次数,这可以使得整个数组逐步趋于“基本有序”,从而在最后一步(gap=1)进行插入排序时,能够以较高的效率完成排序

算法描述:

  • 选择一个增量序列t1,t2,…,tk,其中ti>tj,tk=1
  • 按增量序列个数k,对序列进行k 趟排序
  • 每趟排序,根据对应的增量ti,将待排序列分割成若干长度为m 的子序列,分别对各子表进行直接插入排序
  • 仅增量因子为1 时,整个序列作为一个表来处理,表长度即为整个序列的长度

代码:

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;void ShellSort(vector<int> nums) {int n=nums.size();cout<<"Original Nums: ";for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;cout<<"Nums Length: "<<n<<endl;cout<<"ShellSort: "<<endl;for(int gap=n/2;gap>0;gap/=2){cout << "Gap: "<<gap<<endl;for(int i=gap;i<n;i++){int j=i;int now_num=nums[i];while(j-gap>=0 && now_num<nums[j-gap]){nums[j]=nums[j-gap];j-=gap;}nums[j]=now_num;for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;}cout<<"----------------------------------"<<endl;}cout << "Sorted Nums: "<<" ";for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;cout<<"-------------------------------------------------"<<endl;
}int main() {vector<int> nums = {61, 17, 29, 22, 34, 60, 72, 21, 50, 1, 62};ShellSort(nums);return 0;
}

代码运行结果:

Original Nums: 61 17 29 22 34 60 72 21 50 1 62
Nums Length: 11
ShellSort:
Gap: 5
60 17 29 22 34 61 72 21 50 1 62
60 17 29 22 34 61 72 21 50 1 62
60 17 21 22 34 61 72 29 50 1 62
60 17 21 22 34 61 72 29 50 1 62
60 17 21 22 1 61 72 29 50 34 62
60 17 21 22 1 61 72 29 50 34 62
----------------------------------
Gap: 2
21 17 60 22 1 61 72 29 50 34 62
21 17 60 22 1 61 72 29 50 34 62
1 17 21 22 60 61 72 29 50 34 62
1 17 21 22 60 61 72 29 50 34 62
1 17 21 22 60 61 72 29 50 34 62
1 17 21 22 60 29 72 61 50 34 62
1 17 21 22 50 29 60 61 72 34 62
1 17 21 22 50 29 60 34 72 61 62
1 17 21 22 50 29 60 34 62 61 72
----------------------------------
Gap: 1
1 17 21 22 50 29 60 34 62 61 72
1 17 21 22 50 29 60 34 62 61 72
1 17 21 22 50 29 60 34 62 61 72
1 17 21 22 50 29 60 34 62 61 72
1 17 21 22 29 50 60 34 62 61 72
1 17 21 22 29 50 60 34 62 61 72
1 17 21 22 29 34 50 60 62 61 72
1 17 21 22 29 34 50 60 62 61 72
1 17 21 22 29 34 50 60 61 62 72
1 17 21 22 29 34 50 60 61 62 72
----------------------------------
Sorted Nums:  1 17 21 22 29 34 50 60 61 62 72
-------------------------------------------------

时间复杂度:

希尔排序的时间复杂度不好计算,因为gap的取值方法很多,导致很难计算,一般来说,​最坏时间复杂度: O ( n 2 ) O(n^2) O(n2),最好的时间复杂度 O ( n ) O(n) O(n),平均时间复杂度有 O ( n 1.3 ) O(n^{1.3}) O(n1.3)

空间复杂度: O(1)

稳定性: 不稳定(希尔排序不稳定的原因在于分组插入排序,具体来说,希尔排序通过将数组分成若干子序列(基于增量序列),对每个子序列进行插入排序。虽然单次插入排序是稳定的​(不会改变相同元素的相对顺序),但在希尔排序中,​相同元素可能被分到不同的子序列中,并在各自的插入排序过程中移动,从而导致它们的相对顺序被打乱)

5)归并排序(Merge Sort)

思想: 一种基于分治法​的高效排序算法,它的核心思想是将待排序的数组不断二分,递归地对每个子数组进行排序,最后将两个已排序的子数组合并成一个有序的整体

算法描述:

  • 分解:将数组从中间分成两个子数组,递归地对每个子数组进行排序
  • 合并:将两个已排序的子数组合并成一个有序的数组。合并过程通过比较两个子数组的元素,依次将较小的元素放入新数组中,直到所有元素合并完成

代码:

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;vector<int> mergesort(vector<int>& nums) {int n=nums.size();if(n==1) return nums;int mid=n/2;vector<int> left(nums.begin(),nums.begin()+mid);vector<int> left_sorted=mergesort(left);vector<int> right(nums.begin()+mid,nums.end());vector<int> right_sorted=mergesort(right);vector<int> merged_nums;int i=0;int j=0;while(i<left_sorted.size() && j<right_sorted.size()){if(left_sorted[i]<=right_sorted[j]){merged_nums.push_back(left_sorted[i]);i++;}else{merged_nums.push_back(right_sorted[j]);j++;}}while(i<left_sorted.size()){merged_nums.push_back(left_sorted[i]);i++;}while(j<right_sorted.size()){merged_nums.push_back(right_sorted[j]);j++;}return merged_nums;
}void MergeSort(vector<int> nums) {int n=nums.size();cout<<"Original Nums: ";for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;cout<<"Nums Length: "<<n<<endl;cout<<"MergeSortSort: "<<endl;vector<int> merged_nums=mergesort(nums);cout << "Sorted Nums: "<<" ";for (int i = 0; i < n; i++)cout << merged_nums[i] << ' ';cout << endl;cout<<"-------------------------------------------------"<<endl;
}int main() {vector<int> nums = {61, 17, 29, 22, 34, 60, 72, 21, 50, 1, 62};MergeSort(nums);return 0;
}

代码运行结果:

Original Nums: 61 17 29 22 34 60 72 21 50 1 62
Nums Length: 11
MergeSortSort:
Sorted Nums:  1 17 21 22 29 34 50 60 61 62 72
-------------------------------------------------

时间复杂度: 稳定的O(n*longn)

  • 分解阶段每次将数组分成两半,递归地对每个子数组进行排序,递归的深度为 logn
  • 每次合并两个有序子数组的时间复杂度为 O(n),因为需要遍历所有元素进行比较和移动
  • 递归树的每一层都需要 O(n) 的时间进行合并,递归树共有 logn 层,因此总时间复杂度为 O(nlogn)

空间复杂度: O(n)

因为归并排序在合并过程中需要使用一个与原始数组大小相同的临时数组来存储合并后的结果 O(n)。此外,递归调用栈会占用额外的空间,其空间复杂度为 O(logn)。因此,归并排序的总空间复杂度为 O(n)+O(logn)=O(n)

稳定性: 稳定

6)快速排序(Quick Sort)

思想: 一种基于分治法​的高效排序算法,它的核心思想是将待排序的数组不断分割成独立的两部分,其中一部分的元素均比另一部分的元素都要小,递归地对每个子部分进行排序,以达到整个序列有序

算法描述:

  • ​选择基准值(Pivot)​:从数组中选择一个元素作为基准值(通常选择第一个、最后一个或中间元素)
  • 分区(Partition)​:将数组重新排列,使得所有小于基准值的元素位于基准值的左侧,所有大于基准值的元素位于基准值的右侧。基准值最终位于其正确的位置
  • 递归排序:对基准值左侧和右侧的子数组递归地执行上述步骤,直到子数组长度为1或0,此时数组已完全有序

具体实现:

  • ​选择基准值:从数组中选择一个基准值。

  • 分区操作:
    1) 初始化两个指针,i从左向右扫描i,j从右向左扫描
    2)从左向右找到第一个大于或等于基准值的元素,从右向左找到第一个小于或等于基准值的元素
    3)交换这两个元素,直到指针相遇
    4)将基准值放置到其正确的位置,使得基准左边的值全部小于他,右边的值全部大于它

  • 递归排序:对基准值左侧和右侧的子数组分别递归执行上述步骤

代码:

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;vector<int> quicksort1(vector<int>& nums) {int n=nums.size();if(n<=1) return nums;int pivot_num=nums[0];int i=0;int j=n-1;while(i<=j){while(i<=j && nums[i]<=pivot_num) i++;while(i<=j && nums[j]>pivot_num) j--;if(i<=j){int tmp=nums[j];nums[j]=nums[i];nums[i]=tmp;i++;j--;}}int tmp=nums[j];nums[j]=pivot_num;nums[0]=tmp;vector<int> left(nums.begin(),nums.begin()+j);vector<int> right(nums.begin()+j+1,nums.end());vector<int> left_sorted=quicksort1(left);vector<int> right_sorted=quicksort1(right);vector<int> merged_nums;for(auto x:left_sorted) merged_nums.push_back(x);merged_nums.push_back(pivot_num);for(auto x:right_sorted) merged_nums.push_back(x);return merged_nums;
}void quicksort2(vector<int>& nums,int l,int r) {int n=r-l+1;if(n<=1) return ;int pivot_num=nums[l];int i=l+1;int j=r;while(i<=j){while(i<=j && nums[i]<=pivot_num) i++;while(i<=j && nums[j]>pivot_num) j--;if(i<=j){int tmp=nums[j];nums[j]=nums[i];nums[i]=tmp;i++;j--;}}int tmp=nums[j];nums[j]=pivot_num;nums[l]=tmp;quicksort2(nums,0,j-1);quicksort2(nums,j+1,r);return ;
}void QuickSort(vector<int> nums) {int n=nums.size();cout<<"Original Nums: ";for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;cout<<"Nums Length: "<<n<<endl;cout<<"QuickSort: "<<endl;//vector<int> merged_nums=quicksort1(nums);quicksort2(nums,0,n-1);cout << "Sorted Nums: "<<" ";for (int i = 0; i < n; i++)cout << nums[i] << ' ';cout << endl;cout<<"-------------------------------------------------"<<endl;
}int main() {vector<int> nums = {61, 17, 29, 22, 34, 60, 72, 21, 50, 1, 62};QuickSort(nums);return 0;
}

代码运行结果:

Original Nums: 61 17 29 22 34 60 72 21 50 1 62
Nums Length: 11
QuickSort:
Sorted Nums:  1 17 21 22 29 34 50 60 61 62 72
-------------------------------------------------

时间复杂度:

  • ​最优情况(Best Case)​:当每次分区操作都能将数组均匀地分成两部分时,快速排序的性能最好。这种情况下,递归树的深度为 O(logn),每次分区的时间复杂度为 O(n),因此最优情况下的时间复杂度为 O(nlogn)
  • ​最坏情况(Worst Case)​:最坏情况发生在每次分区操作都导致极不均衡的分区时,例如数组已经有序或逆序。这种情况下,递归树的深度为 O(n),每次分区的时间复杂度为 O(n),因此最坏情况下的时间复杂度为 O(n *n )
  • ​平均情况(Average Case)​:在平均情况下,即输入数据的分布是随机的,快速排序的时间复杂度仍然为 O(nlogn)。这是因为每次分区操作大致能将数组分成两个大小相近的子数组,递归树的深度保持在 O(logn) 范围内。

空间复杂度: O(n)

快速排序是一种原地排序算法,不需要额外的存储空间来存储数据,但递归调用栈会占用额外的空间,在最优和平均情况下,快速排序的递归树深度为 O(logn),因此空间复杂度为 O(logn),在最坏情况下,例如数组已经有序或逆序时,递归树的深度为 O(n),空间复杂度退化为 O(n)

稳定性: 不稳定

7)堆排序(Heap Sort)

待补充

8)计数排序(Counting Sort)

待补充

9)桶排序(Bucket Sort)

待补充

相关文章:

【算法】经典排序算法介绍+代码示例

排序算法介绍 1&#xff09;冒泡排序 (Bubble Sort)2&#xff09;选择排序&#xff08;Selection Sort&#xff09;3&#xff09;插入排序&#xff08;Insertion Sort&#xff09;4&#xff09;希尔排序&#xff08;Shell Sort&#xff09;5&#xff09;归并排序&#xff08;Me…...

DeepSeek 助力 Vue3 开发:打造丝滑的表格(Table)之添加列宽调整功能,示例Table14_02带边框和斑马纹的固定表头表格

前言&#xff1a;哈喽&#xff0c;大家好&#xff0c;今天给大家分享一篇文章&#xff01;并提供具体代码帮助大家深入理解&#xff0c;彻底掌握&#xff01;创作不易&#xff0c;如果能帮助到大家或者给大家一些灵感和启发&#xff0c;欢迎收藏关注哦 &#x1f495; 目录 Deep…...

【Linux】线程控制

目录 一、原生线程库&#xff1a; 二、线程控制&#xff1a; 1、线程创建&#xff1a; 2、线程等待&#xff1a; 自定义类型的接收对象&#xff1a; ​编辑 3、线程终止&#xff1a; pthread_exit&#xff1a; pthread_cancel: 4、线程ID&#xff1a; 线程库的底层原…...

pyqt联合designer的运用和设置

PyQt Designer 简介 PyQt Designer 是一个用于创建和设计 PyQt 应用程序用户界面的可视化工具。它允许用户通过拖放方式添加和排列各种控件,如按钮、文本框、滑块等,并设置它们的属性和样式,从而快速构建出美观且功能完整的 UI 界面。 Windows版本:【免费】安装包别管啊啊…...

spring boot3.4.3+MybatisPlus3.5.5+swagger-ui2.7.0

使用 MyBatis-Plus 操作 books 表。我们将实现以下功能&#xff1a; 创建实体类 Book。 创建 Mapper 接口 BookMapper。 创建 Service 层 BookService 和 BookServiceImpl。 创建 Controller 层 BookController。 配置 MyBatis-Plus 和数据库连接。 1. 项目结构 src ├─…...

利用微软的 HTML 应用程序宿主程序的攻击

mshta.exe 是微软的 HTML 应用程序宿主程序&#xff08;Microsoft HTML Application Host&#xff09;&#xff0c;属于 Windows 系统组件。它的核心功能是运行 .hta&#xff08;HTML Application&#xff09;文件&#xff0c;允许通过 HTML、JavaScript、VBScript 等技术创建交…...

【深度学习】读写文件

读写文件 到目前为止&#xff0c;我们讨论了如何处理数据&#xff0c;以及如何构建、训练和测试深度学习模型。 然而&#xff0c;有时我们希望保存训练的模型&#xff0c;以备将来在各种环境中使用&#xff08;比如在部署中进行预测&#xff09;。 此外&#xff0c;当运行一个…...

Bert的使用

一、Data.py # data负责产生两个dataloader from torch.utils.data import DataLoader, Dataset from sklearn.model_selection import train_test_split #给X,Y 和分割比例&#xff0c; 分割出来一个训练集和验证机的X, Y import torchdef read_file(path):data []label …...

Unity使用UGUI制作无限滑动列表

原理参照上一篇使用NGUI的制作无限滑动列表的文章 Unity 使用NGUI制作无限滑动列表_unity 滑动列表很多物体-CSDN博客 准备工作&#xff1a; 新建一个空物体命名为LoopList&#xff0c;并调整其大小&#xff0c; 并增加Scroll Rect组件&#xff08;用于滑动&#xff09;、Re…...

ThinkPHP6用户登录系统的全过程

ThinkPHP6用户登录系统的全过程涉及请求处理、数据传输、路由分发、控制器逻辑、模型验证及中间件协作等多个模块的交互。详细的过程解析如下&#xff1a; 1. 前端请求与路由分发 前端发起请求&#xff1a;用户在前端页面&#xff08;如Vue组件或HTML表单&#xff09;输入用户…...

C++全栈聊天项目(2) 单例模式封装Http管理者

完善注册类界面 先在注册类构造函数里添加lineEdit的模式为密码模式 ui->lineEdit_Passwd->setEchoMode(QLineEdit::Password); ui->lineEdit_Confirm->setEchoMode(QLineEdit::Password);我们在注册界面的ui里添加一个widget&#xff0c;widget内部包含一个tip居…...

【鸿蒙开发】OpenHarmony调测工具hdc使用教程(设备开发者)

00. 目录 文章目录 00. 目录01. OpenHarmony概述02. hdc简介03. hdc获取04. option相关的命令05. 查询设备列表的命令06. 服务进程相关命令07. 网络相关的命令08. 文件相关的命令09. 应用相关的命令10. 调试相关的命令11. 常见问题12. 附录 01. OpenHarmony概述 OpenHarmony是…...

ORACLE EBS数据库RELINK方式搭建克隆环境

ORACLE EBS系统的数据库&#xff0c;一般都安装了很多特定功能的小补丁来解决特定的BUG&#xff1b;因此对于已经安装好的系统&#xff0c;想要克隆一套测试环境、搭建一个新的备机做测试等&#xff0c;如果按照生产环境标准&#xff0c;则需要安装大量补丁&#xff0c;带来很大…...

MySQL regexp 命令

REGEXP命令是一种用于进行正则表达式匹配的运算符&#xff0c;允许在查询中使用正则表达式来匹配字符串模式‌‌1。 基本语法 基本的语法结构如下&#xff1a; SELECT * FROM table_name WHERE column_name REGEXP pattern; 这里&#xff0c;pattern是你要匹配的正则表达式模…...

前端实习到工作的经历

看了很多人的程序员生涯之路&#xff0c;我突然意识到我也该记录一些东西&#xff0c;因此有感而发。 我是一个24届毕业生&#xff0c;大三下就开始找前端实习&#xff0c;当时学校不让走&#xff0c;我们都是先面着然后准备放假就去。当时周围小伙伴都找好了&#xff0c;考完…...

Vue3——Fragment

文章目录 一、Fragment的核心意义1. 解决Vue2的单根限制问题2. 减少不必要的 DOM 嵌套3. 语义化和结构化 二、Fragment 的实现原理三、Fragment 使用方式1. 基本用法2. 结合条件渲染3. 动态组件 四、实际应用场景1. 列表/表格组件2. 布局组件3. 语义化标签 五、注意事项1. 属性…...

Linux_16进程地址空间

CPU内的寄存器只有一套&#xff0c;但是CPU内寄存器的数据可能会有多份&#xff01; 一、程序地址空间 下面这个图对应的是内存吗&#xff1f;&#xff08;实际上是虚拟的进程地址空间&#xff09; 32位机器内存最大为多少&#xff1f; 32位操作系统的地址总线为32位&#x…...

职坐标机器学习编程实战:调试优化与自动化测试精要

内容概要 在机器学习编程实践中&#xff0c;代码调试优化与自动化测试工具的应用是构建高可靠性系统的核心环节。本书聚焦从数据预处理到模型部署的全流程&#xff0c;通过特征工程优化、训练过程监控及持续集成方案的设计&#xff0c;系统化解决算法工程化中的典型问题。在特…...

git文件过大导致gitea仓库镜像推送失败问题解决(push failed: context deadline exceeded)

问题描述&#xff1a; 今天发现gitea仓库推送到某个镜像仓库的操作几个月前已经报错终止推送了&#xff0c;报错如下&#xff1a; 首先翻译报错提示可知是因为git仓库大小超过1G限制。检查本地.git文件&#xff0c;发现.git文件大小已达到1.13G。确定是.git文件过大导致&…...

llvm数据流分析

llvm数据流分析 1.数据流分析2.LLVM实现2.1.常量传播2.2.活跃性分析 相关参考文档&#xff1a;DataFlowAnalysisIntro、ustc编译原理课程、南大程序分析课程1、南大程序分析课程2。 1.数据流分析 数据流分析在编译优化等程序分析任务上都有重要应用。通常数据流分析可被抽象为…...

Vite为什么选用Rollup打包?

Vite 在生产阶段使用 Rollup 打包&#xff0c;但这不是唯一选择。它的设计背后有明确的权衡和考量&#xff0c;同时开发者也可以选择其他替代方案。 一、为什么 Vite 默认使用 Rollup&#xff1f; 1. Rollup 的核心优势 • Tree-shaking&#xff1a;Rollup 的静态分析能力极强&…...

Docker 入门与实战指南

Docker 入门与实战指南 一、Docker 简介 Docker 是一个开源的容器化平台&#xff0c;允许开发者将应用程序及其依赖打包成一个可移植的容器。容器可以在任何安装了 Docker 的环境中运行&#xff0c;确保应用的一致性和可移植性。 1.1 为什么使用 Docker&#xff1f; 环境一…...

C# 常用数据类型

C# 数据类型分为 ‌值类型‌、‌引用类型‌ 和 ‌特殊类型‌&#xff0c;以下是详细分类及对应范围/说明&#xff1a; 一、值类型&#xff08;Value Types&#xff09;‌ 值类型直接存储数据&#xff0c;分配在栈内存中&#xff0c;默认不可为 null。 简单类型‌ 整数类型‌…...

深入解读 JavaScript 中 `this` 的指向机制:覆盖所有场景与底层原理

this 是 JavaScript 中最容易引发困惑的核心概念之一&#xff0c;它的指向在不同场景下呈现截然不同的行为。本文将系统性地解析 this 的所有使用场景&#xff0c;结合代码示例和底层原理&#xff0c;帮助你彻底掌握其运行机制。 一、全局环境下的 this 1. 浏览器环境 在浏览器…...

无人机全景应用解析与技术演进趋势

无人机全景应用解析与技术演进趋势 ——从立体安防到万物互联的空中革命 一、现有应用场景全景解析 &#xff08;一&#xff09;公共安全领域 1. 立体安防体系 空中哨兵&#xff1a;搭载 77 GHz 77\text{GHz} 77GHz毫米波雷达&#xff08;探测距离 5 km 5\text{km} 5km&…...

手写简易Tomcat核心实现:深入理解Servlet容器原理

目录 一、Tomcat概况 1. tomcat全局图 2.项目结构概览 二、实现步骤详解 2.1 基础工具包&#xff08;com.qcby.util&#xff09; 2.1.1 ResponseUtil&#xff1a;HTTP响应生成工具 2.1.2 SearchClassUtil&#xff1a;类扫描工具 2.1.3 WebServlet&#xff1a;自定义注解…...

【音视频】ffmpeg命令提取像素格式

1、提取YUV数据 提取yuv数据&#xff0c;并保持分辨率与原视频一致 使用-pix_fmt或-pixel_format指定yuv格式提取数据&#xff0c;并保持原来的分辨率 ffmpeg -i music.mp4 -t "01:00" -pixel_format yuv420p music.yuv提取成功后&#xff0c;可以使用ffplay指定y…...

深度剖析Redis:双写一致性问题及解决方案全景解析

在高并发场景下&#xff0c;缓存与数据库的双写一致性是每个开发者必须直面的核心挑战。本文通过5大解决方案&#xff0c;带你彻底攻克这一技术难关&#xff01; 一、问题全景图&#xff1a;当缓存遇到数据库 1.1 典型问题场景 // 典型问题代码示例 public void updateProduc…...

Redis----大key、热key解决方案、脑裂问题

文章中相关知识点在往期已经更新过了&#xff0c;如果有友友不理解可翻看往期内容 出现脑裂问题怎么保证集群还是高可用的 什么是脑裂问题 脑裂说的就是当我们的主节点没有挂&#xff0c;但是因为网络延迟较大&#xff0c;然后和主节点相连的哨兵通信较差&#xff0c;之后主…...

Android 调用c++报错 exception of type std::bad_alloc: std::bad_alloc

一、报错信息 terminating with uncaught exception of type std::bad_alloc: std::bad_alloc 查了那部分报错c++代码 szGridSize因为文件太大,初始化溢出了 pEGM->pData = new float[szGridSize]; 解决办法 直接抛出异常,文件太大就失败吧 最后还增加一个日志输出,给…...

【从零开始学习计算机科学】操作系统(五)处理器调度

【从零开始学习计算机科学】操作系统(五)处理器调度 处理器调度一些简单的短程调度算法的思路先来先服务(First-Come-First-Served,FCFS)优先级调度及其变种最短作业优先调度算法(SJF)--非抢占式最短作业优先调度算法(SJF)--抢占式最高响应比优先调度算法轮转调度算法…...

LeetCode1871 跳跃游戏VII

LeetCode 跳跃游戏 IV&#xff1a;二进制字符串的跳跃问题 题目描述 给定一个下标从 0 开始的二进制字符串 s 和两个整数 minJump 和 maxJump。初始时&#xff0c;你位于下标 0 处&#xff08;保证该位置为 0&#xff09;。你需要判断是否能到达字符串的最后一个位置&#xf…...

ResNet50深度解析:原理、结构与PyTorch实现

ResNet50深度解析&#xff1a;原理、结构与PyTorch实现 1. 引言 ResNet&#xff08;残差网络&#xff09;是深度学习领域的一项重大突破&#xff0c;它巧妙解决了深层神经网络训练中的梯度消失/爆炸问题&#xff0c;使得构建和训练更深的网络成为可能。作为计算机视觉领域的里…...

MATLAB 控制系统设计与仿真 - 24

PID 控制器分析- 控制器的形式 连续控制器的结构&#xff1a; 为滤波时间常数&#xff0c;这类PID控制器在MATLAB系统控制工具箱称为并联PID控制器&#xff0c;可由MATLAB提供的pid函数直接输入&#xff0c;格式为&#xff1a; 其他类型的控制器也可以由该函数直接输入&#x…...

数字IC后端设计实现教程 |Innovus ICC2 Routing Pin Access Setting设置方法

默认情况下routing 引擎可以在标准单元可以打孔的任何地方&#xff08;via region&#xff09;打孔&#xff0c;甚至工具还会先拉出一块metal&#xff0c;然后再打孔过渡到高层。 随之工艺节点越做越小&#xff0c;标准单元内部的结构也越来越复杂。此时如果还沿用传统工艺的走…...

mysql经典试题共34题

1、准备数据 -- drop drop table if exists dept; drop table if exists emp; drop table if exists salgrade;-- CREATE CREATE TABLE dept (deptno int NOT NULL COMMENT 部门编号,dname varchar(14) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci DEFAULT NULL COMM…...

网络编程-----服务器(多路复用IO 和 TCP并发模型)

一、单循环服务器模型 1. 核心特征 while(1){newfd accept();recv();close(newfd);}2. 典型应用场景 HTTP短连接服务&#xff08;早期Apache&#xff09;CGI快速处理简单测试服务器 3. 综合代码 #include <stdio.h> #include <sys/types.h> /* See NO…...

GitHub 项目版本管理与 Release 发布流程记录

GitHub 项目版本管理与 Release 发布流程记录 1. 项目环境设置 1.1 打开 VS Code 并进入项目目录 E:\adb\Do>code .1.2 配置 Git 用户信息 E:\adb\Do>git config --global user.name "n" E:\adb\Do>git config --global user.email "**gmail.com&q…...

GStreamer —— 2.15、Windows下Qt加载GStreamer库后运行 - “播放教程 1:Playbin 使用“(附:完整源码)

运行效果 介绍 我们已经使用了这个元素&#xff0c;它能够构建一个完整的播放管道&#xff0c;而无需做太多工作。 本教程介绍如何进一步自定义&#xff0c;以防其默认值不适合我们的特定需求。将学习&#xff1a; • 如何确定文件包含多少个流&#xff0c;以及如何切换 其中。…...

Python+DeepSeek:开启AI编程新次元——从自动化到智能创造的实战指南

文章核心价值 技术热点:结合全球最流行的编程语言与国产顶尖AI模型实用场景:覆盖代码开发/数据分析/办公自动化等高频需求流量密码:揭秘大模型在编程中的创造性应用目录结构 环境搭建:5分钟快速接入DeepSeek场景一:AI辅助代码开发(智能补全+调试)场景二:数据分析超级助…...

使用OpenCV和MediaPipe库——驼背检测(姿态监控)

目录 驼背检测的运用 1. 驾驶姿态与疲劳关联分析 2. 行业应用案例 1. 教育场景痛点分析 2. 智能教室系统架构 代码实现思路 1. 初始化与配置 2. MediaPipe和摄像头设置 3. 主循环 4. 资源释放 RGB与BGR的区别 一、本质区别 二、OpenCV的特殊性 内存结构示意图&…...

maven的项目构建

常用构建命令 命令说明mvn clean清理编译结果&#xff08;删掉target目录&#xff09;mvn compile编译核心代码&#xff0c;生成target目录mvn test-compile编译测试代码&#xff0c;生成target目录mvn test执行测试方法mvn package打包&#xff0c;生成jar或war文件mvn insta…...

光电感知赋能智能未来 灵途科技护航新质生产力发展

2024年《政府工作报告》将大力推进现代化产业体系建设&#xff0c;加快发展新质生产力作为首要工作任务。这是“新质生产力”首次出现在《政府工作报告》中。 发展新质生产力具体包括 新兴产业 &#xff1a;推动商业航天、低空经济等新兴产业实现安全健康发展。 未来产业 &a…...

文件上传靶场(10--20)

目录 实验环境&#xff1a; 具体内容实现&#xff1a; 第十关&#xff08;双写绕过&#xff09;&#xff1a; 第十一关&#xff1a;&#xff08;%00截断&#xff0c;此漏洞在5.2版本中&#xff09; 正确用法 错误用法 思路&#xff1a; 操作过程&#xff1a; 第十二关…...

deepseek在pycharm中的配置和简单应用

对于最常用的调试python脚本开发环境pycharm&#xff0c;如何接入deepseek是我们窥探ai代码编写的第一步&#xff0c;熟悉起来总没坏处。 1、官网安装pycharm社区版&#xff08;免费&#xff09;&#xff0c;如果需要安装专业版&#xff0c;需要另外找破解码。 2、安装Ollama…...

Linux 生成静态库

文章目录 前提小知识生成和使用.a库操作步骤 在应用程序中&#xff0c;有一些公共的代码需要反复使用的&#xff0c;可以把这些代码制作成“库文件”&#xff1b;在链接的步骤中&#xff0c;可以让链接器在“库文件”提取到我们需要使用到的代码&#xff0c;复制到生成的可执行…...

yolo-TensorRT相关代码逐步详解-pt转engine

基于TensorRT 的推论运行速度会比仅使用CPU 快40倍,提供精度INT8 和FP16 优化,支援TensorFlow、Caffe、Mxnet、Pytorch 等深度学习框架,其中Mxnet、Pytorch 需先转换为ONNX 格式。 TensorRT的构建流程大致分为几个步骤:创建构建器和网络、解析模型、配置构建参数、构建引擎…...

简记_ MCU管脚的防静电处理

一、分析&#xff08;一&#xff09; 接口处的信号要先过 ESD/TVS 管&#xff0c;然后拉到被保护器件&#xff1b; 建个 ESD 电路发生器的模型&#xff0c;代入到我们的电路中去分析&#xff1a; 继电器实现这两个“开关”&#xff0c;并且还会感应出一些额外的RLC寄生。 ES…...

C语言实现算法(二)

以下是 “10个不重复的C语言经典算法案例“,包含可运行代码、开发环境配置及系统要求。所有代码基于标准C语法,已在GCC 9.3.0环境下测试通过。 开发环境配置 编译器:GCC(推荐) Windows:安装 MinGW 或 Visual Studio Linux:sudo apt-get install gcc macOS:通过Xcode Co…...

transformer模型介绍——大语言模型 LLMBook 学习(二)

1. transformer模型 1.1 注意力机制 **注意力机制&#xff08;Attention Mechanism&#xff09;**在人工智能中的应用&#xff0c;实际上是对人类认知系统中的注意力机制的一种模拟。它主要模仿了人类在处理信息时的选择性注意&#xff08;Selective Attention&#xff09;&a…...