php语言的7种基本的排序方法


Posted in PHP onDecember 28, 2020

本文总结了一下常用的7种排序方法,并用php语言实现。

1、直接插入排序

/*
 * 直接插入排序,插入排序的思想是:当前插入位置之前的元素有序,
 * 若插入当前位置的元素比有序元素最后一个元素大,则什么也不做,
 * 否则在有序序列中找到插入的位置,并插入
 */
function insertSort($arr) {
 $len = count($arr); 
 for($i = 1; $i < $len; $i++) {
  if($arr[$i-1] > $arr[i]) {
   for($j = $i - 1;$j >= 0; $j-- ) {
    $tmp = $arr[$j+1];
    if($tmp < $arr[$j]) {
     $arr[$j+1] = $arr[$j];
     $arr[$j] = $tmp;
    }else{
     break;
    }     
   } 
  }
 } 
 return $arr;
}

2、冒泡排序

/*
 冒泡排序,冒泡排序思想:进行 n-1 趟冒泡排序, 每趟两两比较调整最大值到数组(子数组)末尾
*/
function bubbleSort($arr) {
 $len = count($arr);
 for($i = 1; $i < $len; $i++) {
  for($j = 0; $j < $len-$i; $j++) {
   if($arr[$j] > $arr[$j+1]) {
    $tmp = $arr[$j+1];
    $arr[$j+1] = $arr[$j];
    $arr[$j] = $tmp;
   }
  }
 }
 return $arr;
}

3、简单选择排序

/*
 简单选择排序, 简单排序思想:从数组第一个元素开始依次确定从小到大的元素
*/
function selectSort($arr) {
 $len = count($arr);
 for($i = 0; $i < $len; $i++) {
  $k = $i;
  for($j = $i+1; $j < $len; $j++) {
   if($arr[$k] > $arr[$j]) {
    $k = $j;
   }
  }
  if($k != $i) {
   $tmp = $arr[$i];
   $arr[$i] = $arr[$k];
   $arr[$k] = $tmp;
  }
 }
 return $arr;
}

4、希尔排序

/*
 希尔排序,希尔排序原理:将数组按指定步长分隔成若干子序列,然后分别对子序列进行排序(在这是直接)
*/
function shellSort($arr) {
 $len = count($arr);
 $k = floor($len/2);
 while($k > 0) {
  for($i = 0; $i < $k; $i++) {
   for($j = $i; $j < $len, ($j + $k) < $len; $j = $j + $k) {
    if($arr[$j] > $arr[$j+$k]) {
     $tmp = $arr[$j+$k];
     $arr[$j+$k] = $arr[$j];
     $arr[$j] = $tmp;
    }
   }
  }
  $k = floor($k/2);
 }
 return $arr;
}

5、快速排序

/*
 * 快速排序,快排思想:通过一趟排序将待排的记录分为两个独立的部分,其中一部分的记录的关键字均不大于
 * 另一部分记录的关键字,然后再分别对这两部分记录继续进行快速排序,以达到整个序列有序,具体做法需要
 * 每趟排序设置一个标准关键字和分别指向头一个记录的关键字和最后一个记录的关键字的指针。
 * quickSort($arr, 0, count($arr) -1);
 */
function quickSort(&$arr,$low,$high) {
 if($low < $high) {
  $i = $low;
  $j = $high;
  $primary = $arr[$low];
  while($i < $j) {
   while($i < $j && $arr[$j] >= $primary) {
    $j--;
   }
   if($i < $j) {
    $arr[$i++] = $arr[$j];
   }
   while($i < $j && $arr[$i] <= $primary) {
    $i++;
   }
   if($i < $j) {
    $arr[$j--] = $arr[$i];
   }
  }
  $arr[$i] = $primary;
  quickSort($arr, $low, $i-1);
  quickSort($arr, $i+1, $high);
 }
}

6、堆排序

/*
 堆排序
*/

// 调整子堆的为大根堆的过程,$s为子堆的根的位置,$m为堆最后一个元素位置
function heapAdjust(&$arr, $s, $m) {
 $tmp = $arr[$s];
 // 在调整为大根堆的过程中可能会影响左子堆或右子堆
 // for循环的作用是要保证子堆也是大根堆
 for($j = 2*$s + 1; $j <= $m; $j = 2*$j + 1) {
  // 找到根节点的左右孩子中的最大者,然后用这个最大者与根节点比较,
  // 若大则进行调整,否则符合大根堆的 特点跳出循环 
  if($j < $m && $arr[$j] < $arr[$j+1]) {
   $j++;
  }
  if($tmp >= $arr[$j] ) {
   break;
  }
  $arr[$s] = $arr[$j];
  $s = $j;
 }
 $arr[$s] = $tmp;
}

// 堆排序
function heapSort($arr) {
 $len = count($arr);
 // 依次从子堆开始调整堆为大根堆
 for($i = floor($len/2-1); $i >= 0; $i--) {
  heapAdjust($arr, $i, $len-1);
 }
 // 依次把根节点调换至最后一个位置,再次调整堆为大根堆,找到次最大值,
 // 依次类推得到一个有序数组
 for($n = $len-1; $n > 0; $n--) {
  $tmp = $arr[$n];
  $arr[$n] = $arr[0];
  $arr[0] = $tmp;
  heapAdjust($arr, 0, $n-1);
 }
 return $arr;
}

7、归并排序

/*
 归并排序,这里实现的是两路归并
*/
// 分别将有序的$arr1[s..m]、$arr2[m+1..n]归并为有序的$arr2[s..n]
function Merge(&$arr1, &$arr2, $s, $m, $n) {
 for($k = $s,$i = $s, $j = $m+1; $i <= $m && $j <= $n; $k++) {
  if($arr1[$i]<$arr1[$j]) {
   $arr2[$k] = $arr1[$i++];
  }else {
   $arr2[$k] = $arr1[$j++];
  }
 }
 if($i <= $m) {
  for(; $i <= $m; $i++) {
   $arr2[$k++] = $arr1[$i];
  }
 } else if($j <= $n) {
  for(; $j <= $n; $j++) {
   $arr2[$k++] = $arr1[$j];
  }
 }
}

// 递归形式的两路归并
function MSort(&$arr1, &$arr2, $s, $t) {
 if($s == $t) {
  $arr2[$s] = $arr1[$s];
 }else {
  $m = floor(($s+$t)/2);
  $tmp_arr = array();
  MSort($arr1, $tmp_arr, $s, $m);
  MSort($arr1, $tmp_arr, $m+1, $t);
  Merge($tmp_arr, $arr2, $s, $m, $t);
 }
}

// 对一位数组$arr[0..n-1]中的元素进行两路归并
function mergeSort($arr) {
 $len = count($arr);
 MSort($arr, $arr, 0, $len-1);
 return $arr;
}

使用经验

  1. 若排序的记录数目n较小时,可以采用直接插入排序和简单选择排序,当记录本身信息量较大时,用简单选择排序方法较好。
  2. 若待排序记录按关键字基本有序,适合采用直接插入排序和冒泡排序。
  3. 若n值较大时,可以采用快速排序、堆排序和归并排序。另外快速排序被认为是内部排序方法中最好的方法。

以上就是本文的全部内容,希望对大家的学习有所帮助。

PHP 相关文章推荐
坏狼的PHP学习教程之第2天
Jun 15 PHP
php 缩略图实现函数代码
Jun 23 PHP
php遍历文件夹下的所有文件和子文件夹示例
Mar 20 PHP
PHP的Yii框架入门使用教程
Feb 15 PHP
php使用正则验证中文
Apr 06 PHP
Yii数据读取与跳转参数传递用法实例分析
Jul 12 PHP
Yii2.0表关联查询实例分析
Jul 18 PHP
thinkPHP中钩子的两种配置调用方法详解
Nov 11 PHP
PHP+jQuery实现滚屏无刷新动态加载数据功能详解
May 04 PHP
yii2中LinkPager增加总页数和总记录数的实例
Aug 28 PHP
PHP XML Expat解析器知识点总结
Feb 15 PHP
基于laravel where的高级使用方法
Oct 10 PHP
php实现图片上传并利用ImageMagick生成缩略图
Mar 14 #PHP
YII Framework框架教程之国际化实现方法
Mar 14 #PHP
YII Framework框架教程之缓存用法详解
Mar 14 #PHP
YII Framework框架教程之安全方案详解
Mar 14 #PHP
YII Framework框架教程之日志用法详解
Mar 14 #PHP
YII Framework教程之异常处理详解
Mar 14 #PHP
Zend Framework教程之Application用法实例详解
Mar 14 #PHP
You might like
php xml文件操作实现代码(二)
2009/03/20 PHP
php中的观察者模式
2010/03/24 PHP
php计算2个日期的差值函数分享
2015/02/02 PHP
替换php字符串中的单引号为双引号的方法
2017/02/16 PHP
修改发贴的编辑功能
2007/03/07 Javascript
jquery下实现overlay遮罩层代码
2010/08/25 Javascript
javascript的创建多行字符串的7种方法
2014/04/29 Javascript
javascript日期处理函数,性能优化批处理
2015/09/06 Javascript
vue使用watch 观察路由变化,重新获取内容
2017/03/08 Javascript
vue图片加载失败时用默认图片替换的方法
2019/08/29 Javascript
微信小程序实现锚点功能
2019/11/20 Javascript
浅谈Node新版本13.2.0正式支持ES Modules特性
2019/11/25 Javascript
[03:02]安得倚天剑,跨海斩长鲸——中国军团出征DOTA2国际邀请赛
2018/08/14 DOTA
Python开发实例分享bt种子爬虫程序和种子解析
2014/05/21 Python
Python中生成Epoch的方法
2017/04/26 Python
Django通过dwebsocket实现websocket的例子
2019/11/15 Python
Pytorch DataLoader 变长数据处理方式
2020/01/08 Python
用pytorch的nn.Module构造简单全链接层实例
2020/01/14 Python
Python随机数函数代码实例解析
2020/02/09 Python
Python ConfigParser模块的使用示例
2020/10/12 Python
python 爬取腾讯视频评论的实现步骤
2021/02/18 Python
CSS3中的5个有趣的新技术
2009/04/02 HTML / CSS
德国骆驼商店:ActiveFashionWorld
2017/11/18 全球购物
Jacadi Paris英国官网:法国童装品牌
2019/08/09 全球购物
行政助理岗位职责范文
2013/12/03 职场文书
大学生求职自荐信
2013/12/12 职场文书
五十岁生日宴会答谢词
2014/01/15 职场文书
手工社团活动方案
2014/02/17 职场文书
毕业评语大全
2014/05/04 职场文书
建筑院校毕业生求职信
2014/06/13 职场文书
单位婚育证明范本
2014/11/21 职场文书
小学生通知书评语
2014/12/31 职场文书
营运督导岗位职责
2015/04/10 职场文书
银行资信证明
2015/06/17 职场文书
Python NumPy灰度图像的压缩原理讲解
2021/08/04 Python
Java线程的6种状态与生命周期
2022/05/11 Java/Android