C++系列-谓词predicate

谓词predicate


赵师秀 宋《约客》

黄梅时节家家雨,青草池塘处处蛙。

有约不来过夜半,闲敲棋子落灯花


💢什么是谓词

  • 🥝谓词是一个可以调用的对象
  • 🥝它接收一个或一个或多个参数,并返回一个或布尔值
  • 🥝谓词通常用于算法中 ,以指定特定的条件,例如在排序算法中,可以用谓词指定排序的规则,在查找算法中,可以使用谓词来确定是否找到了满足特定条件的元素。

💢💢函数(function)谓词

  • 🍋函数的定义满足谓词的定义条件。
  • 🍋在使用时通过传递函数名实现 。

👉👉👉

先看find_if函数的定义,返回_First,是一个迭代器对象。

_Pred(_UFirst),_Pred是执行实参中传入的参数,在本类中,是一个函数名,函数名()是函数的调用。 _UFirst是迭代器指向的内容。

在for循环中,循环从_First到_Last迭代器指向的每一个元素,将其作为函数的参数,执行函数操作,当出现返回true的时候,退出循环,即找到第一个满足条件的元素就退出。

vector::iterator it = find_if(vec1.begin(), vec1.end(), is_even)中的is_even要传递函数名称。是一个谓词。

cpp 复制代码
_EXPORT_STD template <class _InIt, class _Pr>
_NODISCARD _CONSTEXPR20 _InIt find_if(_InIt _First, const _InIt _Last, _Pr _Pred) 
{ // find first satisfying _Pred
    _Adl_verify_range(_First, _Last);
    auto _UFirst      = _Get_unwrapped(_First);
    const auto _ULast = _Get_unwrapped(_Last);
    for (; _UFirst != _ULast; ++_UFirst) {
        if (_Pred(*_UFirst)) {
            break;
        }
    }

    _Seek_wrapped(_First, _UFirst);
    return _First;
}
cpp 复制代码
code:
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;

bool is_even(int &val)
{
	return val % 2 == 0;
}

void test01()
{
	vector<int> vec1{1, 2, 3, 4, 5, 6};
	// find_if(_InIt _First, const _InIt _Last, _Pr _Pred) { // find first satisfying _Pred
	vector<int>::iterator it = find_if(vec1.begin(), vec1.end(), is_even);
	if(it == vec1.end())
	{
		cout << "没有找到" << endl;
	}
	else
	{
		cout << "第一个偶数是: " << *it << endl;
	}
}

int main()
{
	test01();
	system("pause");
	return 0;
}

result:
第一个偶数是: 2

💢💢函数指针(function pointer)谓词

  • 🍓使用函数指针做谓词,与上述函数做谓词基本一致,只是使用函数指针中转了一次。
  • 🍓在使用时通过传递函数指针实现 。
cpp 复制代码
code:
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;

bool greater_than10(int &val)
{
	return val > 10;
}

void test01()
{
	vector<int> vec1{1, 2, 13, 24, 5, 6};
	bool (*p_greater_than10)(int&); // 定义函数指针
	p_greater_than10 = greater_than10;
	vector<int>::iterator it = find_if(vec1.begin(), vec1.end(), greater_than10); // 函数指针作为谓词
	if(it == vec1.end())
	{
		cout << "没有找到" << endl;
	}
	else
	{
		cout << "第一个大于10的数是: " << *it << endl;
	}
}

int main()
{
	test01();
	system("pause");
	return 0;
}

result:
第一个大于10的数是: 13

💢💢函数对象(Function Object)谓词

  • 🍈 函数对象中重载函数的调用(), 使其可以被调用, 作为谓词。
  • 🍈 函数对象使用的本质就是对()进行运算符重载,并且返回类型为bool型,满足谓词的条件。

👉👉👉

在find_if函数中,执行 _Pred(_UFirst),_Pred是执行实参中传入的参数,在本类中,即函数对象,函数对象( _UFirst)即调用重载的()成员函数,实现了重载()函数中执行的语句。

Greater(1),创建匿名函数对象,其中的参数传入,使得在调用()时,有了明确的比较对象。

cpp 复制代码
code:
#include<iostream>
#include<vector>
#include<algorithm>

using namespace std;
class Greater
{
public:
	Greater(int val) :m_base(val) {}
	bool operator()(int val)
	{
		return val > m_base;
	}
private:
	int m_base;
};

int main()
{
	vector<int>vec1 = { 0,-1,1,2,4,9,5,54,45,56 };
	// 函数对象做谓词,并且使用有参构造传入参数作为谓词判断的条件
	vector<int>::iterator iter = find_if(vec1.begin(), vec1.end(), Greater(1));
	cout << *iter << endl;
	system("pause");
	return 0;
}

result:
2

💢💢lambda表达式谓词

👉👉👉

[](int val)->bool{return val > 3;} lambda表达式作参数,这是一个匿名函数,在find_if 的_Pred(*_UFirst)中的_Pred就指向了该匿名函数。

cpp 复制代码
code:
#include<iostream>
#include<vector>
#include<algorithm>
using namespace std;


int main()
{
	vector<int>vec1 = { 0,-1,1,2,4,9,5,54,45,56 };
	// lambda做谓词
	vector<int>::iterator iter = find_if(vec1.begin(), vec1.end(), [](int val)->bool{return val > 3;});
	cout << *iter << endl;
	system("pause");
	return 0;
}

result:
4

库定义的函数对象(Library-Defined Function Object)谓词

💢💢库定义的函数对象谓词

  • 🍉 标准库中已经定义好一些用于特定条件判断的函数对象。
  • 🍉 比如less是用于比较两个对象的大小的函数对象谓词。

💢谓词的应用

  • 🍐谓词通常用于算法中,指定特定的条件。
  • 🍐 例如在排序算法中,可以使用谓词来定义排序的规则。在查找算法中,可以使用谓词来确定是否找到了满足特定条件的元素。

💢一元和二元谓词

  • 🥝一元谓词接受一个参数。
  • 🥝二元谓词接受两个个参数。

一元和二元谓词举例

cpp 复制代码
code:
#include<iostream>
#include<vector>
#include<algorithm>

using namespace std;
class Greater
{
public:
	Greater(int val) :m_base(val) {} // 重载的(), 函数可以接受一个参数,故Greater的对象是二元谓词
	bool operator()(int val)
	{
		return val > m_base;
	}
private:
	int m_base;
};

template<class T>
class MyCompare
{
public:
	bool operator()(T val1, T val2) // 重载的(), 函数可以接受两个参数,故MyCompare的对象是二元谓词
	{
		return val1 > val2;
	}
};

void print_vector(vector<int> &vec)
{
	for (auto i_vec : vec)
	{
		cout << i_vec << " ";
	}
	cout << endl;
}

int main()
{
	vector<int>vec1 = { 0,-1,1,2,4,9,5,54,45,56 };
	cout << "排序前:" << endl;
	print_vector(vec1);
	// sort默认是升序排列,传递函数对象作为参数,改为降序排列
	sort(vec1.begin(), vec1.end(), MyCompare<int>());
	cout << "排序后:" << endl;
	print_vector(vec1);
	vector<int>::iterator iter = find_if(vec1.begin(), vec1.end(), Greater(1));
	cout << *iter << endl;

	system("pause");
	return 0;
}

result:
排序前:
0 -1 1 2 4 9 5 54 45 56
排序后:
56 54 45 9 5 4 2 1 0 -1
56
相关推荐
S-X-S11 分钟前
自定义异常模块
java·开发语言·spring boot
重生之Java开发工程师11 分钟前
Java中Map常用遍历方式以及性能对比
java·开发语言
Xiezequan13 分钟前
c++ 手写queue循环队列
开发语言·c++
计算机小混子18 分钟前
C++实现设计模式--- 观察者模式 (Observer)
c++·观察者模式·设计模式
SomeB1oody18 分钟前
【Rust自学】12.6. 使用TDD(测试驱动开发)开发库功能
开发语言·后端·重构·rust
robin_suli31 分钟前
穷举vs暴搜vs深搜vs回溯vs剪枝系列一>优美的排列
算法·剪枝·深度优先遍历·回溯
比特在路上1 小时前
OJ12:160. 相交链表
c语言·数据结构·算法·链表·1024程序员节
fadtes1 小时前
C++ using(八股总结)
c++
在线OJ的阿川1 小时前
【OJ刷题】同向双指针问题3
c++·学习·算法
比特在路上1 小时前
初阶数据结构【队列及其接口的实现】
c语言·开发语言·数据结构