回调的原理、实现与应用

什么是回调

 函数指针有什么用呢?一个最常用的地方就是回调

 什么回调?维基百科是这样解释的:回调一段可执行的代码通过参数传递给别一段代码,以期望在一个合适的时间调用这个参数(可执行的代码)

从一个需求开始

假设有这么一个需求,有一Person类定义如下:

struct Person

{

    int age;

    float weight;

    float height;

};

   现要对Person的一组对象进行排序,但并没有确定根据什么规则来排序,有时需要根据年龄进行排序,有时需要根据身高进行排序,有时可能是根据身高和体重的综合情况来排序,还有可能……

 

    你可能会想到这样写,定义三个函数分别根据年龄、体重、身高进行排序:

void SortByAge(Person* persons, int count);

void SortByWeight(Person* persons, int count);

void SortByHeight(Person* persons, int count);

    如果要根据身高和体重的综合情况来排序,那你还得再定义一个函数。这样是不是代码冗余且很繁琐?但如果你会用回调,这个问题就会很简单。

 

用回调实现对Person的排序:

 

  1. typedef int (*Compare)(const Person&, const Person&);  
  2.    
  3. //交换两个元素  
  4. void swap(Person* p1, Person *p2)  
  5. {  
  6.     Person p = *p1;  
  7.     *p1 = *p2;  
  8.     *p2 = p;  
  9. }  
  10. //排序(本例中采用冒泡排序)  
  11. void PersonSort(Person* persons, int count, Compare pCompare)  
  12. {  
  13.     for (int i = 0; i < count-1; i ++)  
  14.     {  
  15.         for (int j = 0; j < count - i -1; j++)  
  16.         {  
  17.             if (pCompare(persons[j], persons[j+1]) > 0)  
  18.             {  
  19.                 swap(persons+j, persons+j+1);  
  20.             }  
  21.         }  
  22.     }  
  23. }  
 

    如果你要根据年龄来进行排序,只要实现一个Compare类型的函数,再调用上面的PersonSort函数就可以实现根据年龄排序的功能。如:

  1. int CompareByAge(const Person& p1, const Person& p2)  
  2. {  
  3.     return p1.age - p2.age;  
  4. }  
  5.    
  6. void TestCallBack()  
  7. {  
  8.     //创建Person的一组对象persons,对象中的各个值为0到100的随机数  
  9.     srand((unsigned)time(NULL));   
  10.     Person persons[10];  
  11.     for(int i = 0; i < 10; i ++)  
  12.     {  
  13.         persons[i].age = rand()%100;  
  14.         persons[i].weight = rand()%100;  
  15.         persons[i].height = rand()%100;  
  16.     }  
  17.     //【todo】  
  18.     //根据年龄进行排序。  
  19.     PersonSort(persons, 10, CompareByAge);  
  20.       
  21.     for(int i = 0; i < 10; i ++)  
  22.     {  
  23.         std::cout << persons[i].age << "\t" <<    persons[i].weight << "\t" << persons[i].height << std::endl;  
  24.     }  
  25. }  

说明:以上调用TestCallBack函数时需要包含头文件<iostream><time.h>

 

    这样如果需求发生变更(如要根据每个Person身高和体重的总和来排序),只需求再定义一个Compare类型的函数,而不用再对PersonSort函数做任何改动。如下:

  1. int CompareByHeightWeight(const Person& p1, const Person& p2)  
  2. {  
  3.     return (p1.height + p1.weight) - (p2.height + p2.weight);  
  4. }  
  5.    
  6. void TestCallBack()  
  7. {  
  8.     //创建Person的一组对象persons,对象中的各个值为0到100的随机数  
  9.     srand((unsigned)time(NULL));   
  10.     Person persons[10];  
  11.     for(int i = 0; i < 10; i ++)  
  12.     {  
  13.         persons[i].age = rand()%100;  
  14.         persons[i].weight = rand()%100;  
  15.         persons[i].height = rand()%100;  
  16.     }  
  17.     //【todo】  
  18.     //根据年龄进行排序。  
  19.     PersonSort(persons, 10, CompareByHeightWeight);  
  20.       
  21.     for(int i = 0; i < 10; i ++)  
  22.     {  
  23.         std::cout << persons[i].age << "\t" <<    persons[i].weight << "\t" << persons[i].height << "\t" << persons[i].weight + persons[i].height << std::endl;  
  24.     }  
  25. }  


     C++ STL中的Sort(在<algorithm>头文件中)用的就是这种技术:

template<class RandomAccessIterator>
     void sort(
        RandomAccessIterator first, 
        RandomAccessIterator last
     );
  template<class RandomAccessIterator, class Predicate>
     void sort(
        RandomAccessIterator first, 
        RandomAccessIterator last, 
        Predicate comp
     );
Parameters

first

A random-access iterator addressing the position of the first element in the range to be sorted.

last

A random-access iterator addressing the position one past the final element in the range to be sorted.

comp

User-defined predicate function object that defines the comparison criterion to be satisfied by successive elements in the ordering. This binary predicate takes two arguments and returns true if the two arguments are in order and false otherwise. This comparator function must impose a strict weak ordering on pairs of elements from the sequence. For more information, see Algorithms


  回调函数说白了就是定义一个函数,然后通过参数传递给另一个函数调用。回调不仅是一种技术,更是一种编程思想,上面是通过回调函数来实现的,但它不仅限于回调函数,也可以用其它的技术实现(如面向对象的实现)

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
图像识别技术在病虫害检测中的应用是一个快速发展的领域,它结合了计算机视觉和机器学习算法来自动识别和分类植物上的病虫害。以下是这一技术的一些关键步骤和组成部分: 1. **数据收集**:首先需要收集大量的植物图像数据,这些数据包括健康植物的图像以及受不同病虫害影响的植物图像。 2. **图像预处理**:对收集到的图像进行处理,以提高后续分析的准确性。这可能包括调整亮度、对比度、去噪、裁剪、缩放等。 3. **特征提取**:从图像中提取有助于识别病虫害的特征。这些特征可能包括颜色、纹理、形状、边缘等。 4. **模型训练**:使用机器学习算法(如支持向量机、随机森林、卷积神经网络等)来训练模型。训练过程中,算法会学习如何根据提取的特征来识别不同的病虫害。 5. **模型验证和测试**:在独立的测试集上验证模型的性能,以确保其准确性和泛化能力。 6. **部署和应用**:将训练好的模型部署到实际的病虫害检测系统中,可以是移动应用、网页服务或集成到智能农业设备中。 7. **实时监测**:在实际应用中,系统可以实时接收植物图像,并快速给出病虫害的检测结果。 8. **持续学习**:随着时间的推移,系统可以不断学习新的病虫害样本,以提高其识别能力。 9. **用户界面**:为了方便用户使用,通常会有一个用户友好的界面,显示检测结果,并提供进一步的指导或建议。 这项技术的优势在于它可以快速、准确地识别出病虫害,甚至在早期阶段就能发现问题,从而及时采取措施。此外,它还可以减少对化学农药的依赖,支持可持续农业发展。随着技术的不断进步,图像识别在病虫害检测中的应用将越来越广泛。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值