均摊复杂度和防止复杂度的震荡
关于上一节中我们对添加操作的时间复杂度归结为O(n)是考虑了扩容操作(resize)在内的。就addLast(e)操作而言,时间复杂度为O(1),在考虑最坏情况下,每次添加均会触发扩容操作,需要移动n个元素,因此此时addLast操作的时间复杂度为O(n)。
(1)addLast(e)均摊时间复杂度分析
resize(n) O(n)
假设当前capacity=8,并且每一次添加操作都使用addLast方法
17次基本操作包括:9次添加操作,8次转移操作。均摊每次addLast操作进行大约两次基本操作:
平均值为:17/9≈ 2。
假设capacity=n,n+1次addLast操作,触发resize,总共进行了2n+1=(n+1)+ n次基本操作;
均摊每次addLast操作进行大约两次基本操作:
平均值为: 2n+1 / n+1 ≈ 2
结论:因此addLast均摊时间复杂度为O(1),均摊时间复杂度会比最坏情况有意义,因为一般情况下resize不会每一次都会触发,因此可以分摊到其他上面。 同理,removeLast操作均摊时间复杂度也是O(1)
(1)addLast(e)和removeLast(e)复杂度震荡分析
设数组的容量为n,此时数组中的个数为n个,此时我们向数组中添加一个元素,则会触发扩容操作;然后在从数组中删除一个元素时又会重新触发缩容操作,这样反复执行都会耗费O(n)的复杂度,导致复杂度震荡。
演示如下:
第一次执行addLast(e)时间复杂度:O(n)
第二次执行removeLast(e)时间复杂度:O(n)
第三次执行addLast(e)时间复杂度:O(n)
第四次执行removeLast(e)时间复杂度:O(n)
产生复杂度震荡的原因为:removeLast时resize过于着急(Eager)。
解决办法为:Lazy(remove延迟执行resize)
容量2n,size=n+1时:
容量2n,size=n时,进行缩容1/2:
容量2n,size=1/4*2n,进行缩容1/2 :
当size==capacity/4时,才将capacity减半。
现在我们来进一步改进我们的程序代码:
//从数组中删除index位置的元素,返回删除的元素
public E remove(int index) {
//1.判断索引的选择是否合法
if (index < 0 || index > size)
throw new IllegalArgumentException("您选择的位置不合法");
//2.先存储需要删除的索引对应的值
E ret = data[index];
//将索引为index之后(index)的元素依次向前移动
for (int i = index + 1; i < size; i++) {
//3.执行删除--实质为索引为index之后(index)的元素依次向前移动,将元素覆盖
data[i - 1] = data[i];
}
//4.维护size变量
size--;
// loitering objects != memory leak 手动释放内存空间
data[size] = null;
//缩容操作
if (size == data.length / 4 && data.length != 0) {
resize(data.length / 2);
}
//5.返回被删除的元素
return ret;
}
到此我们完成了一个比较完善的动态数组的封装。
- 拒绝全英文垃圾评论!仅用代码实现
- 在.NET Core 里使用 BouncyCastle 的DES加密算法
- 微信小程序:为了满足三方需求,我们一直在改变
- 让陌生人能够相互自由交易和支付
- 斯坦福发布首份 AI Index 报告,AI 研究者不再茫然
- jQuery仿极客公园火箭发射“返回顶部”效果(初始篇)
- Java程序员必读,Java设计模式应该遵循哪些原则
- Visual Studio 2010快速参考指南里头的Scrum海报
- TFS2010 Team Project Collections
- 基于Ext.Panel扩展一个更容易操作的Canvas
- 好玩的效果很好的Html游戏和范例
- jQuery仿极客公园火箭发射“返回顶部”效果(优化篇)
- 北京市首个无人驾驶试运营基地落户顺义
- 后rtx集成时代
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- python3用ARIMA模型进行时间序列预测
- R语言马尔可夫转换模型研究交通伤亡人数事故预测
- scrapy爬虫框架和selenium的使用:对优惠券推荐网站数据LDA文本挖掘
- 使用R语言进行Metroplis-in-Gibbs采样和MCMC运行分析
- R语言中的马尔科夫机制转换(Markov regime switching)模型
- R语言ARMA-EGARCH模型、集成预测算法对SPX实际波动率进行预测
- nginx快速入门
- R语言中进行期权定价的Heston模型
- 使用R语言随机波动模型SV处理时间序列中的随机波动率
- 20个ES6面试高频问题
- i++和++i傻傻分不清楚?这里给你最清楚的解答
- android APT 使用
- Flutter异步编程async与await的基本使用
- 教大家一个万能PPT图片排版技巧,太赞了!
- 重复读取 HttpServletRequest 中 InputStream 的方法