当前位置:  首页>> 技术小册>> 数据结构与算法(中)

题目描述

统计一个数字在排序数组中出现的次数。

例如输入排序数组 [1, 2, 3, 3, 3, 3, 4, 5] 和数字 3,由于 3 在这个数组中出现了 4 次,因此输出 4。

样例

  1. 输入:[1, 2, 3, 3, 3, 3, 4, 5] , 3
  2. 输出:4

解法

找出第一个 k 和最后一个 k 出现的位置。

找第一个 k 时,利用二分法,如果 nums[m] == k,判断它的前一个位置是不是也是 k,如果不是,说明这是第一个 k,直接返回。如果是,那么递归在左边查找第一个 k。

找最后一个 k 也同理。

  1. /**
  2. * @author bingo
  3. * @since 2018/12/8
  4. */
  5. class Solution {
  6. /**
  7. * 求数字k在排序数组中出现的次数
  8. *
  9. * @param nums 数组
  10. * @param k 数字k
  11. * @return k在数组中出现的次数
  12. */
  13. public int getNumberOfK(int[] nums, int k) {
  14. if (nums == null || nums.length == 0) {
  15. return 0;
  16. }
  17. int start = 0, end = nums.length - 1;
  18. int first = getFirstK(nums, start, end, k);
  19. int last = getLastK(nums, start, end, k);
  20. if (first > -1 && last > -1) {
  21. return last - first + 1;
  22. }
  23. return 0;
  24. }
  25. private int getFirstK(int[] nums, int start, int end, int k) {
  26. if (start > end) {
  27. return -1;
  28. }
  29. int m = start + ((end - start) >> 1);
  30. if (nums[m] == k) {
  31. if (m == 0 || (m > 0 && nums[m - 1] != k)) {
  32. return m;
  33. } else {
  34. end = m - 1;
  35. }
  36. } else {
  37. if (nums[m] > k) {
  38. end = m - 1;
  39. } else {
  40. start = m + 1;
  41. }
  42. }
  43. return getFirstK(nums, start, end, k);
  44. }
  45. private int getLastK(int[] nums, int start, int end, int k) {
  46. if (start > end) {
  47. return -1;
  48. }
  49. int m = start + ((end - start) >> 1);
  50. if (nums[m] == k) {
  51. if (m == nums.length - 1 || (m < nums.length - 1 && nums[m + 1] != k)) {
  52. return m;
  53. } else {
  54. start = m + 1;
  55. }
  56. } else {
  57. if (nums[m] > k) {
  58. end = m - 1;
  59. } else {
  60. start = m + 1;
  61. }
  62. }
  63. return getLastK(nums, start, end, k);
  64. }
  65. }

测试用例

  1. 功能测试(数组中包含要查找的数字;数组中没有要查找的数字;要查找的数字在数组中出现一次/多次);
  2. 边界值测试(查找数组中的最大值、最小值;数组中只有一个数字);
  3. 特殊输入测试(表示数组的指针为空指针)。

该分类下的相关小册推荐: