03.二分、复杂度、动态数组、哈希表和有序表

本文深入探讨了二分查找在有序数组中的应用,包括寻找特定数值、大于等于或小于等于目标值的最左和最右位置。此外,介绍了时间复杂度的概念及其常见类型。接着,文章讲解了动态数组的扩容机制和时间复杂度分析。还讨论了哈希表和有序表在数据查找中的高效性。最后,简要阐述了Java中按值传递和按引用传递的区别。
摘要由CSDN通过智能技术生成

二分、复杂度、动态数组、哈希表和有序表

一、二分法

  1. 有序数组中找到num
  2. 有序数组中找到>=num最左的位置
  3. 有序数组中找到<=num最右的位置
  4. 局部最小值问题

例1 有序数组中找到num

import java.util.Arrays;

public class BSExist {

   // arr保证有序
   public static boolean find(int[] arr, int num) {
      if (arr == null || arr.length == 0) {
         return false;
      }
      int L = 0;
      int R = arr.length - 1;
      while (L <= R) {
         int mid = (L + R) / 2; // 可能会溢出
         if (arr[mid] == num) {
            return true;
         } else if (arr[mid] < num) {
            L = mid + 1;
         } else {
            R = mid - 1;
         }
      }
      return false;
   }

   // for test
   public static boolean test(int[] sortedArr, int num) {
      for (int cur : sortedArr) {
         if (cur == num) {
            return true;
         }
      }
      return false;
   }

   // for test
   public static int[] generateRandomArray(int maxSize, int maxValue) {
      int[] arr = new int[(int) ((maxSize + 1) * Math.random())];
      for (int i = 0; i < arr.length; i++) {
         arr[i] = (int) ((maxValue + 1) * Math.random()) - (int) (maxValue * Math.random());
      }
      return arr;
   }

   public static void main(String[] args) {
      int testTime = 500000;
      int maxSize = 10;
      int maxValue = 100;
      boolean succeed = true;
      for (int i = 0; i < testTime; i++) {
         int[] arr = generateRandomArray(maxSize, maxValue);
         Arrays.sort(arr);
         int value = (int) ((maxValue + 1) * Math.random()) - (int) (maxValue * Math.random());
         if (test(arr, value) != find(arr, value)) {
            System.out.println("出错了!");
            succeed = false;
            break;
         }
      }
      System.out.println(succeed ? "Nice!" : "Fucking fucked!");
   }

}

例2 有序数组中找到>=num最左的位置

import java.util.Arrays;

public class BSNearLeft {

   // arr有序的,>=num 最左
   public static int mostLeftNoLessNumIndex(int[] arr, int num) {
      if (arr == null || arr.length == 0) {
         return -1;
      }
      int L = 0;
      int R = arr.length - 1;
      int ans = -1;
      while (L <= R) {
         int mid = (L + R) / 2; // 可能会溢出
         if (arr[mid] >= num) {
            ans = mid;
            R = mid - 1;
         } else {
            L = mid + 1;
         }
      }
      return ans;
   }

   // for test
   public static int test(int[] arr, int value) {
      for (int i = 0; i < arr.length; i++) {
         if (arr[i] >= value) {
            return i;
         }
      }
      return -1;
   }

   // for test
   public static int[] generateRandomArray(int maxSize, int maxValue) {
      int[] arr = new int[(int) ((maxSize + 1) * Math.random())];
      for (int i = 0; i < arr.length; i++) {
         arr[i] = (int) ((maxValue + 1) * Math.random()) - (int) (maxValue * Math.random());
      }
      return arr;
   }

   // for test
   public static void printArray(int[] arr) {
      if (arr == null) {
         return;
      }
      for (int i = 0; i < arr.length; i++) {
         System.out.print(arr[i] + " ");
      }
      System.out.println();
   }

   public static void main(String[] args) {
      int testTime = 500000;
      int maxSize = 10;
      int maxValue = 100;
      boolean succeed = true;
      for (int i = 0; i < testTime; i++) {
         int[] arr = generateRandomArray(maxSize, maxValue);
         Arrays.sort(arr);
         int value = (int) ((maxValue + 1) * Math.random()) - (int) (maxValue * Math.random());
         if (test(arr, value) != mostLeftNoLessNumIndex(arr, value)) {
            printArray(arr);
            System.out.println(value);
            System.out.println(test(arr, value));
            System.out.println(mostLeftNoLessNumIndex(arr, value));
            succeed = false;
            break;
         }
      }
      System.out.println(succeed ? "Nice!" : "Fucking fucked!");
   }

}

例3 有序数组中找到<=num最右的位置

import java.util.Arrays;

public class BSNearRight {

   // 在arr上,找满足<=value的最右位置
   public static int nearestIndex(int[] arr, int value) {
      int L = 0;
      int R = arr.length - 1;
      int index = -1; // 记录最右的对号
      while (L <= R) {
         int mid = L + ((R - L) >> 1);
         if (arr[mid] <= value) {
            index = mid;
            L = mid + 1;
         } else {
            R = mid - 1;
         }
      }
      return index;
   }

   // for test
   public static int test(int[] arr, int value) {
      for (int i = arr.length - 1; i >= 0; i--) {
         if (arr[i] <= value) {
            return i;
         }
      }
      return -1;
   }

   // for test
   public static int[] generateRandomArray(int maxSize, int maxValue) {
      int[] arr = new int[(int) ((maxSize + 1) * Math.random())];
      for (int i = 0; i < arr.length; i++) {
         arr[i] = (int) ((maxValue + 1) * Math.random()) - (int) (maxValue * Math.random());
      }
      return arr;
   }

   // for test
   public static void printArray(int[] arr) {
      if (arr == null) {
         return;
      }
      for (int i = 0; i < arr.length; i++) {
         System.out.print(arr[i] + " ");
      }
      System.out.println();
   }

   public static void main(String[] args) {
      int testTime = 500000;
      int maxSize = 10;
      int maxValue = 100;
      boolean succeed = true;
      for (int i = 0; i < testTime; i++) {
         int[] arr = generateRandomArray(maxSize, maxValue);
         Arrays.sort(arr);
         int value = (int) ((maxValue + 1) * Math.random()) - (int) (maxValue * Math.random());
         if (test(arr, value) != nearestIndex(arr, value)) {
            printArray(arr);
            System.out.println(value);
            System.out.println(test(arr, value));
            System.out.println(nearestIndex(arr, value));
            succeed = false;
            break;
         }
      }
      System.out.println(succeed ? "Nice!" : "Fucking fucked!");
   }

}

例4 局部最小值问题

无序数组,返回局部最小索引

  1. arr[0]<arr[1] : 0 为局部最小
  2. arr[N-1]<arr[N-2] : N-1 为局部最小
  3. arr[i-1]>arr[i]<arr[i+1] : i 为局部最小
public class BSAwesome {

   // arr 整体无序
   // arr 相邻的数不相等!
   public static int oneMinIndex(int[] arr) {
      if (arr == null || arr.length == 0) {
         return -1;
      }
      int N = arr.length;
      if (N == 1) {
         return 0;
      }
      if (arr[0] < arr[1]) {
         return 0;
      }
      if (arr[N - 1] < arr[N - 2]) {
         return N - 1;
      }
      int L = 0;
      int R = N - 1;
      // L...R 肯定有局部最小
      while (L < R - 1) {
         int mid = (L + R) / 2;
         if (arr[mid] < arr[mid - 1] && arr[mid] < arr[mid + 1]) {
            return mid;
         } else {
            if (arr[mid] > arr[mid - 1]) {
               R = mid - 1;
            } else {
               L = mid + 1;
            }
         }
      }
      return arr[L] < arr[R] ? L : R;
   }

   // 生成随机数组,且相邻数不相等
   public static int[] randomArray(int maxLen, int maxValue) {
      int len = (int) (Math.random() * maxLen);
      int[] arr = new int[len];
      if (len > 0) {
         arr[0] = (int) (Math.random() * maxValue);
         for (int i = 1; i < len; i++) {
            do {
               arr[i] = (int) (Math.random() * maxValue);
            } while (arr[i] == arr[i - 1]);
         }
      }
      return arr;
   }

   // 也用于测试
   public static boolean check(int[] arr, int minIndex) {
      if (arr.length == 0) {
         return minIndex == -1;
      }
      int left = minIndex - 1;
      int right = minIndex + 1;
      boolean leftBigger = left >= 0 ? arr[left] > arr[minIndex] : true;
      boolean rightBigger = right < arr.length ? arr[right] > arr[minIndex] : true;
      return leftBigger && rightBigger;
   }

   public static void printArray(int[] arr) {
      for (int num : arr) {
         System.out.print(num + " ");
      }
      System.out.println();
   }

   public static void main(String[] args) {
      int maxLen = 100;
      int maxValue = 200;
      int testTime = 1000000;
      System.out.println("测试开始");
      for (int i = 0; i < testTime; i++) {
         int[] arr = randomArray(maxLen, maxValue);
         int ans = oneMinIndex(arr);
         if (!check(arr, ans)) {
            printArray(arr);
            System.out.println(ans);
            break;
         }
      }
      System.out.println("测试结束");

   }

}

二、时间复杂度

1.时间复杂度的定义

时间复杂度(Time complexity)是一个函数,它定性描述该算法的运行频率。这是一个代表算法输入值的字符串的长度的函数. 时间复杂度常用大O表述,不包括这个函数的低阶项和首项系数。

2.常见时间复杂度列表

O ( 1 ) , O ( l o g N ) , O ( N ) , O ( N ∗ l o g N ) , O ( N 2 ) , O ( N 3 ) , O ( k N ) , O ( n ! ) O(1),O(logN),O(N),O(N*logN),O(N^2),O(N^3),O(k^N),O(n!) O(1),O(logN),O(N),O(NlogN),O(N2),O(N3)O(kN),O(n!)

三、动态数组

1.动态数组是什么?

ArrayList 动态扩容

2.动态数组的使用和扩容

3.动态数组的扩容与时间复杂度分析

数组查找 index 位置的时间复杂度是 O ( n ) O(n) O(n) ,我们思考扩容对时间复杂度的影响。计算知道扩容代价是 O ( n ) O(n) O(n) ,均摊到每一次操作就是 O ( 1 ) O(1) O(1) ,证明 ArrayList 和普通数组的时间复杂度一样都是常数级别的,但是比普通数组略慢一点。

四、哈希表和有序表

1.哈希表的使用

哈希表无论多少数据,增删改查的时间复杂度都是 O ( 1 ) O(1) O(1) ,但是这个 O ( 1 ) O(1) O(1) 的系数有些大。

哈希表的按值查询

2.有序表的使用

有序表无论多少数据,增删改查的时间复杂度都是 O ( l o g N ) O(logN) O(logN)

key 要可以比较,不然不能用,要自定义比较器

import java.util.HashMap;
import java.util.TreeMap;

public class TreeMap {

   // (K V)表
   public static void main(String[] args) {

      TreeMap<Integer, String> treeMap1 = new TreeMap<>();

      treeMap1.put(3, "我是3");
      treeMap1.put(0, "我是3");
      treeMap1.put(7, "我是3");
      treeMap1.put(2, "我是3");
      treeMap1.put(5, "我是3");
      treeMap1.put(9, "我是3");

      System.out.println(treeMap1.containsKey(7)); // true
      System.out.println(treeMap1.containsKey(6)); // false
      System.out.println(treeMap1.get(3)); // 我是3

      treeMap1.put(3, "他是3");
      System.out.println(treeMap1.get(3)); // 他是3

      treeMap1.remove(3);
      System.out.println(treeMap1.get(3)); // false

      System.out.println(treeMap1.firstKey()); // 0
      System.out.println(treeMap1.lastKey()); // 9
      // <=5 离5最近的key告诉我
      System.out.println(treeMap1.floorKey(5));
      // <=6 离6最近的key告诉我
      System.out.println(treeMap1.floorKey(6));
      // >=5 离5最近的key告诉我
      System.out.println(treeMap1.ceilingKey(5));
      // >=6 离6最近的key告诉我
      System.out.println(treeMap1.ceilingKey(6));

//    Node node3 = new Node(3);
//    Node node4 = new Node(4);
//    TreeMap<Node, String> treeMap2 = new TreeMap<>();
//    treeMap2.put(node3, "我是node3");
//    treeMap2.put(node4, "我是node4");

   }

}

五、Java中按值传递和按引用传递

1.按值传递

哈希表的按值传递(原生数据结构 Integer Double Float String Character)

内存消耗为真实长度

public class HashMap {
    // (K V)表
    public static void main(String[] args) {
        HashMap<String, String> map = new HashMap<>();
        
		map.put(1234567, "我是1234567");
        Integer a = 1234567;
        Integer b = 1234567;

        System.out.println(a == b); // false
        System.out.println(map.containsKey(a)); // true
        System.out.println(map.containsKey(b)); // true
    }
}

2.按引用传递

按引用传递(非原生数据结构 自定义 Node

内存消耗为地址内存和

import java.util.HashMap;
import java.util.TreeMap;

public class HashMap {

   public static class Node {
      public int value;

      public Node(int v) {
         value = v;
      }
   }

   // (K V)表
   public static void main(String[] args) {
      Node node1 = new Node(1);
      Node node2 = new Node(1);
      HashMap<Node, String> map3 = new HashMap<>();
      map3.put(node1, "我进来了!");
      System.out.println(map3.containsKey(node1)); // true
      System.out.println(map3.containsKey(node2)); // false
   }

}

六、写在后面

欢迎关注,会经常记录一些算法学习中遇到的问题。

欢迎随时留言讨论,与君共勉,知无不答!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值