基数排序代码实现

  1. 要求:将数组 {53, 3, 542, 748, 14, 214 } 使用基数排序, 进行升序排序
  2. 思路分析:前面的图文已经讲明确
  3. 代码实现:看老师演示

    基数排序的说明:

  4. 基数排序是对传统桶排序的扩展,速度很快.

  5. 基数排序是经典的空间换时间的方式,占用内存很大, 当对海量数据排序时,容易造成 OutOfMemoryError 。
  6. 基数排序时稳定的。[注:假定在待排序的记录序列中,存在多个具有相同的关键字的记录,若经过排序,这些记录的相对次序保持不变,即在原序列中,r[i]=r[j],且r[i]在r[j]之前,而在排序后的序列中,r[i]仍在r[j]之前,则称这种排序算法是稳定的;否则称为不稳定的]
  7. 有负数的数组,我们不用基数排序来进行排序, 如果要支持负数,参考: https://code.i-harness.com/zh-CN/q/e98fa9 ``` package com.atguigu.sort;

/**

  • ClassName:
  • Description:
  • Date: 2021-02-23 10:49
  • @project data_algorithm
  • @package com.atguigu.sort */

import java.text.SimpleDateFormat; import java.util.Arrays; import java.util.Date;

public class RadixSort {

  1. public static void main(String[] args) {
  2. int arr[] = { 53, 3, 542, 748, 14, 214};
  3. // 80000000 * 11 * 4 / 1024 / 1024 / 1024 =3.3G

// int[] arr = new int[8000000]; // for (int i = 0; i < 8000000; i++) { // arr[i] = (int) (Math.random() * 8000000); // 生成一个[0, 8000000) 数 // } System.out.println(“排序前”); Date data1 = new Date(); SimpleDateFormat simpleDateFormat = new SimpleDateFormat(“yyyy-MM-dd HH:mm:ss.SS”); String date1Str = simpleDateFormat.format(data1); System.out.println(“排序前的时间是=” + date1Str);

  1. radixSort(arr);
  2. Date data2 = new Date();
  3. String date2Str = simpleDateFormat.format(data2);
  4. System.out.println("排序前的时间是=" + date2Str);
  5. System.out.println("基数排序后 " + Arrays.toString(arr));
  6. }
  7. //基数排序方法
  8. public static void radixSort(int[] arr) {
  9. //根据前面的推导过程,我们可以得到最终的基数排序代码
  10. //1. 得到数组中最大的数的位数
  11. int max = arr[0]; //假设第一数就是最大数
  12. for(int i = 1; i < arr.length; i++) {
  13. if (arr[i] > max) {
  14. max = arr[i];
  15. }
  16. }
  17. //得到最大数是几位数
  18. int maxLength = (max + "").length();
  19. //定义一个二维数组,表示10个桶, 每个桶就是一个一维数组
  20. //说明
  21. //1. 二维数组包含10个一维数组
  22. //2. 为了防止在放入数的时候,数据溢出,则每个一维数组(桶),大小定为arr.length
  23. //3. 名明确,基数排序是使用空间换时间的经典算法
  24. int[][] bucket = new int[10][arr.length];
  25. //为了记录每个桶中,实际存放了多少个数据,我们定义一个一维数组来记录各个桶的每次放入的数据个数
  26. //可以这里理解
  27. //比如:bucketElementCounts[0] , 记录的就是 bucket[0] 桶的放入数据个数
  28. int[] bucketElementCounts = new int[10];
  29. //这里我们使用循环将代码处理
  30. for(int i = 0 , n = 1; i < maxLength; i++, n *= 10) {
  31. //(针对每个元素的对应位进行排序处理), 第一次是个位,第二次是十位,第三次是百位..
  32. for(int j = 0; j < arr.length; j++) {
  33. //取出每个元素的对应位的值
  34. int digitOfElement = arr[j] / n % 10;
  35. //放入到对应的桶中
  36. bucket[digitOfElement][bucketElementCounts[digitOfElement]] = arr[j];
  37. bucketElementCounts[digitOfElement]++;
  38. }
  39. //按照这个桶的顺序(一维数组的下标依次取出数据,放入原来数组)
  40. int index = 0;
  41. //遍历每一桶,并将桶中是数据,放入到原数组
  42. for(int k = 0; k < bucketElementCounts.length; k++) {
  43. //如果桶中,有数据,我们才放入到原数组
  44. if(bucketElementCounts[k] != 0) {
  45. //循环该桶即第k个桶(即第k个一维数组), 放入
  46. for(int l = 0; l < bucketElementCounts[k]; l++) {
  47. //取出元素放入到arr
  48. arr[index++] = bucket[k][l];
  49. }
  50. }
  51. //第i+1轮处理后,需要将每个 bucketElementCounts[k] = 0 !!!!
  52. bucketElementCounts[k] = 0;
  53. }
  54. //System.out.println("第"+(i+1)+"轮,对个位的排序处理 arr =" + Arrays.toString(arr));
  55. }
  56. /*
  57. //第1轮(针对每个元素的个位进行排序处理)
  58. for(int j = 0; j < arr.length; j++) {
  59. //取出每个元素的个位的值
  60. int digitOfElement = arr[j] / 1 % 10;
  61. //放入到对应的桶中
  62. bucket[digitOfElement][bucketElementCounts[digitOfElement]] = arr[j];
  63. bucketElementCounts[digitOfElement]++;
  64. }
  65. //按照这个桶的顺序(一维数组的下标依次取出数据,放入原来数组)
  66. int index = 0;
  67. //遍历每一桶,并将桶中是数据,放入到原数组
  68. for(int k = 0; k < bucketElementCounts.length; k++) {
  69. //如果桶中,有数据,我们才放入到原数组
  70. if(bucketElementCounts[k] != 0) {
  71. //循环该桶即第k个桶(即第k个一维数组), 放入
  72. for(int l = 0; l < bucketElementCounts[k]; l++) {
  73. //取出元素放入到arr
  74. arr[index++] = bucket[k][l];
  75. }
  76. }
  77. //第l轮处理后,需要将每个 bucketElementCounts[k] = 0 !!!!
  78. bucketElementCounts[k] = 0;
  79. }
  80. System.out.println("第1轮,对个位的排序处理 arr =" + Arrays.toString(arr));
  81. //==========================================
  82. //第2轮(针对每个元素的十位进行排序处理)
  83. for (int j = 0; j < arr.length; j++) {
  84. // 取出每个元素的十位的值
  85. int digitOfElement = arr[j] / 10 % 10; //748 / 10 => 74 % 10 => 4
  86. // 放入到对应的桶中
  87. bucket[digitOfElement][bucketElementCounts[digitOfElement]] = arr[j];
  88. bucketElementCounts[digitOfElement]++;
  89. }
  90. // 按照这个桶的顺序(一维数组的下标依次取出数据,放入原来数组)
  91. index = 0;
  92. // 遍历每一桶,并将桶中是数据,放入到原数组
  93. for (int k = 0; k < bucketElementCounts.length; k++) {
  94. // 如果桶中,有数据,我们才放入到原数组
  95. if (bucketElementCounts[k] != 0) {
  96. // 循环该桶即第k个桶(即第k个一维数组), 放入
  97. for (int l = 0; l < bucketElementCounts[k]; l++) {
  98. // 取出元素放入到arr
  99. arr[index++] = bucket[k][l];
  100. }
  101. }
  102. //第2轮处理后,需要将每个 bucketElementCounts[k] = 0 !!!!
  103. bucketElementCounts[k] = 0;
  104. }
  105. System.out.println("第2轮,对个位的排序处理 arr =" + Arrays.toString(arr));
  106. //第3轮(针对每个元素的百位进行排序处理)
  107. for (int j = 0; j < arr.length; j++) {
  108. // 取出每个元素的百位的值
  109. int digitOfElement = arr[j] / 100 % 10; // 748 / 100 => 7 % 10 = 7
  110. // 放入到对应的桶中
  111. bucket[digitOfElement][bucketElementCounts[digitOfElement]] = arr[j];
  112. bucketElementCounts[digitOfElement]++;
  113. }
  114. // 按照这个桶的顺序(一维数组的下标依次取出数据,放入原来数组)
  115. index = 0;
  116. // 遍历每一桶,并将桶中是数据,放入到原数组
  117. for (int k = 0; k < bucketElementCounts.length; k++) {
  118. // 如果桶中,有数据,我们才放入到原数组
  119. if (bucketElementCounts[k] != 0) {
  120. // 循环该桶即第k个桶(即第k个一维数组), 放入
  121. for (int l = 0; l < bucketElementCounts[k]; l++) {
  122. // 取出元素放入到arr
  123. arr[index++] = bucket[k][l];
  124. }
  125. }
  126. //第3轮处理后,需要将每个 bucketElementCounts[k] = 0 !!!!
  127. bucketElementCounts[k] = 0;
  128. }
  129. System.out.println("第3轮,对个位的排序处理 arr =" + Arrays.toString(arr)); */
  130. }

}

```