js算法之冒泡排序、快速排序和洗牌算法

随着前端的发展和进步,对前端的要求相应提高,作为前端开发不仅需要了解常规的业务逻辑,产品思维,还需要对算法有了解和思考,也是前端进阶高级的必经之路。最近的面试中经常遇到排序算法的题目,忙里偷闲整理了一下冒泡和快速排序。

冒泡排序原理

通过相邻两个元素之间的比较和交换,如果后面的比前面的小,则将小的元素排到前面, 反之排到后面,依照这个规则进行多次并且递减的迭代,直到顺序正确。

// 冒泡排序
function bubbleSort(arr){
	for(let i=arr.length-1; i>0; i--){
		for(let j=0; j<i; j++) {
			// 相邻元素对比,如果前面的数字比后面的大,则交换
			if (arr[j] > arr[i]){
				[arr[i], arr[j]] = [arr[j], arr[i]]
			}
		}
	}
	return arr
}

let arr = [1,3,2,9,5,8,6,7]

bubbleSort(arr) // [1, 2, 3, 5, 6, 7, 8, 9]

快速排序原理

元素的比较和交换是从两端向中间进行的,从数组中选定一个基数,把数组中的每一项与此基数做比较,小的放入left数组,大的放入right数组,然后再采用这样的方法操作新数组。直到所有子集只剩下一个元素,排序完成。

// 快速排序
function quickSort(arr){
	// 停止递归条件
	if (arr.length <=1) {return arr}
	
	// 获取中间值索引
	var pivotIndex = Math.floor(arr.length / 2)
	
	// 利用索引取出中间值
	var pivot = arr.splice(pivotIndex, 1)[0]
	
	// 定义左右数组,比基准小的放在left数组,比基准大的放入right数组
	var left = [], right = []
	
	// 遍历数组,进行判断分配
	for (var i=0; i<arr.length;i++){
		if (arr[i] < pivot) {
			left.push(arr[i])
		} else {
			right.push(arr[i])
	}
  }
  // 对数组进行合并操作
  return quickSort(left).concat([pivot], quickSort(right))
}

let arr = [1,3,2,9,5,8,6,7]

quickSort(arr) // [1, 2, 3, 5, 6, 7, 8, 9]

洗牌算法原理

洗牌算法是将原来的数组通过random进行打散,从原数组中随机抽取一个元素放入新数组,得到一个新数组。(应用场景:音乐随机播放、棋牌类游戏洗牌)

// 洗牌算法
function shuffle(arr){
	let newArr = [], random
	
	while(arr.length >0){
        // 获取随机数
		random = Math.floor(Math.random() * arr.length)
  
        // 将随机数值放入新数组中
		newArr.push(arr[random])

        // 删除原数组值
		arr.splice(random, 1)
	}
	return newArr
}

let arr = [1,2,3,4,5,6]

shuffle(arr) // [5, 2, 3, 1, 4, 6]

时间复杂度
最佳情况:当数组已经排序时,冒泡排序只需要进行一轮,复杂度为 O(n)
平均情况:冒泡排序需要进行 n-1轮,每轮需要进行大约 n-i 次比较,复杂度为 O(n²)


空间复杂度
冒泡排序是原地排序算法,只需要使用常数级别的额外空间来交换元素。因此,其空间复杂度为 O(1)

总体来说,快速排序的效率明显高于冒泡排序,特别是在处理大规模数据时。快速排序在大多数情况下的时间复杂度为 O(n log n),而冒泡排序的时间复杂度为 O(n²)。因此,除非在一些非常特殊的情况下(如小规模数组或几乎已经有序的数组),快速排序通常比冒泡排序更常用,因为它具有更好的性能表现和更低的时间复杂度。

  • 5
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值