数组随机排序

如何将一个数组随机排序?

这个题目很有意思,我在直播的时候看的,大家讨论如何将一个数组进行随机排序。然后我想到的是sort+Math.random()

也就是下面这样:

function shuffle(array) {return array.sort(() => Math.random() - 0.5)
}

利用数组的sort算法,比较函数返回一个-0.5到0.5的随机数,这个方法看似可行,其实有很大的问题。

V8 sort

从v8 bloghttps://v8.dev/blog/array-sort上可以看到,v8 sort现在是采用Timsort算法,TimSort算法是一种起源于归并排序和插入排序的混合排序算法;而在以前,则是采用插入排序(长度小于10时)+快排的算法.

既然怎样都跑不开插入排序,那就以插入排序为排序算法,以数组[1, 2, 3] 为样例,计算得出通过random sort后得到[1, 2, 3]的概率为25%(第一次插入得到[1, 2]概率为50%,得到[1, 2]后3直接插入队尾的概率为50%),而我们知道,真正的随机算法得到[1, 2, 3]的概率应该为1/6.

除了插入排序,其他排序算法也会有这些问题,可以在这个网站上测试你的随机排序算法:https://bost.ocks.org/mike/shuffle/compare.html

改进

function shuffle(array) {const random = {}array.forEach(item => {random[item] = Math.random()})return array.sort((a, b) => {return random[a] - random[b]})
}

这样做解决了上述问题,每个数在比较时的概率不再独立。但也有缺点:会有额外的空间,时间复杂度为O(nlogn),有没有更好的方法呢?

再次改进:Fisher–Yates算法

Fisher–Yates算法是由Ronald Fisher和Frank Yates共同提出,其用js的实现如下:

function shuffle(array) {let n = array.length;while (n > 0) {const i = Math.floor(Math.random() * n--)const tmp = array[i]array[i] = array[n]array[n] = tmp}return array
}

核心思想:每次从剩余的项里随机取一个项,记录并剔除该项再进入下次循环.

Fisher–Yates算法做到了不额外申请空间&O(n)的时间复杂度,可以说是最优的随机排序算法了.

再次优化一下

function shuffle(array) {for (let i = array.length; i>= 0; i--) {let curIndex = Math.floor(i * Math.ramdon())[array[i], array[curIndex]] = [array[curIndex], array[i]]}return array
}

这样可以利用数组解构方法, 少一个中间变量, 这就是最终版本了, 还有其他更好的办法吗?欢迎大家评论区讨论一下?

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

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

相关文章

Golang reflect.MakeFunc() 的用法及示例

Golang 作为一门强类型语言,在某些场景下,我们需要动态地创建函数或者修改函数,这个时候就可以使用反射的方法去实现。在反射中,我们可以使用 reflect.MakeFunc() 方法来创建一个新的函数,本文我将介绍使用反射及其 Ma…

NX二次开发,在指定的装配节点下新建或者添加组件

目录 一、概述 二、存在的问题 三、在指定的装配节点下新建或者添加组件的实现 四、结果显示 一、概述 最近学习装配内容,说一说体会吧,用起来很爽,开发起开是真的别扭啊,而且,网上可以收到的关于装配的知识有很多,都是简单的添加组件,移动组件,删除组件,等等一些…

【Linux】-网络请求和下载、端口[6]

目录 一、网络请求和下载 1、ping命令 2、wget命令 3、curl命令 二、端口 1、虚拟端口 2、查看端口占用 一、网络请求和下载 1、ping命令 可以通过ping命令,检查指定的网络服务器是否可联通状态 语法:ping [ -c num ] ip或主机名 选项&…

Flutter3.x get-cli中运行get init初始化项目报错如何处理

Flutter get-cli中运行get init初始化项目会提示如下错误: get init s E:\flutter\flutter study\tempstudy\misapp01> get init 1)Getx Pattern (by Kau) 2)CLEAN (by Arktekko) which architecture do you want to use? [1] unhandled exception: Synchromu…

Fendi club啤酒与精酿文化的碰撞与整合

在啤酒的世界中,精酿啤酒文化正逐渐崛起,成为了品质和品味的象征。作为精酿啤酒的代表,Fendi club啤酒与精酿文化的碰撞与整合,不仅推动了啤酒产业的发展,更是一种全新的生活方式。 Fendi club啤酒一直秉持着对品质的追…

从零自制docker-12-【overlayfs】

文章目录 overlayfsexec.Command("tar", "-xvf", busyboxTarURL, "-C", busyboxURL).CombinedOutput()exec.Command格式差异 挂载mount卸载unmount代码地址结果演示 overlayfs 就是联合文件系统,将多个文件联合在一起成为一个统一的…

内存拆解分析表:学习版[图片]

对拆解system中主要是对比测试机和对比机之间的差距,测试机那些地方高于对比机 拆解表,作为理解 在拆解表中system测试机比对比机多出113M 这说明是有问题的 对system拆解: system12345对比机9102294380941069391081628测试机10252010331…

WHAT - 前端安全性测试和常见攻击手段

目录 一、安全性测试二、前端安全性测试三、跨站脚本(XSS)攻击1. 介绍2. 三大类型反射型 XSS(Reflected XSS)存储型 XSS(Stored XSS)DOM 型 XSS(DOM-based XSS) 3. xss 盲打4. xss 水…

代码随想录第五十天|最佳买卖股票时机含冷冻期、买卖股票的最佳时机含手续费

题目链接:. - 力扣(LeetCode) 最佳买卖股票时机含冷冻期与打家劫舍的题目有异曲同工之妙,主要是出现了天数的间隔,一次需要在买卖股票的最佳时机II 题目上做一点调整,代码如下: 如代码所示&…

unity给物体添加可以包裹所有子物体的BoxCollider

代码如下可直接调用 MeshTool.SpawnCollider(mode);using UnityEngine;public class MeshTool {public static Bounds SpawnCollider(Transform target){Vector3 pMax Vector3.zero;Vector3 pMin Vector3.zero;Vector3 center Vector3.zero;Vector3 oldPos target.transfor…

Linux系统编程——进程控制

目录 一,进程创建 1.1 fork回顾 1.2 写时拷贝 1.3 fork用处 1.4 fork调用失败原因 二,进程退出 2.1 进程退出场景 2.2 mainCRTStartup调用 2.3 进程退出码 2.3.1 main函数返回值 2.3.2 strerror ​编辑 2.3.3 命令的退出码 2.4 进程正常退…

【第19章】spring-mvc之全局异常处理

文章目录 前言一、全局异常处理1. 前端2. 后端 二、常见错误页1.增加界面2.web.xml3.异常处理4.效果 总结 前言 例如:随着人工智能的不断发展,机器学习这门技术也越来越重要,很多人都开启了学习机器学习,本文就介绍了机器学习的基…

go使用redis连接池技术操作redis数据库实例

我们都知道,在程序开发中,网络链接的开销是比较大的, 当我们在链接redis 时,如果是操作一次就执行一次DIal拨号,那性能是很低的,redis的效率瞬间就被你降低了50%, 提速redis使用性能的第一步就是…

CSS-伪类选择器

结构伪类选择器 作用&#xff1a;根据元素的结构关系查找元素 分类&#xff1a; 选择器说明元素名:first-child查找第一个元素元素名:last-child查找最后一个元素元素名:nth-child(N)查找第N名元素 <!DOCTYPE html> <html lang"en"> <head><me…

react框架对Excel文件进行上传和导出

1.首先需要安装xlsx第三方的库库 引入插件 npm install xlsx在react引入 import * as XLSX from xlsx;1&#xff0c;首先设置jsx部分的 以下代码包含有导入excel文件和导出excel文件&#xff0c;读着可以根据需要&#xff0c;自己选择想要实现的功能 代码如下&#xff0…

动态代理(JDK、Cglib、Bytebuddy)

文章目录 动态代理JDK动态代理代理对象的生成动态代理执行流程简化代码静态代理 Cglib动态代理代理类的生成动态代理执行流程 bytebuddy常用api生成一个类对实例方法进行插桩实例方法进行插桩的扩展三种增强方式插入新方法方法委托动态修改入参对构造方法进行插桩对静态方法进行…

【北京迅为】《iTOP-3588从零搭建ubuntu环境手册》-第3章 Ubuntu20.04系统设置

RK3588是一款低功耗、高性能的处理器&#xff0c;适用于基于arm的PC和Edge计算设备、个人移动互联网设备等数字多媒体应用&#xff0c;RK3588支持8K视频编解码&#xff0c;内置GPU可以完全兼容OpenGLES 1.1、2.0和3.2。RK3588引入了新一代完全基于硬件的最大4800万像素ISP&…

【论文阅读笔记】jTrans(ISSTA 22)

个人博客地址 [ISSTA 22] jTrans&#xff08;个人阅读笔记&#xff09; 论文&#xff1a;《jTrans: Jump-Aware Transformer for Binary Code Similarity》 仓库&#xff1a;https://github.com/vul337/jTrans 提出的问题 二进制代码相似性检测&#xff08;BCSD&#xff0…

2024数维杯数学建模B题完整论文讲解(含每一问python代码+结果+可视化图)

大家好呀&#xff0c;从发布赛题一直到现在&#xff0c;总算完成了2024数维杯数学建模挑战赛生物质和煤共热解问题的研究完整的成品论文。 本论文可以保证原创&#xff0c;保证高质量。绝不是随便引用一大堆模型和代码复制粘贴进来完全没有应用糊弄人的垃圾半成品论文。 B题论…

跟我学C++中级篇——封装对象的实践

一、对象封装 在面向对象编程中&#xff0c;首要的事情就是如何进行对象的封装。说的直白一些&#xff0c;就是如何设计类或者是结构体。许多开发者看过不少的书&#xff0c;也学过很多的设计方法&#xff0c;更看过很多别人的代码。那么如何指导自己进行对象的封装呢&#xf…