函数式编程思想

2024-09-09 07:04
文章标签 函数 编程 思想

本文主要是介绍函数式编程思想,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

我们经常会用到各种各样的编程思想,例如面向过程、面向对象。不过笔者在该博客简单介绍一下函数式编程思想.

如果对函数式编程思想进行概括,就是f(x) = na(x) , y=uf(x)…至于其他的编程思想,可能是y=a(x)+b(x)+c(x)…,也有可能是y=f(x)=f(x)/a + f(x)/b+f(x)/c…

面向过程的指令式编程

面向过程,简单理解就是y=a(x)+b(x)+c(x)…这种,可能有人会问面向对象,嗯,对于这种小型精巧的分治算法,使用面向对象有点像是无稽之谈,所以指令式的面向过程往往是一个好选择。

下面以快速排序为例,现在我们需要对数组进行从小到大排序。

快速排序的核心就是:将数组划分为左右两个子集,保证右边的元素比左边大,然后不断递归重复这个过程。

我们要实现排序的过程如下:

分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
全集
左子集
右子集
左子集的左子集
左左左子集
左左右子集
左子集的右子集
左右左子集
左右右子集
右子集的左子集
右左左子集
右左右子集
右子集的右子集
右右左子集
右右右子集

显然,这是一个不断递归的过程,但是我们可以观察到,程序总是在重复分割交换这个过程,因此将交换和分割单独写一个函数,作为基本指令。因此我们需要有三个函数:交换函数、分区函数、排序函数。

过程如下:

#include <stdio.h>// 交换两个元素
void swap(int* a, int* b) {int temp = *a;*a = *b;*b = temp;
}// 分区函数
int partition(int arr[], int low, int high) {int pivot = arr[high]; // 选择最后一个元素作为枢轴int i = (low - 1); for (int j = low; j <= high - 1; j++) {// 如果当前元素小于或等于枢轴if (arr[j] <= pivot) {i++; //j查找比枢纽小的元素,i++后必然指向比枢纽大的元素,否则i与j会同步更新swap(&arr[i], &arr[j]);}}swap(&arr[i + 1], &arr[high]);return (i + 1);
}/*
//笔者更喜欢使用下面这种,分析源码就可以知道,下面更接近二分查找的形式,而上面更接近从头遍历。
//经过笔者测试,上面的平均耗时几乎是下面的1.7倍int partition(int arr[], int low, int high) {int pivot = arr[high]; // 选择最后一个元素作为枢轴int i = low - 1; // 较小元素的索引int j = high ;for( ; ; ){while( arr[++i] < pivot){ }while( arr[--j] > pivot && j > low){ }if( i < j)swap(&arr[i],&arr[j]);elsebreak;}swap(&arr[i], &arr[high]);return i ;
}*/// 快速排序函数
void quickSort(int arr[], int low, int high) {if (low < high) {// pi 是分区索引,arr[pi] 已经排好序int pi = partition(arr, low, high);// 分别排序两个子数组quickSort(arr, low, pi - 1);quickSort(arr, pi + 1, high);}
}// 打印数组
void printArray(int arr[], int size) {for (int i = 0; i < size; i++)printf("%d ", arr[i]);printf("\n");
}int main() {int arr[] = {10, 7, 8, 9, 1, 5};int n = sizeof(arr) / sizeof(arr[0]);quickSort(arr, 0, n - 1);printf("Sorted array: \n");printArray(arr, n);return 0;
}

现在让我们来看看函数式编程:

函数式编程

函数式编程的重点在于数组,简单用该图理解:

简单来说就是,重点关心输入和输出,屏蔽其他因素。

图中的过程是,输入全集数组,输出结果是无数个小数组,输入小数组,输出结果是合并后的全集数组。所以我们需要有两个函数:排序分割函数、合并函数。

分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
分割与交换
合并
合并
合并
合并
合并
合并
合并
合并
全集
左左左子集
左左右子集
左右左子集
左右右子集
右左左子集
右左右子集
右右左子集
右右右子集
排序后的数组

代码如下:

#include <stdio.h>
#include <stdlib.h>//排序的本质是对数组的操作,它将数组划分为一个个的小单元
typedef struct {int* array;int length;
} SubArray;//合并小单元
SubArray concatenate(SubArray left, int pivot, SubArray right) {int* new_array = (int*)malloc((left.length + right.length + 1) * sizeof(int));for (int i = 0; i < left.length; i++) {new_array[i] = left.array[i];}new_array[left.length] = pivot;for (int i = 0; i < right.length; i++) {new_array[left.length + 1 + i] = right.array[i];}return (SubArray){new_array, left.length + right.length + 1};
}SubArray quicksort(int* array, int length) {if (length <= 1) {return (SubArray){array, length};}int pivot = array[0];int* left_array = (int*)malloc(length * sizeof(int));int* right_array = (int*)malloc(length * sizeof(int));int left_size = 0, right_size = 0;for (int i = 1; i < length; i++) {if (array[i] <= pivot) {left_array[left_size++] = array[i];} else {right_array[right_size++] = array[i];}}SubArray left_sorted = quicksort(left_array, left_size);SubArray right_sorted = quicksort(right_array, right_size);SubArray result = concatenate(left_sorted, pivot, right_sorted);free(left_sorted.array);free(right_sorted.array);return result;
}void printArray(int* array, int length) {for (int i = 0; i < length; i++) {printf("%d ", array[i]);}printf("\n");
}int main() {int array[] = {10, 7, 8, 9, 1, 5};int length = sizeof(array) / sizeof(array[0]);SubArray sorted = quicksort(array, length);printf("Sorted array: ");printArray(sorted.array, sorted.length);free(sorted.array);return 0;
}

这篇关于函数式编程思想的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1150480

相关文章

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

Python异步编程中asyncio.gather的并发控制详解

《Python异步编程中asyncio.gather的并发控制详解》在Python异步编程生态中,asyncio.gather是并发任务调度的核心工具,本文将通过实际场景和代码示例,展示如何结合信号量... 目录一、asyncio.gather的原始行为解析二、信号量控制法:给并发装上"节流阀"三、进阶控制

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++中函数模板与类模板的简单使用及区别介绍

《C++中函数模板与类模板的简单使用及区别介绍》这篇文章介绍了C++中的模板机制,包括函数模板和类模板的概念、语法和实际应用,函数模板通过类型参数实现泛型操作,而类模板允许创建可处理多种数据类型的类,... 目录一、函数模板定义语法真实示例二、类模板三、关键区别四、注意事项 ‌在C++中,模板是实现泛型编程

kotlin的函数forEach示例详解

《kotlin的函数forEach示例详解》在Kotlin中,forEach是一个高阶函数,用于遍历集合中的每个元素并对其执行指定的操作,它的核心特点是简洁、函数式,适用于需要遍历集合且无需返回值的场... 目录一、基本用法1️⃣ 遍历集合2️⃣ 遍历数组3️⃣ 遍历 Map二、与 for 循环的区别三、高

C语言字符函数和字符串函数示例详解

《C语言字符函数和字符串函数示例详解》本文详细介绍了C语言中字符分类函数、字符转换函数及字符串操作函数的使用方法,并通过示例代码展示了如何实现这些功能,通过这些内容,读者可以深入理解并掌握C语言中的字... 目录一、字符分类函数二、字符转换函数三、strlen的使用和模拟实现3.1strlen函数3.2st

MySQL中COALESCE函数示例详解

《MySQL中COALESCE函数示例详解》COALESCE是一个功能强大且常用的SQL函数,主要用来处理NULL值和实现灵活的值选择策略,能够使查询逻辑更清晰、简洁,:本文主要介绍MySQL中C... 目录语法示例1. 替换 NULL 值2. 用于字段默认值3. 多列优先级4. 结合聚合函数注意事项总结C

Java8需要知道的4个函数式接口简单教程

《Java8需要知道的4个函数式接口简单教程》:本文主要介绍Java8中引入的函数式接口,包括Consumer、Supplier、Predicate和Function,以及它们的用法和特点,文中... 目录什么是函数是接口?Consumer接口定义核心特点注意事项常见用法1.基本用法2.结合andThen链

MySQL 日期时间格式化函数 DATE_FORMAT() 的使用示例详解

《MySQL日期时间格式化函数DATE_FORMAT()的使用示例详解》`DATE_FORMAT()`是MySQL中用于格式化日期时间的函数,本文详细介绍了其语法、格式化字符串的含义以及常见日期... 目录一、DATE_FORMAT()语法二、格式化字符串详解三、常见日期时间格式组合四、业务场景五、总结一、

golang panic 函数用法示例详解

《golangpanic函数用法示例详解》在Go语言中,panic用于触发不可恢复的错误,终止函数执行并逐层向上触发defer,最终若未被recover捕获,程序会崩溃,recover用于在def... 目录1. panic 的作用2. 基本用法3. recover 的使用规则4. 错误处理建议5. 常见错