C++中的仿函数有点难,这篇文章却讲的通俗易懂!

零、小序

仿函数是C++中一个强大的解决问题的手段,对于初学者来说这个词可能比较陌生,但是如果你学习过C++的STL,你可能对这个词就不陌生了,也或许你已经在使用C++中预定义的默认仿函数只是你不知道它而已。下面这篇文章就为你揭开仿函数的神秘面纱,也希望通过这篇文章的学习,能够帮助你提高工作的效率。

一、仿函数介绍

1、仿函数概念

仿函数是定义了一个含有operator()成员函数的对象,可以视为一个一般的函数,只不过这个函数功能是在一个类中的运算符operator()中实现,是一个函数对象,它将函数作为参数传递的方式来使用。

2、仿函数的优缺点

优点:
1)仿函数比函数指针的执行速度快,函数指针时通过地址调用,而仿函数是对运算符operator进行自定义来提高调用的效率。
2)仿函数比一般函数灵活,可以同时拥有两个不同的状态实体,一般函数不具备此种功能。
3)仿函数可以作为模板参数使用,因为每个仿函数都拥有自己的类型。
缺点:
1)需要单独实现一个类。
2)定义形式比较复杂。

3、仿函数作用

仿函数通常有下面四个作用:
1)作为排序规则,在一些特殊情况下排序是不能直接使用运算符<或者>时,可以使用仿函数。
2)作为判别式使用,即返回值为bool类型。
3)同时拥有多种内部状态,比如返回一个值得同时并累加。
4)作为算法for_each的返回值使用。

二、仿函数使用代码示例

1、仿函数作为排序规则示例

// functionObject.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//

#include <iostream>
#include <string>
#include <set>

using namespace std;

// 学生的名字和分数
class Student
{
public:
    string getName()
    {
        return m_name;
    };

    int getScore()
    {
        return m_score;
    }

public:
    string m_name;
    int m_score;;
};

// 根据学生的名字进行排序
class StudentSortRule
{
public:
    bool operator()(Student S1, Student S2)
    {
        return (S1.getName() < S2.getName());
    };
};

int main()
{
    cout << "----------------仿函数作为排序规则使用--------------" << endl;
    typedef set<Student, StudentSortRule> StudentSet;
    StudentSet stuSet;
    stuSet.clear();
    Student stuJack;
    stuJack.m_name = "Jack";
    stuJack.m_score = 80;

    Student stuToby;
    stuToby.m_name = "Toby";
    stuToby.m_score = 90;

    Student stuISmileLi;
    stuISmileLi.m_name = "ISmileLi";
    stuISmileLi.m_score = 100;

    // 插入数值
    stuSet.insert(stuJack);
    stuSet.insert(stuToby);
    stuSet.insert(stuISmileLi);

    // 打印查看
    StudentSet::iterator iter;
    for (iter=stuSet.begin(); iter!=stuSet.end(); iter++)
    {
        cout << "Name: " << (*iter).m_name << "Score: " << (*iter).m_score << endl;
    }

    std::cout << "Hello World!n";
    getchar();
}

运行结果:
在这里插入图片描述

2、作为判别式示例

// functionBool.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//

#include <iostream>
#include <algorithm>
#include <list>
using namespace std;

// 删除符合条件的数值
class RmNum 
{
private:
    int m_num;
    int m_count;

public:
    RmNum(int n) :m_num(n), m_count(0)
    {
    }

    bool operator()(int)
    {    
        return ++m_count == m_num;
    }
};

// 打印
void myPrintf(list<int>& lt)
{
    list<int>::iterator it;
    for (it = lt.begin(); it != lt.end(); it++)
    {
        cout << *it << "  ";
    }
    cout << endl;
}

int main()
{
    cout << "----------------仿函数作为判别式使用--------------" << endl;
    cout << "----------------list中插入数值--------------" << endl;
    list<int> tmpList;
    for (int i = 0; i < 9; i++)
    {
        tmpList.emplace_back(i);
    }
    cout << "----------------打印list中插入的数值--------------" << endl;
    myPrintf(tmpList);

    cout << "----------------删除list中符合条件的数值--------------" << endl;
    list<int>::iterator pos;
    pos = remove_if(tmpList.begin(), tmpList.end(), RmNum(2));
    tmpList.erase(pos, tmpList.end());
    cout << "----------------打印删除后list中的数值--------------" << endl;
    myPrintf(tmpList);
    std::cout << "Hello World!n";
    getchar();
}

运行结果:
在这里插入图片描述

3、拥有内部状态示例

// functionState.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//

#include <iostream>
#include <algorithm>
#include <list>

using namespace std;



class mySequence
{
private:
    int m_value;

public:
    mySequence(int ivalue) :m_value(ivalue)
    {
    }

    int operator()()
    {
        return m_value++;
    }
};

void myPrintf(list<int>& lt)
{
    list<int>::iterator it;
    for (it = lt.begin(); it != lt.end(); it++)
    {
        cout << *it << " , ";
    }
    cout << endl;
}

int main()
{
    cout << "----------------仿函数拥有内部状态--------------" << endl;
    cout << "----------------list中插入数值--------------" << endl;
    list<int> tmpList;
    cout << "----------------从1开始序列化list--------------" << endl;
    generate_n(back_inserter(tmpList), 9, mySequence(1));
    myPrintf(tmpList);
    cout << "----------------从10开始序列化list--------------" << endl;
    generate(tmpList.begin(), tmpList.end(), mySequence(10));
    myPrintf(tmpList);
    std::cout << "Hello World!n";
    getchar();
}

运行结果:
在这里插入图片描述

4、作为算法for_each的返回值示例

// functionForEach.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//

#include <iostream>
#include <algorithm>
#include <vector>

using namespace std;

class AverageValue 
{
private:
    int m_num;
    int m_sum;

public:
    AverageValue() :m_num(0), m_sum(0)
    {

    }

    void operator()(int elem)
    {
        m_num++;
        m_sum += elem;
    }

    double value()
    {
        return static_cast<double>(m_sum*1.0 / m_num);
    }
};

void myPrintf(vector<int>& v)
{
    vector<int>::iterator it;
    for (it = v.begin(); it != v.end(); it++)
    {
        cout << *it << " , ";
    }
    cout << endl;
}

int main()
{
    cout << "----------------仿函数作为算法for_each的返回值使用--------------" << endl;

    vector<int> tmpVector;
    cout << "----------------vector中插入数值--------------" << endl;
    for (int i=0; i<10; i++)
    {
        tmpVector.emplace_back(i);
    }
    cout << "----------------打印vector中插入的数值--------------" << endl;
    myPrintf(tmpVector);

    cout << "----------------打印vector中插入的数值--------------" << endl;
    AverageValue averValue = for_each(tmpVector.begin(), tmpVector.end(), AverageValue());
    cout << "AverageValue: " << averValue.value() << endl;

    std::cout << "Hello World!n";
    getchar();
}

运行结果:
在这里插入图片描述
能力有限,如有错误,多多指教!觉得有用,点赞鼓励一下吧!

原文链接: https://www.cnblogs.com/ISmileLi/p/12699732.html

欢迎关注

微信关注下方公众号,第一时间获取干货硬货;公众号内回复【pdf】免费获取数百本计算机经典书籍;

也有高质量的技术群,里面有嵌入式、搜广推等BAT大佬

    C++中的仿函数有点难,这篇文章却讲的通俗易懂!

原创文章受到原创版权保护。转载请注明出处:https://www.ccppcoding.com/archives/337672

非原创文章文中已经注明原地址,如有侵权,联系删除

关注公众号【高性能架构探索】,第一时间获取最新文章

转载文章受原作者版权保护。转载请注明原作者出处!

(0)
上一篇 2023年3月1日 下午11:14
下一篇 2023年3月1日 下午11:15

相关推荐