2分钟讲清楚C#的委托, C语言的函数指针,Java的函数式接口

很多小伙伴学习C# 的委托时往往一头雾水, 不明白委托是什么, 有什么作用, 今天我就用2分钟讲清楚

这是一个C# 的控制台程序

定义一个最简单的委托 delegate int Calculate(int a, int b);

这相当于定义了一个Calculate类型, 只不过这个类型需要传入2个int类型的参数

返回值也是int

委托的意义在于, 它可以将函数封装成可以像变量 int, string这样的类型传入到另外的函数中, 例如函数int Test(int a, int b, Calculate c)

这大大提高了函数的功能和可扩展性, 可以在不改变Test函数的参数的同时,

通过改变Calculate 来改变Test内部的逻辑关系

csharp 复制代码
namespace DelegateTest
{


    public class Program
    {

        //定义一个Calculate 类型的委托
        delegate int Calculate(int a, int b); 

        static void Main(string[] args)
        {
            //定义一个返回值和参数都与Calculate相同的加法函数
            int Add(int a, int b)
            {
                return a + b;
            }
            //定义一个返回值和参数都与Calculate相同的减法函数
            int Sub(int a, int b)
            {
                return a - b;
            }

            //将委托类型calculate当参数传入函数
            int Test(int a, int b, Calculate calculate)
            {
                return calculate(a, b) + a;
            }

            //委托的精髓在于, 将委托看作一种类型, 和int, String一样的类型.
            //只不过int a = b; 是将 int类型的变量 b 赋值给 a, 
            //而Calculate calculate1 = Sub; 是将Calculate类型的 Sub 赋值给 calculate1
            Calculate calculate1 = Sub; //定义一个Calculate 类型的calculate1字段, 并将Sub函数传递给它
            Calculate calculate2 = Add; //定义一个Calculate 类型的calculate2字段, 并将Add函数传递给它

            int i = Test(1, 2, calculate1); //返回 1 + (1+2) = 4
            int j = Test(1, 2, calculate2); //返回 1+  (1-2) = 0

            Console.WriteLine("i的值是:" + i);
            Console.WriteLine("j的值是:" + j);
        }
    }




}

很多人不知道, C# 的委托, 和C语言的函数指针是一个东西

我们把C sharp的 delegate int Calculate(int a, int b)

换成C语言的 typedef int (*Calculate)(int a, int b)

c 复制代码
// FunctionPoint.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//

#include <stdio.h>

    //定义一个Calculate 类型的委托
  //delegate int Calculate(int a, int b); 
    typedef int (*Calculate)(int a, int b);

    //定义一个返回值和参数都与Calculate相同的加法函数
    int Add(int a, int b)
    {
        return a + b;
    }
    //定义一个返回值和参数都与Calculate相同的减法函数
    int Sub(int a, int b)
    {
        return a - b;
    }

    //将委托类型calculate当参数传入函数
    int Test(int a, int b, Calculate calculate)
    {
        return a + calculate(a, b);
    }

int main()
{
    //委托/函数指针的精髓在于, 将委托看作一种类型, 和int, String一样的类型.
    //只不过int a = b; 是将 int类型的变量 b 赋值给 a, 
    //而Calculate calculate1 = Sub; 是将Calculate类型的 Sub 赋值给 calculate1
    Calculate calculate1 = Add;

    Calculate calculate2 = Sub;

    int i = Test(1, 2, calculate1); //返回 1 + (1+2) = 4
    int j = Test(1, 2, calculate2); //返回 1+  (1-2) = 0

    printf("i的值是: %d\n", i);
    printf("j的值是: %d\n", j);
}

可以运行一下, 这两个程序返回的结果是一样的. 如果放在java 上怎么实现?

函数式接口和委托, 函数指针是一脉相承的!

java 复制代码
package com.example.design;

import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;

@SpringBootTest
class DesignApplicationTests {

    //定义一个Calculate 类型的委托
    //delegate int Calculate(int a, int b);
    //typedef int (*Calculate)(int a, int b);
    @FunctionalInterface
    interface ICalculate{
        Integer Calculate (Integer a, Integer b);
    }

    //定义一个返回值和参数都与Calculate相同的加法函数
    Integer Add(Integer a, Integer b)
    {
        return a + b;
    }
    //定义一个返回值和参数都与Calculate相同的减法函数
    Integer Sub(Integer a, Integer b)
    {
        return a - b;
    }

    //将委托类型calculate当参数传入函数
    Integer Test(Integer a, Integer b, ICalculate iCalculate)
    {
        return a + iCalculate.Calculate(a, b);
    }

    @Test
    void contextLoads() {
        //委托/函数指针的精髓在于, 将委托看作一种类型, 和int, String一样的类型.
        //只不过int a = b; 是将 int类型的变量 b 赋值给 a,
        //而Calculate calculate1 = Sub; 是将Calculate类型的 Sub 赋值给 calculate1
        ICalculate calculate1 = this::Add;
        ICalculate calculate2 = this::Sub;
        Integer i = Test(1,2,calculate1);//返回 1 + (1+2) = 4
        Integer j = Test(1,2,calculate2);//返回 1+  (1-2) = 0

        System.out.println("i的值是:"+i);
        System.out.println("j的值是:"+j);
    }

}
相关推荐
JH30738 小时前
SpringBoot 优雅处理金额格式化:拦截器+自定义注解方案
java·spring boot·spring
HABuo8 小时前
【linux文件系统】磁盘结构&文件系统详谈
linux·运维·服务器·c语言·c++·ubuntu·centos
暖馒8 小时前
Modbus应用层协议的深度剖析
网络·网络协议·c#·wpf·智能硬件
Coder_Boy_9 小时前
技术让开发更轻松的底层矛盾
java·大数据·数据库·人工智能·深度学习
invicinble9 小时前
对tomcat的提供的功能与底层拓扑结构与实现机制的理解
java·tomcat
较真的菜鸟9 小时前
使用ASM和agent监控属性变化
java
黎雁·泠崖9 小时前
【魔法森林冒险】5/14 Allen类(三):任务进度与状态管理
java·开发语言
qq_124987075311 小时前
基于SSM的动物保护系统的设计与实现(源码+论文+部署+安装)
java·数据库·spring boot·毕业设计·ssm·计算机毕业设计
Coder_Boy_11 小时前
基于SpringAI的在线考试系统-考试系统开发流程案例
java·数据库·人工智能·spring boot·后端
Mr_sun.11 小时前
Day06——权限认证-项目集成
java