Android Studio说:使用HashMap不如使用SparseArray?,androidsdk开发封装

int lo = 0;

int hi = size - 1;

while (lo <= hi) {

final int mid = (lo + hi) >>> 1;

final int midVal = array[mid];

if (midVal < value) {

lo = mid + 1;

} else if (midVal > value) {

hi = mid - 1;

} else {

return mid; // value found

}

}

return ~lo; // value not present

}

可以看到,即使在 mKeys 中不存在目标 key,但其返回值也指向了应该让 key 存入的位置。通过将计算出的索引值进行 ~ 运算,则返回值一定是 0 或者负数,从而与“找得到目标key的情况(返回值大于0)”的情况区分开。

从这个可以看出该方法的巧妙之处,单纯的一个返回值就可以区分出多种情况,且通过这种方式来存放数据可以使得 mKeys 的内部值一直是按照值递增的方式来排序的。

public void put(int key, E value) {

//用二分查找法查找指定 key 在 mKeys 中的索引值

int i = ContainerHelpers.binarySearch(mKeys, mSize, key);

//找得到则直接赋值

if (i >= 0) {

mValues[i] = value;

} else {

i = ~i;

//如果目标位置还未赋值,则直接存入数据即可,对应的情况是 2.1

if (i < mSize && mValues[i] == DELETED) {

mKeys[i] = key;

mValues[i] = value;

return;

}

//以下操作对应 2.1、2.2 两种情况:

if (mGarbage && mSize >= mKeys.length) {

gc();

//GC 后再次进行查找,因为值可能已经发生变化了

i = ~ContainerHelpers.binarySearch(mKeys, mSize, key);

}

//通过复制或者扩容数组,将数据存放到数组中

mKeys = GrowingArrayUtils.insert(mKeys, mSize, i, key);

mValues = GrowingArrayUtils.insert(mValues, mSize, i, value);

mSize++;

}

}

移除元素

上文说了,布尔变量mGarbage用于标记当前是否有待垃圾回收(GC)的元素,当该值被置为true时,即意味着当前状态需要进行垃圾回收,但回收操作并不马上进行,而是在后续操作中再完成。以下几个方法在移除元素时,都是只切断了mValues中的引用,而mKeys并没有进行回收,这个操作会留到gc()进行处理。

//如果存在 key 对应的元素值,则将其移除

public void delete(int key) {

//用二分查找法查找指定 key 在 mKeys 中的索引值

int i = ContainerHelpers.binarySearch(mKeys, mSize, key);

if (i >= 0) {

if (mValues[i] != DELETED) {

mValues[i] = DELETED;

//标记当前需要进行垃圾回收

mGarbage = true;

}

}

}

//和 delete 方法基本相同,差别在于会返回 key 对应的元素值

public E removeReturnOld(int key) {

int i = ContainerHelpers.binarySearch(mKeys, mSize, key);

if (i >= 0) {

if (mValues[i] != DELETED) {

final E old = (E) mValues[i];

mValues[i] = DELETED;

mGarbage = true;

return old;

}

}

return null;

}

//省略其它几个比较简单的移除元素的方法

查找元素

查找元素的方法较多,但逻辑都是挺简单的。

//根据 key 查找相应的元素值,查找不到则返回默认值

@SuppressWarnings(“unchecked”)

public E get(int key, E valueIfKeyNotFound) {

//用二分查找法查找指定 key 在 mKeys 中的索引值

int i = ContainerHelpers.binarySearch(mKeys, mSize, key);

//如果找不到该 key 或者该 key 尚未赋值,则返回默认值

if (i < 0 || mValues[i] == DELETED) {

return valueIfKeyNotFound;

} else {

return (E) mValues[i];

}

}

//根据 value 查找对应的索引值

public int indexOfValue(E value) {

if (mGarbage) {

gc();

}

for (int i = 0; i < mSize; i++) {

if (mValues[i] == value) {

return i;

}

}

return -1;

}

//与 indexOfValue 方法类似,但 indexOfValue 方法是通过比较 == 来判断是否同个对象

//而此方法是通过 equals 方法来判断是否同个对象

public int indexOfValueByValue(E value) {

if (mGarbage) {

gc();

}

for (int i = 0; i < mSize; i++) {

if (value == null) {

if (mValues[i] == null) {

return i;

}

} else {

if (value.equals(mValues[i])) {

return i;

}

}

}

最后

都说三年是程序员的一个坎,能否晋升或者提高自己的核心竞争力,这几年就十分关键。

技术发展的这么快,从哪些方面开始学习,才能达到高级工程师水平,最后进阶到Android架构师/技术专家?我总结了这 5大块;

我搜集整理过这几年阿里,以及腾讯,字节跳动,华为,小米等公司的面试题,把面试的要求和技术点梳理成一份大而全的“ Android架构师”面试 PDF(实际上比预期多花了不少精力),包含知识脉络 + 分支细节。

Java语言与原理;
大厂,小厂。Android面试先看你熟不熟悉Java语言

高级UI与自定义view;
自定义view,Android开发的基本功。

性能调优;
数据结构算法,设计模式。都是这里面的关键基础和重点需要熟练的。

NDK开发;
未来的方向,高薪必会。

前沿技术;
组件化,热升级,热修复,框架设计

网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。

我在搭建这些技术框架的时候,还整理了系统的高级进阶教程,会比自己碎片化学习效果强太多,GitHub可见;《Android架构视频+学习笔记》

当然,想要深入学习并掌握这些能力,并不简单。关于如何学习,做程序员这一行什么工作强度大家都懂,但是不管工作多忙,每周也要雷打不动的抽出 2 小时用来学习。

不出半年,你就能看出变化!

oid架构视频+学习笔记》](https://github.com/a120464/Android-P7/blob/master/Android%E5%BC%80%E5%8F%91%E4%B8%8D%E4%BC%9A%E8%BF%99%E4%BA%9B%EF%BC%9F%E5%A6%82%E4%BD%95%E9%9D%A2%E8%AF%95%E6%8B%BF%E9%AB%98%E8%96%AA%EF%BC%81.md)**

当然,想要深入学习并掌握这些能力,并不简单。关于如何学习,做程序员这一行什么工作强度大家都懂,但是不管工作多忙,每周也要雷打不动的抽出 2 小时用来学习。

不出半年,你就能看出变化!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值