【数据结构】深入浅出理解快速排序背后的原理 以及 版本优化【万字详解】(C语言实现)

news2024/11/25 2:15:08

快速排序

  • 快速排序递归实现
  • 前言
  • 一、Hoare版本
  • (一)算法运行图例
  • (二)算法核心思路
  • (三)算法实现步骤
    • (1)单趟
    • (2)多趟
  • (四)码源详解 递归实现
    • (1)Hoare版本:先行版
    • (2)Hoare版本:发行版
  • (五)Hoare算法 实现背后的理论支持
  • (六)效率优化
    • Hoare排序的效率分析
    • (1)时间复杂度 O(N*logN)
    • 稳定性: 不稳定
  • (七)代码优化
  • (1)三数取中
    • 1)优化思路
  • 2)三数取中优化 后的代码
    • 3)效果对比
  • (2)小区间优化—— 小区间不再进行递归分割排序,降低递归次数
    • 1)优化思路
  • 2)整体优化后的完整代码
  • 二、挖坑法版本
  • 前言
  • (一)算法运行图例
  • (二)算法核心思路
  • (三)算法实现步骤
    • (1)单趟
    • (2)多趟
  • (四)码源详解
  • 三、前后指针版本
  • (一)算法运行图例
  • (二)算法核心思路
  • (三)算法实现步骤
    • (1)单趟
    • (2)多趟
  • (四)码源详解



快速排序递归实现

前言

快速排序是Hoare于1962年提出的一种 二叉树结构交换排序方法。


一、Hoare版本



(一)算法运行图例

在这里插入图片描述



(二)算法核心思路

任取待排序元素序列中的某元素作为基准值,按照该排序码将待排序集合 分割成两子序列左子序列中所有元素均小于基准值,右子序列中所有元素均大于基准值,然后最左右子序列重复该过程,直到所有元素都排列在相应位置上为止。



(三)算法实现步骤

(1)单趟

  1. 找一key 【注意:我们实际上控制的是key的下标keyi,而不是用新开辟的变量去储存它的值,用新开辟的变量去储存,最后交换的也只是与该变量进行交换,而不是在数组中与原数组的值进行交换】
    【图讲解】
  2. begin从左开始遍历,左边找比key大的。end从右开始遍历,右边找比key小的。等于key的值不动。
  3. 两边都找到了,互换
  4. 最后相遇的地方(就是key排好序后要放的位置(正确的位置))与key互换 【有理论支持的,请看下文】

此时 【左边的都比key小】中间key【右边的都比key大】
要是左右要是都有序 =》整体实现有序 (多趟解决)

(2)多趟

  1. 对排好的key的左边再进行单趟排序
  2. 对排好的key的右边再进行单趟排序
  3. 左边右边分别再排好两边key的位置,再分左右,再进行单趟

类似 二叉树前序递归的思想:根、左子树、右子树。



(四)码源详解 递归实现

(1)Hoare版本:先行版

按照思路写出来 了,大家看看都存在些什么问题。

//递归实现
// Hoare版本(先行版)
int PartSort1(DataType* a,int left,int right) {

	int key = a[left];          
	 
	while (left < right) {     //相遇是跳出循环的条件     //right 先走
		if (a[key] < a[right]) {      
			right--;          
		}                                              //找到a[right]<a[keyi]的情况就停下

		if (a[left] < a[key]) {      //left 同理
			left++;
		}

		
		Swap(&a[left], &a[right]);       //都找到后互换
	}

	Swap(&a[left], &a[key]);//相遇后跳出循环 交换a[keyi]和left,right相遇的位置
}
  • 坑1:a[key] < a[right],若a[key] 一直比 a[right]要小,则right则回一直向左right–遍历,则会出现越界的问题(同理left那边也是)

    • 解决方法:left<right,控制right不要越界,避免该序列本来就是有序(一直比a[keyi]要大而一直right–,越出边界的情况)的情况
  • 坑2:a[keyi] < a[right],若不跳则要是两边都遇到相等的值,则两边无法再进入if()语句进行right–或left++,而进入死循环

    • 解决方法:a[keyi]<=a[right]时,都跳

  • 坑3:int key = a[left]; Swap(&a[left], &a[key]); 真正在换的是哪个?
    key是局部变量,相当于在栈区开辟了一块空间,用于存储a[left]的值。实际与数组a[left]进行交换的,是局部变量这块空间里面的值,对数组原始的最左边并不产生任何影响

    • 解决方法:int left = keyi; 记录的应该是下标,通过控制下标来达到改变原数组


(2)Hoare版本:发行版

// Hoare版本(发行版)
int PartSort1(DataType* a, int left, int right) {

	int keyi = left;          //保存left的下标

	while (left < right) {     //相遇是跳出循环的条件     //right 先走
		if (left < right && a[keyi] <= a[right]) {      //left<right,控制right不要越界,避免该序列本来就是有序(一直比a[keyi]要大而一直right--,越出边界的情况)的情况
			right--;
		}                                              //找到a[right]<a[keyi]的情况就停下

		if (left < right && a[left] <= a[keyi]) {      //left 同理
			left++;
		}


		Swap(&a[left], &a[right]);       //都找到后互换
	}

	Swap(&a[left], &a[keyi]);//相遇后跳出循环 交换a[keyi]和left,right相遇的位置
}



(五)Hoare算法 实现背后的理论支持

★☆ 最后相遇的地方 就是key排好序后要放的位置(正确的位置)是怎么做到的呢?
右边先走做到的。
在这里插入图片描述

有了上面的理论保证,停下来的值一定比key小,才敢进行交换。


(六)效率优化

Hoare排序的效率分析

(1)时间复杂度 O(N*logN)

如果每次选到的 key=a[ left ] 都是中位数,这将会非常高效
【因为每单趟排好一次,key就能落入到其正确的位置】

  • 理想状态下是:满二叉树的分割【二分 O(N*logN)】

  • 而最差的情况:有序(接近有序)【每次取到的key都选到整个数组中次大或次小的数,每个key,left或right都要遍历一遍数组, O(N^2)】 但随机取的值肯定不是像这样有序(或接近有序)的(这种都是已经给你排好了的)

在这里插入图片描述

所以总体来说,Hoare在时间复杂度上是更接近于 O(N*logN) 的 。

稳定性: 不稳定



(七)代码优化

(1)三数取中

那么我们针对序列可能会出现最差情况(有序的情况),做一个程序优化——三数取中 。

1)优化思路

int mid=(left+right)/ 2 ;
对比 a[mid]、a[left]、a[right] 三个值,取值大小为中间的那个数 。【大大降低了取两边极端的可能,让取到的数更偏向于有序序列中间的数】

//三数取中 取的不是位置在中间的值,而是值大小位于中间的值
int Getmidi(DataType* a,int left,int right) {
	int mid = (left + right) / 2;
	
	//left mid right
	if (a[left] < a[mid]) {
		if (a[mid] < a[right])       //mid为中间值
			return mid;
		else if (a[left] > a[right]) {    //mid为最大值
			return left;                  //left为中间值
		}
		else {
			return right;
		}
	}
	else { //a[left]>a[mid]
		if (a[mid] > a[right]) {         //mid为中间值
			return mid;
	    }
		else if (a[left]<a[right]) {     //left为中间值,mid最小
			return left;
		}
		else {
			return right;
		}
	}

}


2)三数取中优化 后的代码

//三数取中 取的不是位置在中间的值,而是值大小位于中间的值
int Getmidi(DataType* a,int left,int right) {
	int mid = (left + right) / 2;
	
	//left mid right
	if (a[left] < a[mid]) {
		if (a[mid] < a[right])       //mid为中间值
			return mid;
		else if (a[left] > a[right]) {    //mid为最大值
			return left;                  //left为中间值
		}
		else {
			return right;
		}
	}
	else { //a[left]>a[mid]
		if (a[mid] > a[right]) {         //mid为中间值
			return mid;
	    }
		else if (a[left]<a[right]) {     //left为中间值,mid最小
			return left;
		}
		else {
			return right;
		}
	}

}


// Hoare版本(发行版)
int PartSort1(DataType* a, int left, int right) {
	//三数取中
	int midi = Getmidi(a, left, right);
	Swap(&a[left], &a[midi]);   //交换a[left]与a[midi]值的位置     

	int keyi = left;          //保存left的下标

	while (left < right) {     //相遇是跳出循环的条件     //right 先走
		if (left < right && a[keyi] <= a[right]) {      //left<right,控制right不要越界,避免该序列本来就是有序(一直比a[keyi]要大而一直right--,越出边界的情况)的情况
			right--;
		}                                              //找到a[right]<a[keyi]的情况就停下

		if (left < right && a[left] <= a[keyi]) {      //left 同理
			left++;
		}

		//坑1:a[keyi] < a[right],若不跳则要是两边都遇到相等的值,则两边无法再进入if()语句进行right--或left++,而进入死循环
		//更正:a[keyi]<=a[right]时,都跳

		Swap(&a[left], &a[right]);       //都找到后互换
	}

	Swap(&a[left], &a[keyi]);//相遇后跳出循环 交换a[keyi]和left,right相遇的位置
}



3)效果对比

对 有序/接近有序 的序列中优化效果明显
对 随机 就正常。
在这里插入图片描述



(2)小区间优化—— 小区间不再进行递归分割排序,降低递归次数

1)优化思路

满二叉树整体节点的个数:2^h-1等比数列求和 )。
最后一层的节点数就已经占了整体的 50%
最后三层的节点数就已经约占整体的 80% 的节点个数了 。

那么最后这三层约占整体80%的节点数,若采用递归的方式,将是对栈帧的一个很大的开销。因为我们最后这三层区间被分的比较小了,若采用更适用于小区间数排序的优化,无疑是对效率很大的提升!

那么小区间优化我们可以选择哪个小区间数排序方式呢?
希尔排序:数据大时比较有序(目的:让大的数更快的到后面去)
插入排序:少的数(最坏情况:逆序,比所有数都小)

因为小区间,数比较少,对比起来比较快,所以这里选择 插入排序 更快速便捷高效。
在这里插入图片描述

//小区间优化版本—— 小区间不再进行递归分割排序,降低递归次数
void QuickSort1(DataType* a, int begin, int end) {
	if (begin >= end)
		return;

	//10个数就已经占3层递归了,而递归层数中,尾三层所占节点的总数 约占80%,这尾三层再继续用递归的方法,而改用小区间数据排序优化的思路,能让程序整体得到进一步的优化
	if ((end - begin + 1) > 10) {
		int keyi = PartSort3(a, begin, end);
		QuickSort1(a, begin, keyi - 1);
		QuickSort1(a, keyi+1, end);
	}
	else {      //改用小区间数据优化排序
		InsertSort(a + begin, end - begin + 1);    // a+begin 找到对应数组所在的位置
	}
}


2)整体优化后的完整代码

//三数取中 取的不是位置在中间的值,而是值大小位于中间的值
int Getmidi(DataType* a,int left,int right) {
	int mid = (left + right) / 2;
	
	//left mid right
	if (a[left] < a[mid]) {
		if (a[mid] < a[right])       //mid为中间值
			return mid;
		else if (a[left] > a[right]) {    //mid为最大值
			return left;                  //left为中间值
		}
		else {
			return right;
		}
	}
	else { //a[left]>a[mid]
		if (a[mid] > a[right]) {         //mid为中间值
			return mid;
	    }
		else if (a[left]<a[right]) {     //left为中间值,mid最小
			return left;
		}
		else {
			return right;
		}
	}

}



// Hoare版本(发行版)
int PartSort1(DataType* a, int left, int right) {
	//三数取中
	int midi = Getmidi(a, left, right);
	Swap(&a[left], &a[midi]);   //交换a[left]与a[midi]值的位置     

	int keyi = left;          //保存left的下标

	while (left < right) {     //相遇是跳出循环的条件     //right 先走
		if (left < right && a[keyi] <= a[right]) {      //left<right,控制right不要越界,避免该序列本来就是有序(一直比a[keyi]要大而一直right--,越出边界的情况)的情况
			right--;
		}                                              //找到a[right]<a[keyi]的情况就停下

		if (left < right && a[left] <= a[keyi]) {      //left 同理
			left++;
		}

		//坑1:a[keyi] < a[right],若不跳则要是两边都遇到相等的值,则两边无法再进入if()语句进行right--或left++,而进入死循环
		//更正:a[keyi]<=a[right]时,都跳

		Swap(&a[left], &a[right]);       //都找到后互换
	}

	Swap(&a[left], &a[keyi]);//相遇后跳出循环 交换a[keyi]和left,right相遇的位置
}



//小区间优化版本—— 小区间不再进行递归分割排序,降低递归次数
void QuickSort1(DataType* a, int begin, int end) {
	if (begin >= end)
		return;

	//10个数就已经占3层递归了,而递归层数中,尾三层所占节点的总数 约占80%,这尾三层再继续用递归的方法,而改用小区间数据排序优化的思路,能让程序整体得到进一步的优化
	if ((end - begin + 1) > 10) {
		int keyi = PartSort3(a, begin, end);
		QuickSort1(a, begin, keyi - 1);
		QuickSort1(a, keyi+1, end);
	}
	else {      //改用小区间数据优化排序
		InsertSort(a + begin, end - begin + 1);    // a+begin 找到对应数组所在的位置
	}
}


二、挖坑法版本

前言

挖坑法是Hoare排序的一个思路上的优化:不用再考虑为什么右边先走。
因为一开始的坑就在左边,那么一开始就得先从右边开始找,找比坑小的数。

(一)算法运行图例

在这里插入图片描述



(二)算法核心思路

和Hoare的思路一样,只不过这里是先挖好一个坑,再开始排。如果在左边挖坑,则从右边开始找;如果在右边挖坑,则从左边开始找。


(三)算法实现步骤

(1)单趟

  1. int key = a[left]; //保存好key值以后,左边形成第一个坑
  2. 右边先走,找小,填到左边的坑,右边形成新的坑
  3. 左边再走,找大,填到右边的坑,左边形成新的坑位
  4. left、right 相遇,跳出循环。最终相遇,一定在坑上相遇(因为left、right其中一个必在坑位上)且该坑位就是key值该在的有序序列中正确的位置( 和Hoare版本的同理 ) 。将key保留的最开始坑位的值赋给现在left、right相遇所在的坑位。

(2)多趟

  1. 对排好的key的左边再进行单趟排序
  2. 对排好的key的右边再进行单趟排序
  3. 左边右边分别再排好两边key的位置,再分左右,再进行单趟


(四)码源详解

// 快速排序——挖坑法
int PartSort2(int* a, int left, int right) {
	//三数取中——取位值位于中间的值
	int midi = Getmidi(a, left, right);
	Swap(&a[left], &a[midi]);
    
	int key = a[left];      //保存好key值以后,左边形成第一个坑
	int hole = left; 

	//单趟完成的返回的条件
	while (left < right) {
    
    //右边先走,找小,填到左边的坑,右边形成新的坑
    while (left < right && key <= a[right]) {
		right--;
	}
	a[hole] = a[right];
	hole = right;

	//左边再走,找大,填到右边的坑,左边形成新的坑位
	while (left < right && a[left] <= key) {
		left++;
	}
	a[hole] = a[left];
	hole = left;

	}
	
	//最后把储存的key的值填入坑位,不用怕坑位的数被覆盖,因为原坑位的数在这之前就已经赋到其他地方了
	a[hole] = key;
	return hole;      //返回已经排好正确位置的坑位位置

}


三、前后指针版本

(一)算法运行图例

在这里插入图片描述



(二)算法核心思路

cur一直向前遍历,不管是遇到大的还是遇到小的。

而prev遇小的向前++,把小的交换过来;遇到大的就停下,等遇到小的时候就++将其交换覆盖。旨在把小的放其左边,大的放其右边。

cur越界后,遍历结束。此时 prev 所在的地方,正是key值该在的有序序列中正确的地方。


(三)算法实现步骤

(1)单趟

  1. 三数取中 _ 取中间值 赋给key
  2. int prev = left ; int cur = prev + 1;
    cur遇比key小的,交换Swap(&a[++prev],&a[cur]);(把小的放其左边)
  3. cur越界,遍历结束
  4. Swap ( &a[prev],&a[keyi] ); 此时 prev 所在的地方,正是key值该在的有序序列中正确的地方。

(2)多趟

  1. 对排好的key的左边再进行单趟排序
  2. 对排好的key的右边再进行单趟排序
  3. 左边右边分别再排好两边key的位置,再分左右,再进行单趟


(四)码源详解

  • 先行版
    这样写存在什么问题?
// 快速排序前后指针法 —— 先行版
int PartSort3(int* a, int left, int right) {
	int midi = Getmidi(a, left, right);
	Swap(&a[left], &a[midi]);

	int prev = left;
	int cur = prev + 1;

	int keyi = left;
	while (cur <= right) {     //cur越界结束
		while (cur <= right && a[cur] > a[keyi]) {      //没有碰到比key小的就一直向后遍历
			cur++;
		}

		//找到了
		++prev;                                         //遇小的了就++prev,Swap(&a[prev], &a[cur]);交换
		Swap(&a[prev], &a[cur]);
	}

	//cur越界后,代表遍历结束
	//在a[++prev] 的位置将key值赋予它(key该在的正确位置)
	Swap(&a[prev], &a[keyi]);
	return prev;
}

当cur一直没有再遇到小的,cur将会越界,而这仅会跳出内层(cur向后遍历)的循环,但程序仍在外层交换的循环中走。

所以cur越界后,程序仍要向下运行代码,这时交换的不再是数组中的数值,而是数组外后一位cur越界后位置的数值。
在这里插入图片描述

所以应不再进行交换。交换已经结束了。应跳出外层交换 a[prev] 和 a[cur] 的循环。进行下一步 a[prev] 与 a[keyi] 的交换,交换到key在有序中的正确位置。


  • 发行版(一)
    当判断出cur越界,即可break跳出循环,不再进行 a[prev] 与 a[cur] 的交换
// 快速排序前后指针法 —— 先行版
int PartSort3(int* a, int left, int right) {
	int midi = Getmidi(a, left, right);
	Swap(&a[left], &a[midi]);

	int prev = left;
	int cur = prev + 1;

	int keyi = left;
	while (cur <= right) {     //cur越界结束
		while (cur <= right && a[cur] > a[keyi]) {      //没有碰到比key小的就一直向后遍历
			cur++;
		}

		if (cur > right) {     //当判断出cur越界,即可break跳出循环,不再进行 a[prev] 与 a[cur] 的交换
			break;
		}

		//找到了
		++prev;                                         //遇小的了就++prev,Swap(&a[prev], &a[cur]);交换
		Swap(&a[prev], &a[cur]);
	}

	//cur越界后,代表遍历结束
	//在a[++prev] 的位置将key值赋予它(key该在的正确位置)
	Swap(&a[prev], &a[keyi]);
	return prev;
}


  • 发行版(二)
// 快速排序前后指针法
int PartSort3(int* a, int left, int right) {
	int midi = Getmidi(a, left, right);
	Swap(&a[left], &a[midi]);

	int prev = left;
	int cur = prev + 1;

	int keyi = left;
	while (cur <= right) {     //cur越界结束
		
		//写法一
		if (a[cur] < a[keyi]) {            
			Swap(&a[++prev],&a[cur]);    //在还没遇到大的之前,cur = prev + 1; a[++prev] = a[cur];数组自己与自己交换
			                             //遇小交换的意义:将比其小的值放左边,比其大的值放右边
		}

        //写法二
		if (a[cur] < a[keyi] && ++prev != cur); {    //如果不想要自己与自己交换的这种不必要的交换,++prev != cur条件也可以这么写
			Swap(&a[prev], &a[cur]);
 	}
		
		cur++;
	}

	//cur越界后,代表遍历结束
	//在a[prev] 的位置将key值赋予它(key该在的正确位置)
	Swap(&a[prev],&a[keyi]); 
	return prev;
}

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

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

相关文章

Servlet对象生命周期

Servlet 生命周期包括加载与实例化、初始化、服务请求、销毁等阶段。 ervlet 的生命周期包括以下阶段&#xff1a; 加载与实例化&#xff1a;当容器启动或者第一次请求到达时&#xff0c;Servlet 容器加载 Servlet 类并创建 Servlet 实例。 初始化&#xff1a;在 Servlet 实例…

线性回归与线性拟合的原理、推导与算法实现

关于回归和拟合&#xff0c;从它们的求解过程以及结果来看&#xff0c;两者似乎没有太大差别&#xff0c;事实也的确如此。从本质上说&#xff0c;回归属于数理统计问题&#xff0c;研究解释变量与响应变量之间的关系以及相关性等问题。而拟合是把平面的一系列点&#xff0c;用…

UUNet训练自己写的网络

记录贴写的很乱仅供参考。 自己写的Unet网络不带深度监督&#xff0c;但是NNUNet默认的训练方法是深度监督训练的&#xff0c;对应的模型也是带有深度监督的。但是NNUNetV2也贴心的提供了非深度监督的训练方法在该目录下&#xff1a; 也或者说我们想要自己去定义一个nnUNWtTra…

pyusb环境搭建和无法发包问题

pyusb环境搭建和无法发包问题 项目需要对usb设备进行开发调试&#xff0c;选择搭建pyusb环境进行调试测试&#xff0c;这里记录下完整流程和中间解决的一些问题。 我使用的环境是window10 64bit, vscode 1.84.0 , Python 3.11.6 1 安装流程 参考github上的 https://github.…

实用篇-MQ消息队列

一、初识MQ 通讯分为同步通讯和异步通讯&#xff0c;同步通讯就比如我们日常生活中的打电话&#xff0c;看直播&#xff0c;能够得到及时的反馈。而异步通讯则类似于聊天软件聊天&#xff0c;不需要建立实时的连接&#xff0c;并且可以进行建立多个业务一起异步执行 1. 同步通…

关于SNAP的Biophysical Processor模块的计算准确率以及大厂10月种植情况

关于SNAP的Biophysical Processor模块的计算准确率 在处理河北省2022年的10月6日影像&#xff0c;使用SNAP的Biophysical Processor计算LAI时 发现很多农田地块出现了缺失值&#xff0c;但其实就是0值 SNAP的这个模块基于PROSAIL物理模型反演。不得不说&#xff0c;还是挺准…

AI:56-基于深度学习的微表情识别

🚀 本文选自专栏:AI领域专栏 从基础到实践,深入了解算法、案例和最新趋势。无论你是初学者还是经验丰富的数据科学家,通过案例和项目实践,掌握核心概念和实用技能。每篇案例都包含代码实例,详细讲解供大家学习。 📌📌📌在这个漫长的过程,中途遇到了不少问题,但是…

聚观早报 |盒马参战双11;真我GT5 Pro将压轴登场

【聚观365】11月4日消息 盒马参战双11 真我GT5 Pro将压轴登场 奇瑞汽车10月销量创新高 iQOO 12系列将首发电竞芯片Q1 苹果CEO库克称正改善供需平衡 盒马参战双11 不少消费者反映&#xff0c;今年盒马的双11已悄然开始&#xff1a;10月20日起&#xff0c;盒马APP很多商品页…

COE文件之读写操作

在Xilinx的FIR、Block Memory等 IP核的配置中&#xff0c;需要使用COE&#xff08;Coefficient&#xff09;文件来进行初始化。 写COE文件 一般是通过Matlab设计好参数后&#xff0c;再生成COE文件。具体代码如下。 x 1:512; fid fopen(test.coe,w); fprintf(fid, memory_…

MachineLearning 14. 机器学习之集成分类器(AdaBoost)

这期介绍一下NB的最佳集成分类方法之一 AdaBoost&#xff0c;并实现在具体数据集上的应用&#xff0c;尤其是临床数据。 简 介 Adaboost是Adaptive Boosting的缩写&#xff0c;使用一组简单的弱分类器&#xff0c;通过强调被弱分类器错误分类的样本来实现改进的分类器。AdaBoo…

Java金字塔、空心金字塔、空心菱形

Java金字塔 public class TestDemo01 {public static void main(String[] args){//第一个for用于每行输出 从i1开始到i<5,总共5行for(int i1;i<5;i){//每行前缀空格&#xff0c;这个for用于表示每行输出*前面的空格//从上面规律可得,每行输出的空格数为总层数&#xff0c…

【计算机网络】金管局计算机岗位——计算机网络(⭐⭐⭐⭐)

计算机网络知识点 计算机网络基础知识计算机网络的定义与组成、分类网络的发展、常识&#xff08;⭐⭐⭐⭐&#xff09;计算机网络的定义计算机网络的功能计算机网络的组成计算机网络的分类计算机网络的性能指标主要包括&#xff08;⭐⭐⭐⭐&#xff09; 网络体系结构OSI模型定…

英伟达发布 Windows 版 TensorRT-LLM 库

导读英伟达发布了 Windows 版本的 TensorRT-LLM 库&#xff0c;称其将大模型在 RTX 上的运行速度提升 4 倍。 GeForce RTX 和 NVIDIA RTX GPU 配备了名为 Tensor Core 的专用 AI 处理器&#xff0c;正在为超过 1 亿台 Windows PC 和工作站带来原生生成式 AI 的强大功能。 Tens…

Python笔记——linux/ubuntu下安装mamba,安装bob.learn库

Python笔记——linux/ubuntu下安装mamba&#xff0c;安装bob.learn库 一、安装/卸载anaconda二、安装mamba1. 命令行安装&#xff08;大坑&#xff0c;不推荐&#xff09;2. 命令行下载guihub上的安装包并安装&#xff08;推荐&#xff09;3. 网站下载安装包并安装&#xff08;…

R语言中的自带的调色板--五种--全平台可用

R语言中的自带的调色板–五种–全平台可用

YOLOv5论文作图教程(2)— 软件界面布局和基础功能介绍

前言:Hello大家好,我是小哥谈。通过上一节课的学习,相信大家都已成功安装好软件了,本节课就给大家详细介绍一下Axure RP9软件的界面布局及相关基础功能,希望大家学习之后能够有所收获!🌈 前期回顾: YOLOv5论文作图教程(1)— 软件介绍及下载安装(包括软件包+下载安…

Java字符串常用函数 详解5000字 (刷题向 / 应用向)

1.直接定义字符串 直接定义字符串是指使用双引号表示字符串中的内容&#xff0c;例如"Hello Java"、"Java 编程"等。具体方法是用字符串常量直接初始化一个 String 对象&#xff0c;示例如下&#xff1a; 1. String str"Hello Java"; 或者 …

生成m3u8视频:批量剪辑与分割的完美结合

在视频处理领域&#xff0c;m3u8视频格式的出现为高效处理和优化视频内容提供了新的可能。尤其在批量剪辑和分割视频的过程中&#xff0c;掌握m3u8视频的生成技巧&#xff0c;意味着更高效的工作流程和更出色的创作效果。现在一起来看看云炫AI智剪如何生成m3u8视频的操作吧。 步…

代码生成器

Easycode Entity ##导入宏定义 $!{define.vm}##保存文件&#xff08;宏定义&#xff09; #save("/entity", ".java")##包路径&#xff08;宏定义&#xff09; #setPackageSuffix("entity")##自动导入包&#xff08;全局变量&#xff09; $!{au…

港科夜闻|香港科大戴希教授被选为腾讯公司新基石研究员

关注并星标 每周阅读港科夜闻 建立新视野 开启新思维 1、香港科大戴希教授被选为腾讯公司“新基石研究员”。10月30日&#xff0c;作为目前国内社会力量资助基础研究力度最大的公益项目之一&#xff0c;“新基石研究员项目”揭晓了第二期获资助名单&#xff0c;来自13个城市28家…