函数式编程思想

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

相关文章

Python itertools中accumulate函数用法及使用运用详细讲解

《Pythonitertools中accumulate函数用法及使用运用详细讲解》:本文主要介绍Python的itertools库中的accumulate函数,该函数可以计算累积和或通过指定函数... 目录1.1前言:1.2定义:1.3衍生用法:1.3Leetcode的实际运用:总结 1.1前言:本文将详

轻松上手MYSQL之JSON函数实现高效数据查询与操作

《轻松上手MYSQL之JSON函数实现高效数据查询与操作》:本文主要介绍轻松上手MYSQL之JSON函数实现高效数据查询与操作的相关资料,MySQL提供了多个JSON函数,用于处理和查询JSON数... 目录一、jsON_EXTRACT 提取指定数据二、JSON_UNQUOTE 取消双引号三、JSON_KE

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意

C#多线程编程中导致死锁的常见陷阱和避免方法

《C#多线程编程中导致死锁的常见陷阱和避免方法》在C#多线程编程中,死锁(Deadlock)是一种常见的、令人头疼的错误,死锁通常发生在多个线程试图获取多个资源的锁时,导致相互等待对方释放资源,最终形... 目录引言1. 什么是死锁?死锁的典型条件:2. 导致死锁的常见原因2.1 锁的顺序问题错误示例:不同

Java function函数式接口的使用方法与实例

《Javafunction函数式接口的使用方法与实例》:本文主要介绍Javafunction函数式接口的使用方法与实例,函数式接口如一支未完成的诗篇,用Lambda表达式作韵脚,将代码的机械美感... 目录引言-当代码遇见诗性一、函数式接口的生物学解构1.1 函数式接口的基因密码1.2 六大核心接口的形态学

PyCharm接入DeepSeek实现AI编程的操作流程

《PyCharm接入DeepSeek实现AI编程的操作流程》DeepSeek是一家专注于人工智能技术研发的公司,致力于开发高性能、低成本的AI模型,接下来,我们把DeepSeek接入到PyCharm中... 目录引言效果演示创建API key在PyCharm中下载Continue插件配置Continue引言

Oracle的to_date()函数详解

《Oracle的to_date()函数详解》Oracle的to_date()函数用于日期格式转换,需要注意Oracle中不区分大小写的MM和mm格式代码,应使用mi代替分钟,此外,Oracle还支持毫... 目录oracle的to_date()函数一.在使用Oracle的to_date函数来做日期转换二.日

C#反射编程之GetConstructor()方法解读

《C#反射编程之GetConstructor()方法解读》C#中Type类的GetConstructor()方法用于获取指定类型的构造函数,该方法有多个重载版本,可以根据不同的参数获取不同特性的构造函... 目录C# GetConstructor()方法有4个重载以GetConstructor(Type[]

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

hdu1496(用hash思想统计数目)

作为一个刚学hash的孩子,感觉这道题目很不错,灵活的运用的数组的下标。 解题步骤:如果用常规方法解,那么时间复杂度为O(n^4),肯定会超时,然后参考了网上的解题方法,将等式分成两个部分,a*x1^2+b*x2^2和c*x3^2+d*x4^2, 各自作为数组的下标,如果两部分相加为0,则满足等式; 代码如下: #include<iostream>#include<algorithm