当前位置:  首页>> 技术小册>> Java语言基础9-常用API和常见算法

5.1.1 数组的反转

  • 所谓的数组的反转:就是一开始是 [1,2,3,4,5] ,经过数组的反转之后变成了 [5,4,3,2,1] 。
  • 其实,数组的反转就是对称位置的元素交换。
    注意:数组的反转不是数组的倒叙遍历。
  • 方法一:
    • 借助一个新的数组。
    • 首尾对应位置交换。
    • 将新数组的地址赋值给旧数组。

示例:

  1. package com.github.array.demo1;
  2. import java.util.Arrays;
  3. /**
  4. * @author maxiaoke.com
  5. * @version 1.0
  6. *
  7. */
  8. public class Test {
  9. public static void main(String[] args) {
  10. // 原来的数组
  11. int[] arr = {1, 2, 3, 4, 5, 6, 7, 8};
  12. // 新的数组
  13. int[] newArr = new int[arr.length];
  14. // 给新的数组中插入值
  15. for (int i = 0; i < newArr.length; i++) {
  16. newArr[i] = arr[arr.length - 1 - i];
  17. }
  18. // 将原来的数组指向新的数组
  19. arr = newArr;
  20. // 打印输出
  21. System.out.println(Arrays.toString(arr));
  22. }
  23. }

方法二:数组对称位置的元素互换。

示例:

  1. package com.github.array.demo2;
  2. import java.util.Arrays;
  3. /**
  4. * @author maxiaoke.com
  5. * @version 1.0
  6. *
  7. */
  8. public class Test {
  9. public static void main(String[] args) {
  10. // 原来的数组
  11. int[] arr = {1, 2, 3, 4, 5, 6, 7, 8};
  12. // 数组反转
  13. for (int i = 0, j = arr.length - 1; i < j; i++, j--) {
  14. int temp = arr[i];
  15. arr[i] = arr[j];
  16. arr[j] = temp;
  17. }
  18. // 打印输出
  19. System.out.println(Arrays.toString(arr));
  20. }
  21. }

5.1.2 数组的二分查找法

  • 数组的查找:判断一个元素是否存在于数组中。
  • 数组的基本查找法:遍历数组,判断即可。
  • 数组的二分查找法(折半查找法):要求数组元素必须支持比较大小,并且数组中的元素已经按大小排好序,然后对半查找。

  • 二分查找法的步骤:

    • ① 定义两个遍历,表示要查找的范围。默认min = 0,max = 最大索引。
    • ② 循环查找,但是min <= max。
    • ③ 计算出 mid 的值。
    • ④ 判断 mid 位置的元素是否为要查找的元素,如果是,直接返回的索引。
    • ⑤ 如果要查找的值在 mid 的左半边,那么 min 值不变,max = mid -1,继续下次循环查找。
    • ⑥ 如果要查找的值在 mid 的右半边,那么 max 值不变,min = mid +1,继续下次循环查找。
    • ⑦ 当 min > max 的时候,表示要查找的元素在数组中不存在,返回 -1 。

示例:

  1. package com.github.array.demo3;
  2. /**
  3. * 基本查找法
  4. *
  5. * @author maxiaoke.com
  6. * @version 1.0
  7. *
  8. */
  9. public class Test {
  10. public static void main(String[] args) {
  11. int[] arr = {1, 2, 3, 4, 5};
  12. int result = basicSearch(arr, 6);
  13. System.out.println("result = " + result);
  14. }
  15. /**
  16. *
  17. * 数组的基本查找法
  18. *
  19. * @param arr 数组
  20. * @param key 元素
  21. * @return 如果元素在数组中,则返回元素在数组中的索引位置;否则,返回-1
  22. */
  23. public static int basicSearch(int[] arr, int key) {
  24. for (int i = 0; i < arr.length; i++) {
  25. if (arr[i] == key) {
  26. return i;
  27. }
  28. }
  29. return -1;
  30. }
  31. }

示例:

  1. package com.github.array.demo4;
  2. /**
  3. * @author maxiaoke.com
  4. * @version 1.0
  5. *
  6. */
  7. public class Test {
  8. public static void main(String[] args) {
  9. int[] arr = {1, 5, 8, 12, 15};
  10. int result = binarySearch(arr, 9);
  11. System.out.println(result);
  12. }
  13. /**
  14. *
  15. * 数组的二分查找法
  16. *
  17. * @param arr 数组
  18. * @param key 元素
  19. * @return 如果元素在数组中,则返回元素在数组中的索引位置;否则,返回-1
  20. */
  21. public static int binarySearch(int[] arr, int key) {
  22. // 起始元素索引
  23. int min = 0;
  24. // 最后元素索引
  25. int max = arr.length - 1;
  26. while (min <= max) {
  27. // 计算中间元素的索引
  28. int mid = (min + max) / 2;
  29. // 如果中间元素就是需要查找的元素,直接返回索引
  30. if (key == arr[mid]) {
  31. return mid;
  32. }
  33. // 如果需要查找的元素比中间的元素大,那么min就是此时的mid+1,然后再计算mid
  34. if (key > arr[mid]) {
  35. min = mid + 1;
  36. }
  37. // 如果需要查找的元素比中间的元素小,那么max就是此时的mid-1,然后再计算mid
  38. if (key < arr[mid]) {
  39. max = mid - 1;
  40. }
  41. }
  42. return -1;
  43. }
  44. }

5.1.3 冒泡排序

  • 排序:将一组数据按照固定的规则进行排列。
  • 冒泡排序:相邻的数据两两比较,小的放在前面,大的放在后面。

注意:

  • 如果有 n 个数据进行排序,总共需要比较 n-1 次。
  • 每一次比较完毕,下一次的比较就会少一个数据参与。

示例:

  1. package com.github.array.demo5;
  2. import java.util.Arrays;
  3. /**
  4. * @author maxiaoke.com
  5. * @version 1.0
  6. *
  7. */
  8. public class Test {
  9. public static void main(String[] args) {
  10. int[] arr = {3, 5, 2, 1, 4};
  11. System.out.println("排序之前:" + Arrays.toString(arr));
  12. sort(arr);
  13. System.out.println("排序之后:" + Arrays.toString(arr));
  14. }
  15. /**
  16. * 冒泡排序
  17. *
  18. * @param arr 需要排序的数据
  19. */
  20. public static void sort(int[] arr) {
  21. // 外层循环控制的是次数,比数组的长度少1
  22. for (int i = 0; i < arr.length - 1; i++) {
  23. // 内层循环就是实际循环比较的
  24. // -1 为了让数组不越界
  25. // -i 每一轮结束之后,就会少比一个数字
  26. for (int j = 0; j < arr.length - 1 - i; j++) {
  27. if (arr[j] > arr[j + 1]) {
  28. int temp = arr[j];
  29. arr[j] = arr[j + 1];
  30. arr[j + 1] = temp;
  31. }
  32. }
  33. }
  34. }
  35. }

5.1.4 递归
递归:以编程的角度来看,递归指的是方法定义中调用方法本身的现象。
递归解决问题的思路:
① 将一个复杂的问题层层转换为一个 和原问题类似的规模较小 的问题来求解。
② 递归策略只需要 少量的程序 就可以描述出解题过程所需要的多次的重复计算。
递归解决问题要遵循以下的规则:
① 递归出口:否则会内存溢出。
② 递归规则:和原问题相似的规模较小的问题。
示例:

  1. package com.github.array.demo7;
  2. /**
  3. * @author maxiaoke.com
  4. * @version 1.0
  5. *
  6. */
  7. public class Test {
  8. public static void main(String[] args) {
  9. int sum = sum(100);
  10. System.out.println("sum = " + sum);
  11. }
  12. /**
  13. * 使用递归求和
  14. *
  15. * @param n 整数
  16. * @return
  17. */
  18. public static int sum(int n) {
  19. if (n == 1) {
  20. return 1;
  21. }
  22. return n + sum(n - 1);
  23. }
  24. }

示例:

  1. package com.github.array.demo8;
  2. /**
  3. * 需求:用递归求5的阶乘,并将结果输出在控制台上。
  4. * 分析:
  5. * 阶乘:一个正整数的阶乘是所有小于以及等于该数的正整数的乘积,自然数n的阶乘写作n!
  6. * 递归出口:1!=1
  7. * 递归规则:n!=n*(n-1)!
  8. *
  9. * @author maxiaoke.com
  10. * @version 1.0
  11. *
  12. */
  13. public class Test {
  14. public static void main(String[] args) {
  15. int factorial = factorial(5);
  16. System.out.println("factorial = " + factorial);
  17. }
  18. public static int factorial(int n) {
  19. if (n == 1) {
  20. return 1;
  21. }
  22. return n * factorial(n - 1);
  23. }
  24. }

5.1.5 数组的扩容
当原来的数组长度不够的时候需要扩容,那么新建一个数组,并指定长度,其长度为原来的 1.5 倍或 2 倍等,然后将元素复制到新数组中,并将新添加的元素放到新数组的后面。

注意:
新数组的长度定义多少合适,看实际情况,如果新增的元素个数确定,那么可以直接指定长度;如果新增的元素个数不确定,那么可以扩容为原来的 1.5 倍或 2 倍等。
数组扩容太多会造成资源浪费,太少会导致频繁扩容,效率低下。

示例:

  1. package com.github.array.demo9;
  2. import java.util.Arrays;
  3. /**
  4. * 数组扩容
  5. *
  6. * @author maxiaoke.com
  7. * @version 1.0
  8. *
  9. */
  10. public class Test {
  11. public static void main(String[] args) {
  12. // 原数组
  13. int[] arr = {1, 2, 3, 4, 5, 6, 7, 8};
  14. // 扩容的数组
  15. int[] newArr = new int[arr.length * 2];
  16. // 将数组的数据拷贝到新数组中
  17. System.arraycopy(arr, 0, newArr, 0, arr.length);
  18. // 如果还需要使用arr,可以让arr指向新数组
  19. arr = newArr;
  20. // 添加新数据到arr的最后
  21. arr[8] = 10;
  22. // 遍历显示
  23. System.out.println(Arrays.toString(arr));
  24. }
  25. }

5.1.6 数组元素的插入

  • 在原数组的某个 index 插入一个元素。
  • ① 如果原数组未满,那么直接将 index 后面的元素(包括 index 上的元素)移动到后面,然后将元素插入到 index 上。
  • ② 如果原数组已经满了,那么先扩容,然后重复 ① 。

  • 示例:原数组未满

  1. package com.github.array.demo10;
  2. import java.util.Arrays;
  3. /**
  4. * @author maxiaoke.com
  5. * @version 1.0
  6. *
  7. */
  8. public class Test {
  9. public static void main(String[] args) {
  10. // 原数组
  11. String[] str = new String[5];
  12. str[0] = "张三";
  13. str[1] = "李四";
  14. str[2] = "王五";
  15. // 现在想在index1的位置上插入一个元素赵六
  16. // index1的位置以及后面的元素依次移动1
  17. System.arraycopy(str, 1, str, 2, 2);
  18. // index1的位置插入元素
  19. str[1] = "赵六";
  20. // 打印输出
  21. System.out.println(Arrays.toString(str));
  22. }
  23. }

示例:原数组已满

  1. package com.github.array.demo11;
  2. import java.util.Arrays;
  3. /**
  4. * @author maxiaoke.com
  5. * @version 1.0
  6. *
  7. */
  8. public class Test {
  9. public static void main(String[] args) {
  10. // 原数组
  11. String[] str = new String[3];
  12. str[0] = "张三";
  13. str[1] = "李四";
  14. str[2] = "王五";
  15. // 现在想在index1的位置上插入一个元素赵六
  16. String[] dest = new String[4];
  17. // 扩容
  18. System.arraycopy(str, 0, dest, 0, str.length);
  19. // str指向新数组
  20. str = dest;
  21. // index1的位置以及后面的元素依次移动1
  22. System.arraycopy(str, 1, str, 2, 2);
  23. // index1的位置插入元素
  24. str[1] = "赵六";
  25. // 打印输出
  26. System.out.println(Arrays.toString(str));
  27. }
  28. }

5.1.7 数组元素的删除

  • 希望删除某个 index 上的元素,但是不希望数组中间空的元素,那么就将 index 后面的元素依次向前移动一位,最后一个元素置空即可。

  • 示例:

  1. package com.github.array.demo12;
  2. import java.util.Arrays;
  3. /**
  4. * @author maxiaoke.com
  5. * @version 1.0
  6. *
  7. */
  8. public class Test {
  9. public static void main(String[] args) {
  10. String[] arr = {"张三", "李四", "王五"};
  11. // 希望删除index1的元素,但是不希望index1上为空
  12. System.arraycopy(arr, 2, arr, 1, 1);
  13. // 最后一个元素置空
  14. arr[2] = null;
  15. // 打印输出
  16. System.out.println(Arrays.toString(arr));
  17. }
  18. }