PHP 零基础入门笔记(15):算法 algorithm

简介: PHP 零基础入门笔记(15):算法 algorithm

冒泡排序

两两比较,顺序错误就交换,直到该数列已经完成排序


算法思路


比较相邻的元素,顺序不对就交换


代码实现


<?php
// 将数组由小到大排序
$arr = [3, 4, 2, 8, 9, 1, 6];
// 每次遍历将最大值放在最右边
for($i = 0, $len = count($arr); $i < $len; $i++){
    // 将大值的放在右边
    for($j = 0, $len = count($arr); $j < $len - 1 - $i; $j++){
        // 如果左边的值比右边的大,就交换位置
        if($arr[$j] > $arr[$j + 1]){
            $temp = $arr[$j];
            $arr[$j] = $arr[$j + 1];
            $arr[$j + 1] = $temp;
        }
    }
}
echo json_encode($arr);
// [1,2,3,4,6,8,9]

选择排序

每次从待排的数据中选择最小或最大的一个元素,放在起始位置,直到全部待排序的元素排完


<?php
// 将数组由小到大排序
$arr = [3, 4, 2, 8, 9, 1, 6];
// 1、需要选择的次数,每次只能选择一个最大或者最小值
for($i = 0, $len = count($arr); $i < $len; $i++){
    // 2、假设第一个值就是最小值
    $min = $i;
    // 3、将最小值和其他剩余值一一比较
    for($j = 1 + $i, $len = count($arr); $j < $len; $j++){
        // 4、找到最小值
        if($arr[$min] > $arr[$j]){
            $min = $j;
        }
    }
    // 5、当前值与最小值交换位置
    if($min != $i){
        $temp = $arr[$i];
        $arr[$i] = $arr[$min];
        $arr[$min] = $temp;
    }
}
echo json_encode($arr);
// [1,2,3,4,6,8,9]

插入排序

将一个数据插入到已排序数据中


<?php
// 将数组由小到大排序
$arr = [3, 4, 2, 8, 9, 1, 6];
// 1、循环次数,插入次数 length - 1,第一个数已经是有序的了
for($i = 1, $len = count($arr); $i < $len; $i++){
    // 待插入数据
    $temp = $arr[$i];
    // 2、前面的序列是已排序区域
    for($j = $i - 1; $j >= 0; $j--){
        // 3、将该值插入到合适的位置
        if($arr[$j] > $temp){
            // 有序数组后移一位
            $arr[$j + 1] = $arr[$j];
            $arr[$j] = $temp;
        } else{
            // 该值最大,位置正确
            break;
        }
    }
}
echo json_encode($arr);
// [1,2,3,4,6,8,9]

优化版本, 减少交换次数


<?php
// 将数组由小到大排序
$arr = [3, 4, 2, 8, 9, 1, 6];
// 1、循环次数,插入次数 length - 1,第一个数已经是有序的了
for($i = 1, $len = count($arr); $i < $len; $i++){
    // 待插入数据
    $temp = $arr[$i];
    // 记录插入位置
    $insert_index = -1;
    // 2、前面的序列是已排序区域
    for($j = $i - 1; $j >= 0; $j--){
        // 3、将该值插入到合适的位置
        if($arr[$j] > $temp){
            // 有序数组后移一位
            $arr[$j + 1] = $arr[$j];
            // $arr[$j] = $temp;
            $insert_index = $j;
        } else{
            // 该值最大,位置正确
            break;
        }
    }
    // 如果位置有变动,交换数据
    if($insert_index > -1){
        $arr[$insert_index] = $temp;
    }
}
echo json_encode($arr);
// [1,2,3,4,6,8,9]

快速排序

不稳定排序


冒泡排序的改进,通过一趟排序将要排序的数据分割为独立的两部分,其中一部分的数据都比另外一部分所有数据都要小,然后重复此过程


排序步骤:


从数组中选出一个元素(通常是第一个),作为参照

定义两个数组,将目标数组中剩余元素与参照元素挨个比较,小的放到一组,大的放到另一组

第二步执行完后,前后数组顺序不确定,但是确定了自己的位置

将得到的小数组按照第1步第3步,重复操作(子问题)

回溯最小数组(第一个元素)

<?php
// 将数组由小到大排序
$arr = [3, 4, 2, 8, 9, 1, 6];
/**
 * 快速排序
 */
function quick_sort($arr){
    // 递归出口
    $len = count($arr);
    if($len <= 1){
        return $arr;
    }
    // 定义两个数组,分别存小数和大数
    $left = [];
    $right = [];
    for($i = 1; $i < $len; $i++){
        // 比较:小的放左边,大的放右边
        if($arr[$i] < $arr[0]){
            $left[] = $arr[$i];
        } else {
            $right[] = $arr[$i];
        }
    }
    // 递归点:将两个数组继续排序
    $left = quick_sort($left);
    $right = quick_sort($right);
    // 和并数组返回
    return array_merge($left, [$arr[0]], $right);
}
echo json_encode(quick_sort($arr));
// [1,2,3,4,6,8,9]

归并排序

分治算法


二路归并


<?php
// 将数组由小到大排序
$arr1 = [2, 3, 4, 8];
$arr2 = [1, 6, 9];
// 定义归并空间
$arr3 = [];
while(count($arr1) && count($arr2)){
    // 每次取出一个最小值
    $arr3[] = $arr1[0] < $arr2[0] ? array_shift($arr1): array_shift($arr2);
}
// 合并结果
$ret = array_merge($arr3, $arr1, $arr2);
echo json_encode($ret);
// [1,2,3,4,6,8,9]
<?php
// 将数组由小到大排序
$arr = [3, 4, 2, 8, 9, 1, 6];
function merge_sort($arr){
    // 递归出口
    $len = count($arr);
    if($len <= 1){
        return $arr;
    }
    // 拆分为两个数组
    $middle_index = floor($len/2);
    $left = array_slice($arr, 0, $middle_index);
    $right = array_slice($arr, $middle_index);
    // 递归点
    $left = merge_sort($left);
    $right = merge_sort($right);
    // 定义归并空间
    $merge_array = [];
    while(count($left) && count($right)){
        // 每次取出一个最小值
        $merge_array[] = $left[0] < $right[0] ? array_shift($left): array_shift($right);
    }
    // 合并结果
    return array_merge($merge_array, $left, $right);
}
// 合并结果
$ret = merge_sort($arr);
echo json_encode($ret);
// [1,2,3,4,6,8,9]

顺序查找

按照列表顺序,依次查找


<?php
/**
 * 顺序查找,逐项匹配
 */
function order_find($arr, $value){
    for($i = 0, $len = count($arr); $i < $len; $i++){
        // 判断
        if($arr[$i] == $value){
            return $i;
        }
    }
    return -1;
}
$arr = [3, 4, 2, 8, 9, 1, 6];
$ret = order_find($arr, 1);
var_dump($ret);
// int(5)

二分查找

要求元素组有序,先与中间节点比较,分成两个子表,确定继续在哪个子表查找,递归进行


折半查找


<?php
/**
 * 二分查找,从中间的值开始比较
 */
function binary_find($arr, $value){
    // 边界条件
    $left = 0;
    $right = count($arr);
    // 循环匹配
    while($left <= $right){
        // 每次循环获取中间值
        $middle = floor(($left + $right) / 2);
        if($value > $arr[$middle]){
            $left = $middle + 1;
        } else if($value < $arr[$middle]){
            $right = $middle - 1;
        } else{
            return $middle;
        }
    }
    return -1;
}
// 有序数组
$arr = [1, 2, 4, 8, 9, 13, 16];
$ret = binary_find($arr, 9);
var_dump($ret);
// float(4)

相关文章
|
1月前
|
机器学习/深度学习 算法 网络性能优化
【博士每天一篇文献-算法】A brain-inspired algorithm that mitigates catastrophic forgetting of
本文提出了一种受大脑启发的神经调节辅助信用分配(NACA)算法,该算法通过模拟大脑中的神经调节机制,有效减轻了人工神经网络(ANNs)和脉冲神经网络(SNNs)在学习过程中的灾难性遗忘问题,并具有较低的计算成本。
33 1
|
2月前
|
算法 PHP
【php经典算法】冒泡排序,冒泡排序原理,冒泡排序执行逻辑,执行过程,执行结果 代码
【php经典算法】冒泡排序,冒泡排序原理,冒泡排序执行逻辑,执行过程,执行结果 代码
22 1
|
3月前
|
存储 算法 Java
技术笔记:JVM的垃圾回收机制总结(垃圾收集、回收算法、垃圾回收器)
技术笔记:JVM的垃圾回收机制总结(垃圾收集、回收算法、垃圾回收器)
39 1
|
3月前
|
算法 PHP 数据中心
基于php雪花算法工具类Snowflake -来自chatGPT
基于php雪花算法工具类Snowflake -来自chatGPT
|
2月前
|
存储 算法 C语言
软考中级之数据库系统工程师笔记总结(二)数据结构与算法
软考中级之数据库系统工程师笔记总结(二)数据结构与算法
22 0
|
2月前
|
存储 算法 程序员
|
3月前
|
机器学习/深度学习 算法 数据可视化
技术心得记录:机器学习笔记之聚类算法层次聚类HierarchicalClustering
技术心得记录:机器学习笔记之聚类算法层次聚类HierarchicalClustering
34 0
|
3月前
|
Java BI C#
技术笔记:SM4加密算法实现Java和C#相互加密解密
技术笔记:SM4加密算法实现Java和C#相互加密解密
50 0
|
3月前
|
存储 人工智能 算法
程序与技术分享:Acwing算法笔记
程序与技术分享:Acwing算法笔记
|
3月前
|
算法 安全 Java
技术笔记:MD5加密算法详解
技术笔记:MD5加密算法详解