七大排序算法java实现

最常见的七大排序算法,整理了下,包括时间复杂度和空间复杂度都做一个详细的解说,希望对需要的人能有所帮助。

/**

* 1.插入排序算法

* @paramint[]  未排序数组

* @returnint[]  排完序数组

*

* 插入排序的基本思想是在遍历数组的过程中,假设在序号 i 之前的元素即 [0..i-1] 都已经排好序,

* 本趟需要找到 i 对应的元素 x 的正确位置 k ,并且在寻找这个位置 k 的过程中逐个将比较过的元素往后移一位,

* 为元素 x “腾位置”,最后将 k 对应的元素值赋为 x

*

* 一般情况下,插入排序的时间复杂度和空间复杂度分别为 O(n2 ) 和 O(1)

*/

public int[] sortInsert(int[] array){

for(int i=1;i

int temp = array[i];

int j;

for(j=i-1;j >= 0 && temp< array[j]; j--){

array[j + 1] = array[j];

}

array[j + 1] = temp;

}

return array;

}

/**

* 2.选择排序算法

* @paramint[]  未排序数组

* @returnint[]  排完序数组

*

* 选择排序的基本思想是遍历数组的过程中,以 i 代表当前需要排序的序号,则需要在剩余的 [i…n-1] 中找出其中的最小值,

* 然后将找到的最小值与 i 指向的值进行交换。因为每一趟确定元素的过程中都会有一个选择最大值的子流程,所以人们形象地称之为选择排序。

* 选择排序的时间复杂度和空间复杂度分别为 O(n2 ) 和 O(1)

*/

public int[] sortSelect(int[] arr){

for (int i = 0; i < arr.length; i++) {

int miniPost = i;

for (int m = i + 1; m < arr.length; m++) {

if (arr[m] < arr[miniPost]) {

miniPost = m;

}

}

if (arr[i] > arr[miniPost]) {

int temp;

temp = arr[i];

arr[i] = arr[miniPost];

arr[miniPost] = temp;

}

}

return arr;

}

/**

* 3.冒泡排序算法

* @paramint[]  未排序数组

* @returnint[]  排完序数组

*

* 冒泡排序是將比較大的數字沉在最下面,较小的浮在上面

*

*/

public int[] sortBubble(int[] array){

int temp;

// 第一层循环:表明比较的次数, 比如 length 个元素,比较次数为 length-1 次(肯定不需和自己比)

for(int i=0;i

for (int j = array.length - 1; j > i; j--) {

if (array[j] < array[j - 1]) {

temp = array[j];

array[j] = array[j - 1];

array[j - 1] = temp;

}

}

}

return array;

}

/**

* 4.快速排序算法

* @paramint[]  未排序数组

* @returnint[]  排完序数组

* 通过一趟排序将待排记录分割成独立的两部分,其中一部分记录的关键字均比另一部分的关键字小,

* 则可以分别对这两部分记录继续进行排序,已达到整个序列有序的目的

* 本质就是,找一个基位(枢轴,分水岭,作用是左边的都比它小,右边的都比它大.可随机,取名base

* 首先从序列最右边开始找比base小的,如果小,换位置,从而base移到刚才右边(比较时比base小)的位置(记为临时的high位),这样base右边的都比base大

* 然后,从序列的最左边开始找比base大的

* ,如果大,换位置,从而base移动到刚才左边(比较时比base大)的位置(记为临时的low位),这样base左边的都比base小

* 循环以上两步,直到 low == heigh, 这使才真正的找到了枢轴,分水岭. 返回这个位置,分水岭左边和右边的序列,分别再来递归

*

*/

public int[] sortQuick(int[] array){

return quickSort(array, 0, array.length-1);

}

private  int[] quickSort(int[] arr, int low, int heigh) {

if (low < heigh) {

int division = partition(arr, low, heigh);

quickSort(arr, low, division - 1);

quickSort(arr, division + 1, heigh);

}

return arr;

}

// 分水岭,基位,左边的都比这个位置小,右边的都大

private int partition(int[] arr, int low, int heigh) {

int base = arr[low]; //用子表的第一个记录做枢轴(分水岭)记录

while (low < heigh) {  //从表的两端交替向中间扫描

while (low < heigh && arr[heigh] >= base) {

heigh--;

}

// base 赋值给 当前 heigh 位,base 挪到(互换)到了这里,heigh位右边的都比base大

swap(arr, heigh, low);

while (low < heigh && arr[low] <= base) {

low++;

}

// 遇到左边比base值大的了,换位置

swap(arr, heigh, low);

}

// now low = heigh;

return low;

}

private  void swap(int[] arr, int a, int b) {

int temp;

temp = arr[a];

arr[a] = arr[b];

arr[b] = temp;

}

/**

* 5.合并排序算法

* @paramint[]  未排序数组

* @returnint[]  排完序数组

*

* 归并排序采用的是递归来实现,属于“分而治之”,将目标数组从中间一分为二,之后分别对这两个数组进行排序,

* 排序完毕之后再将排好序的两个数组“归并”到一起,归并排序最重要的也就是这个“归并”的过程,

* 归并的过程中需要额外的跟需要归并的两个数组长度一致的空间

*/

private int[] sort(int[] nums, int low, int high) {

int mid = (low + high) / 2;

if (low < high) {

// 左边

sort(nums, low, mid);

// 右边

sort(nums, mid + 1, high);

// 左右归并

merge(nums, low, mid, high);

}

return nums;

}

private void merge(int[] nums, int low, int mid, int high) {

int[] temp = new int[high - low + 1];

int i = low;// 左指针

int j = mid + 1;// 右指针

int k = 0;

// 把较小的数先移到新数组中

while (i <= mid && j <= high) {

if (nums[i] < nums[j]) {

temp[k++] = nums[i++];

} else {

temp[k++] = nums[j++];

}

}

// 把左边剩余的数移入数组

while (i <= mid) {

temp[k++] = nums[i++];

}

// 把右边边剩余的数移入数组

while (j <= high) {

temp[k++] = nums[j++];

}

// 把新数组中的数覆盖nums数组

for (int k2 = 0; k2 < temp.length; k2++) {

nums[k2 + low] = temp[k2];

}

}

public int[] sortMerge(int[] array) {

return sort(array, 0, array.length - 1);

}

/**

* 6.希尔排序算法

* @paramint[]  未排序数组

* @returnint[]  排完序数组

*

* 希尔排序的诞生是由于插入排序在处理大规模数组的时候会遇到需要移动太多元素的问题。希尔排序的思想是将一个大的数组“分而治之”,

* 划分为若干个小的数组,以 gap 来划分,比如数组 [1, 2, 3, 4, 5, 6, 7, 8] ,如果以 gap = 2 来划分,

* 可以分为 [1, 3, 5, 7] 和 [2, 4, 6, 8] 两个数组(对应的,如 gap = 3 ,

* 则划分的数组为: [1, 4, 7] 、 [2, 5, 8] 、 [3, 6] )然后分别对划分出来的数组进行插入排序,

* 待各个子数组排序完毕之后再减小 gap 值重复进行之前的步骤,直至 gap = 1 ,即对整个数组进行插入排序,

* 此时的数组已经基本上快排好序了,所以需要移动的元素会很小很小,解决了插入排序在处理大规模数组时较多移动次数的问题

*

* 希尔排序是插入排序的改进版,在数据量大的时候对效率的提升帮助很大,数据量小的时候建议直接使用插入排序就好了。

*/

public int[] sortShell(int[] array) {

// 取增量

int step = array.length / 2;

while (step >= 1) {

for (int i = step; i < array.length; i++) {

int temp = array[i];

int j = 0;

// 跟插入排序的区别就在这里

for (j = i - step; j >= 0 && temp < array[j]; j -= step) {

array[j + step] = array[j];

}

array[j + step] = temp;

}

step /= 2;

}

return array;

}

/**

* 7.堆排序算法

* @paramint[]  未排序数组

* @returnint[]  排完序数组

*

* 本质就是先构造一个大顶堆,parent比children大,root节点就是最大的节点

* 把最大的节点(root)与尾节点(最后一个节点,比较小)位置互换

* 剩下最后的尾节点,现在最大,其余的,从第一个元素开始到尾节点前一位,构造大顶堆递归

*

*/

public int[] sortHeap(int[] array) {

buildHeap(array);// 构建堆

int n = array.length;

int i = 0;

for (i = n - 1; i >= 1; i--) {

swap(array, 0, i);

heapify(array, 0, i);

}

return array;

}

private void buildHeap(int[] array) {

int n = array.length;// 数组中元素的个数

for (int i = n / 2 - 1; i >= 0; i--)

heapify(array, i, n);

}

private void heapify(int[] A, int idx, int max) {

int left = 2 * idx + 1;// 左孩子的下标(如果存在的话)

int right = 2 * idx + 2;// 左孩子的下标(如果存在的话)

int largest = 0;// 寻找3个节点中最大值节点的下标

if (left < max && A[left] > A[idx])

largest = left;

else

largest = idx;

if (right < max && A[right] > A[largest])

largest = right;

if (largest != idx) {

swap(A, largest, idx);

heapify(A, largest, max);

}

}

// 建堆函数,认为【s,m】中只有 s

// 对应的关键字未满足大顶堆定义,通过调整使【s,m】成为大顶堆=====================================================

public static void heapAdjust(int[] array, int s, int m) {

// 用0下标元素作为暂存单元

array[0] = array[s];

// 沿孩子较大的结点向下筛选

for (int j = 2 * s; j <= m; j *= 2) {

// 保证j为较大孩子结点的下标,j < m 保证 j+1 <= m ,不越界

if (j < m && array[j] < array[j + 1]) {

j++;

}

if (!(array[0] < array[j])) {

break;

}

// 若S位较小,应将较大孩子上移

array[s] = array[j];

// 较大孩子的值变成S位的较小值,可能引起顶堆的不平衡,故对其所在的堆进行筛选

s = j;

}

// 若S位较大,则值不变;否则,S位向下移动至2*s、4*s、。。。

array[s] = array[0];

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容