• 队列--环形数组实现


     具体代码可参考:https://github.com/1693905917/DataStructure.git

    好处

    1. 对比普通数组,起点和终点更为自由,不用考虑数据移动

    2. “环”意味着不会存在【越界】问题

    3. 数组性能更佳

    4. 环形数组比较适合实现有界队列、RingBuffer 等

     

    下标计算

    例如,数组长度是 5,当前位置是 3 ,向前走 2 步,此时下标为 (3 + 2)\%5 = 0

    • cur 当前指针位置

    • step 前进步数

    • length 数组长度

    注意:

    • 如果 step = 1,也就是一次走一步,可以在 >= length 时重置为 0 即可

    判断空

    判断该数组是否为空:头指针==尾指针

    判断满

    判断该数组是否满:(尾指针的索引+1)%数组长度==头指针的索引

    满之后的策略可以根据业务需求决定

    • 例如我们要实现的环形队列,满之后就拒绝入队

    代码:

    1. //仅用head,tail判断空满,head,tail即为索引值
    2. public class ArrayQueue implements Queue, Iterable{
    3. private int head = 0;
    4. private int tail = 0;
    5. private final E[] array;
    6. private final int length;
    7. //SuppressWarnings:抑制警告
    8. @SuppressWarnings("all")
    9. public ArrayQueue(int capacity) {
    10. //你设定的容量+1:在你添加满容量时,需要有多出一个的位置给尾指针
    11. length = capacity + 1;
    12. array = (E[]) new Object[length];
    13. }
    14. @Override
    15. public boolean offer(E value) {
    16. if (isFull()) {
    17. return false;
    18. }
    19. array[tail] = value;
    20. //当加到数组最大索引位置时,应该让tail=数组初始索引位置0
    21. tail = (tail + 1) % length;
    22. return true;
    23. }
    24. @Override
    25. public E poll() {
    26. if (isEmpty()) {
    27. return null;
    28. }
    29. E value = array[head];
    30. //当加到数组最大索引位置时,应该让tail=数组初始索引位置0
    31. head = (head + 1) % length;
    32. return value;
    33. }
    34. @Override
    35. public E peek() {
    36. if (isEmpty()) {
    37. return null;
    38. }
    39. return array[head];
    40. }
    41. @Override
    42. public boolean isEmpty() {
    43. return tail == head;
    44. }
    45. @Override
    46. public boolean isFull() {
    47. return (tail + 1) % length == head;
    48. }
    49. @Override
    50. public Iterator iterator() {
    51. return new Iterator() {
    52. int p = head;
    53. @Override
    54. public boolean hasNext() {
    55. return p != tail;
    56. }
    57. @Override
    58. public E next() {
    59. E value = array[p];
    60. p = (p + 1) % array.length;
    61. return value;
    62. }
    63. };
    64. }
    65. }

    判断空、满方法2

    //修改在数组满的时候,不用给尾指针留个位置

    引入 size

    1. public class ArrayQueue2 implements Queue, Iterable {
    2. private int head = 0;
    3. private int tail = 0;
    4. private final E[] array;
    5. private final int capacity;
    6. private int size = 0;
    7. @SuppressWarnings("all")
    8. public ArrayQueue2(int capacity) {
    9. this.capacity = capacity;
    10. array = (E[]) new Object[capacity];
    11. }
    12. @Override
    13. public boolean offer(E value) {
    14. if (isFull()) {
    15. return false;
    16. }
    17. array[tail] = value;
    18. tail = (tail + 1) % capacity;
    19. size++;
    20. return true;
    21. }
    22. @Override
    23. public E poll() {
    24. if (isEmpty()) {
    25. return null;
    26. }
    27. E value = array[head];
    28. head = (head + 1) % capacity;
    29. size--;
    30. return value;
    31. }
    32. @Override
    33. public E peek() {
    34. if (isEmpty()) {
    35. return null;
    36. }
    37. return array[head];
    38. }
    39. @Override
    40. public boolean isEmpty() {
    41. return size == 0;
    42. }
    43. @Override
    44. public boolean isFull() {
    45. return size == capacity;
    46. }
    47. @Override
    48. public Iterator iterator() {
    49. return new Iterator() {
    50. int p = head;
    51. @Override
    52. public boolean hasNext() {
    53. return p != tail;
    54. }
    55. @Override
    56. public E next() {
    57. E value = array[p];
    58. p = (p + 1) % capacity;
    59. return value;
    60. }
    61. };
    62. }
    63. }

    判断空、满方法3

    • head 和 tail 不断递增,用到索引时,再用它们进行计算,两个问题

      • 如何保证 head 和 tail 自增超过正整数最大值的正确性

      • 如何让取模运算性能更高

    • 答案:让 capacity 为 2 的幂

    1. public class ArrayQueue3 implements Queue, Iterable {
    2. private int head = 0;
    3. private int tail = 0;
    4. private final E[] array;
    5. private final int capacity;
    6. @SuppressWarnings("all")
    7. public ArrayQueue3(int capacity) {
    8. if ((capacity & capacity - 1) != 0) {
    9. throw new IllegalArgumentException("capacity 必须为 2 的幂");
    10. }
    11. this.capacity = capacity;
    12. array = (E[]) new Object[this.capacity];
    13. }
    14. @Override
    15. public boolean offer(E value) {
    16. if (isFull()) {
    17. return false;
    18. }
    19. array[tail & capacity - 1] = value;
    20. tail++;
    21. return true;
    22. }
    23. @Override
    24. public E poll() {
    25. if (isEmpty()) {
    26. return null;
    27. }
    28. E value = array[head & capacity - 1];
    29. head++;
    30. return value;
    31. }
    32. @Override
    33. public E peek() {
    34. if (isEmpty()) {
    35. return null;
    36. }
    37. return array[head & capacity - 1];
    38. }
    39. @Override
    40. public boolean isEmpty() {
    41. return tail - head == 0;
    42. }
    43. @Override
    44. public boolean isFull() {
    45. return tail - head == capacity;
    46. }
    47. @Override
    48. public Iterator iterator() {
    49. return new Iterator() {
    50. int p = head;
    51. @Override
    52. public boolean hasNext() {
    53. return p != tail;
    54. }
    55. @Override
    56. public E next() {
    57. E value = array[p & capacity - 1];
    58. p++;
    59. return value;
    60. }
    61. };
    62. }
    63. }

    第三个方法暴露了一个问题:因为我们的head、tail都是整型int类型,正整数的最大值int:2147483647

    测试:

    如果是使用C语言就会解决:unsigned int 0 ~2^32-1

    对于JAVA语言,它有种方法:可以将int整型超出的时候,及时将int转换为Long类型:

    Integer.toUnsignedLong(tail)

    优化以后的代码:

    1. /**
    2. * @BelongsProject: arithmetic
    3. * @BelongsPackage: com.hzp.algorithm.queue
    4. * @Author: ASUS
    5. * @CreateTime: 2023-09-25 11:26
    6. * @Description: TODO 环形数组实现3.0
    7. * @Version: 1.0
    8. */
    9. //修改在数组满的时候,不用给尾指针留个位置
    10. public class ArrayQueue3 implements Queue, Iterable{
    11. private int head = 0;
    12. private int tail = 0;
    13. private E[] array;
    14. //SuppressWarnings:抑制警告
    15. @SuppressWarnings("all")
    16. public ArrayQueue3(int capacity) {
    17. array = (E[]) new Object[ capacity ];//这个时候就不需要给尾指针留个位置
    18. }
    19. @Override
    20. public boolean offer(E value) {
    21. if (isFull()) {
    22. return false;
    23. }
    24. //进行(int):数组中只能存储Int类型,不能是long类型所以要转换
    25. array[(int) (Integer.toUnsignedLong(tail)% array.length)] = value;
    26. tail++;
    27. return true;
    28. }
    29. @Override
    30. public E poll() {
    31. if (isEmpty()) {
    32. return null;
    33. }
    34. E value = array[(int) (Integer.toUnsignedLong(head)% array.length)];
    35. head++;
    36. return value;
    37. }
    38. @Override
    39. public E peek() {
    40. if (isEmpty()) {
    41. return null;
    42. }
    43. return array[(int) (Integer.toUnsignedLong(head)% array.length)];
    44. }
    45. @Override
    46. public boolean isEmpty() {
    47. return head==tail;
    48. }
    49. @Override
    50. public boolean isFull() {
    51. return tail-head==array.length;
    52. }
    53. @Override
    54. public Iterator iterator() {
    55. return new Iterator() {
    56. int p = head;
    57. @Override
    58. public boolean hasNext() {
    59. return p!=tail;
    60. }
    61. @Override
    62. public E next() {
    63. E value = array[(int) (Integer.toUnsignedLong(p)%array.length)];
    64. p++;
    65. return value;
    66. }
    67. };
    68. }
    69. }

    判断空、满方法4

    我们以二进制的角度来看求模运算的规律:

    //求模运算:
    //          被除数是什么都无所谓
    //        如果除数是2的n次方
    //        那么被除数的后n位即为余数(馍)
    //        求被除数的后n位方法:与2^n-1按位与

    演示:

    总结规律:当除数是2的n次方,则余数是被除数的二进制后n位,被除数剩余的二进制就是商的二进制

    对于我们求模运算而言,我们只需要余数即可:

    所以结论:求余数:被除数与2^n-1按位与即可得到余数:

    1. //求模运算:
    2. // 如果除数是2的n次方
    3. // 那么被除数的后n位即为余数(馍)
    4. // 求被除数的后n位方法:与2^n-1按位与
    5. public class ArrayQueue3_1 implements Queue, Iterable{
    6. private int head = 0;
    7. private int tail = 0;
    8. private E[] array;
    9. //SuppressWarnings:抑制警告
    10. @SuppressWarnings("all")
    11. //这个方法的条件就是 capacity的取值必须是2的n次方
    12. public ArrayQueue3_1(int capacity) {
    13. array = (E[]) new Object[ capacity ];//这个时候就不需要给尾指针留个位置
    14. }
    15. @Override
    16. public boolean offer(E value) {
    17. if (isFull()) {
    18. return false;
    19. }
    20. //进行(int):数组中只能存储Int类型,不能是long类型所以要转换
    21. //array[(int) (Integer.toUnsignedLong(tail)% array.length)] = value;
    22. //以下方法比以上方法的优点:1.&的运算更加优化 2.这也防止了int类型超出最大值的情况
    23. array[tail& (array.length-1)]=value;
    24. tail++;
    25. return true;
    26. }
    27. @Override
    28. public E poll() {
    29. if (isEmpty()) {
    30. return null;
    31. }
    32. //E value = array[(int) (Integer.toUnsignedLong(head)% array.length)];
    33. E value = array[head& (array.length-1)];
    34. head++;
    35. return value;
    36. }
    37. @Override
    38. public E peek() {
    39. if (isEmpty()) {
    40. return null;
    41. }
    42. //return array[(int) (Integer.toUnsignedLong(head)% array.length)];
    43. return array[head& (array.length-1)];
    44. }
    45. @Override
    46. public boolean isEmpty() {
    47. return head==tail;
    48. }
    49. @Override
    50. public boolean isFull() {
    51. return tail-head==array.length;
    52. }
    53. @Override
    54. public Iterator iterator() {
    55. return new Iterator() {
    56. int p = head;
    57. @Override
    58. public boolean hasNext() {
    59. return p!=tail;
    60. }
    61. @Override
    62. public E next() {
    63. //E value = array[(int) (Integer.toUnsignedLong(p)%array.length)];
    64. E value = array[p& (array.length-1)];
    65. p++;
    66. return value;
    67. }
    68. };
    69. }
    70. }

    但是注意:这个方法的条件就是 capacity的取值必须是2的n次方!!!!

    因此,这个方法是有缺陷的。

    判断空、满方法5

    这个方法的条件就是 capacity的取值没有限制

    就是对于第四种方法的优化:

    第一种优化:就是对于“当输入的数不是2的幂则跑异常”:

    1. //1.抛异常:当输入的数不是2的幂则跑异常
    2. if(((capacity&capacity-1)!=0)){
    3. throw new IllegalArgumentException("capactiy 必须是2的幂");
    4. }

    对于“capacity&capacity-1”演示:

    第二种优化:将输入的数不是2的幂改成2^n

    这是利用第一种结论来写:

    1. /*
    2. 当我输入的是数是30
    3. c=30;
    4. 2^4 == 16
    5. 2^4.? == 30
    6. 2^5 == 32
    7. 我们要找到的是这个数最近并且大于这个数的2^n:
    8. 对于幂的获取:log2(30)==4.? --->(int)log2(30)==4 -->(int)log2(30)+1 ==5
    9. int c=30;
    10. int n= (int)(Math.log10(c-1)/Math.log10(2))+1;
    11. System.out.println(n);
    12. System.out.println(1<
    13. 验证1<
    14. 1 2^0
    15. 10 2^1
    16. 100 2^2
    17. 1000 2^3
    18. */

    利用第二种结论:求离c最近,比c大的2^n(方法2)

    1. c=30;
    2. c -= 1;
    3. c |= c >> 1;
    4. c |= c >> 2;
    5. c |= c >> 4;
    6. c |= c >> 8;
    7. c |= c >> 16;
    8. c += 1;

    代码:

    1. public class ArrayQueue3_2 implements Queue, Iterable{
    2. private int head = 0;
    3. private int tail = 0;
    4. private E[] array;
    5. //SuppressWarnings:抑制警告
    6. @SuppressWarnings("all")
    7. public ArrayQueue3_2(int capacity) {
    8. //1.抛异常:当输入的数不是2的幂则跑异常
    9. if(((capacity&capacity-1)!=0)){
    10. throw new IllegalArgumentException("capactiy 必须是2的幂");
    11. }
    12. //2.将输入的数不是2的幂改成2^n
    13. capacity -= 1;
    14. capacity |= capacity >> 1;
    15. capacity |= capacity >> 2;
    16. capacity |= capacity >> 4;
    17. capacity |= capacity >> 8;
    18. capacity |= capacity >> 16;
    19. capacity += 1;
    20. array = (E[]) new Object[ capacity ];//这个时候就不需要给尾指针留个位置
    21. }
    22. @Override
    23. public boolean offer(E value) {
    24. if (isFull()) {
    25. return false;
    26. }
    27. //进行(int):数组中只能存储Int类型,不能是long类型所以要转换
    28. //array[(int) (Integer.toUnsignedLong(tail)% array.length)] = value;
    29. //以下方法比以上方法的优点:1.&的运算更加优化 2.这也防止了int类型超出最大值的情况
    30. array[tail& (array.length-1)]=value;
    31. tail++;
    32. return true;
    33. }
    34. @Override
    35. public E poll() {
    36. if (isEmpty()) {
    37. return null;
    38. }
    39. //E value = array[(int) (Integer.toUnsignedLong(head)% array.length)];
    40. E value = array[head& (array.length-1)];
    41. head++;
    42. return value;
    43. }
    44. @Override
    45. public E peek() {
    46. if (isEmpty()) {
    47. return null;
    48. }
    49. //return array[(int) (Integer.toUnsignedLong(head)% array.length)];
    50. return array[head& (array.length-1)];
    51. }
    52. @Override
    53. public boolean isEmpty() {
    54. return head==tail;
    55. }
    56. @Override
    57. public boolean isFull() {
    58. return tail-head==array.length;
    59. }
    60. @Override
    61. public Iterator iterator() {
    62. return new Iterator() {
    63. int p = head;
    64. @Override
    65. public boolean hasNext() {
    66. return p!=tail;
    67. }
    68. @Override
    69. public E next() {
    70. //E value = array[(int) (Integer.toUnsignedLong(p)%array.length)];
    71. E value = array[p& (array.length-1)];
    72. p++;
    73. return value;
    74. }
    75. };
    76. }
    77. }

  • 相关阅读:
    Leetcode 1769. 移动所有球到每个盒子所需的最小操作数
    编程学:关于同类词的等长拼写问题
    金仓数据库 KingbaseGIS 使用手册(6.6. 几何对象校验函数、6.7. 空间参考系函数)
    aosp定制android系统
    百度地图自定义覆盖物(html)格式
    【HMS Core】定位地图服务常见问题,穿戴设备支持、比例尺支持、在非华为手机上逆地理编码的支持?
    眺望 2023,一线汽车品牌营销玩得越来越花了
    Servlet执行流程&&Servlet 生命周期
    pandas定位选取某列某指标最大值所在的行记录,比如月底
    Acrel-3000WEB电能管理系统在都巴高速的应用
  • 原文地址:https://blog.csdn.net/m0_60333804/article/details/133419586