查看: 48|回复: 0

[教程] C# params基本语法及典型用法

[复制链接]

1

主题

0

回帖

0

积分

积极分子

金币
0
阅读权限
220
精华
0
威望
0
贡献
0
在线时间
0 小时
注册时间
2009-6-15
发表于 2025-12-16 11:10:03 | 显示全部楼层 |阅读模式

在 C# 中,params 关键字用于定义**可变参数列表(variable-length argument list)**的方法参数。它允许调用者传入 0 个或多个指定类型的参数,而无需显式创建数组。

你提到的 params Type[] interfaceTypes 是一个典型的使用场景:方法接收任意数量的 Type 对象(通常表示接口类型),用于反射、依赖注入、插件系统等。

一、params基本语法

public void MyMethod(params int[] numbers)
{
    foreach (int n in numbers)
        Console.WriteLine(n);
}
// 调用方式:
MyMethod();           // numbers = new int[0]
MyMethod(1);          // numbers = new int[] { 1 }
MyMethod(1, 2, 3);    // numbers = new int[] { 1, 2, 3 }

规则

  • params 必须是方法的最后一个参数
  • 一个方法只能有一个 params 参数。
  • 调用时可以直接传多个值,也可以传一个数组。

二、params Type[] interfaceTypes的典型用法

场景:检查某个类型是否实现了指定的一组接口

public static bool ImplementsAllInterfaces(Type targetType, params Type[] interfaceTypes)
{
    if (interfaceTypes == null || interfaceTypes.Length == 0)
        return true; // 没有要求接口,视为满足
    var implementedInterfaces = targetType.GetInterfaces();
    foreach (var iface in interfaceTypes)
    {
        if (!implementedInterfaces.Contains(iface))
            return false;
    }
    return true;
}

调用示例:

// 定义接口和类
public interface IRunnable { }
public interface IFlyable { }
public class Bird : IRunnable, IFlyable { }
// 使用
Type birdType = typeof(Bird);
// 方式1:直接传多个 Type
bool result1 = ImplementsAllInterfaces(birdType, typeof(IRunnable), typeof(IFlyable));
// 方式2:传数组(等效)
Type[] required = { typeof(IRunnable), typeof(IFlyable) };
bool result2 = ImplementsAllInterfaces(birdType, required);
// 方式3:不传(空参数)
bool result3 = ImplementsAllInterfaces(birdType); // 返回 true

三、其他常见用途

1. 动态创建实现多个接口的代理(如 Castle DynamicProxy)

proxyGenerator.CreateClassProxy(
    typeof(MyClass),
    new[] { typeof(IInterceptor) },
    params Type[] additionalInterfacesToProxy // ← 这里常用 params
);

2. 注册服务时指定多个接口

public void RegisterService(Type implementation, params Type[] serviceTypes)
{
    foreach (var service in serviceTypes)
    {
        container.Register(service, implementation);
    }
}
// 调用
RegisterService(typeof(Logger), typeof(ILogger), typeof(IDisposable));

3. 断言对象是否实现某些接口(单元测试)

public void AssertImplements(object obj, params Type[] expectedInterfaces)
{
    Type actualType = obj.GetType();
    foreach (var iface in expectedInterfaces)
    {
        Assert.IsTrue(actualType.GetInterfaces().Contains(iface));
    }
}

四、注意事项

❗ 1.params参数可以为null

MyMethod(null); // 此时 params 数组为 null!

因此在方法内部应做空值检查:

public void Foo(params string[] args)
{
    if (args == null) 
    {
        // 处理 null 情况
    }
}

❗ 2. 类型安全

params Type[] 要求传入的每个参数必须是 Type 类型(通常是 typeof(接口)),不能传接口实例。

✅ 正确:

Check(typeof(ISerializable), typeof(IDisposable));

❌ 错误:

ISerializable obj = ...;
Check(obj); // 编译错误!obj 不是 Type 类型

❗ 3. 性能

每次调用会隐式创建数组(除非传入已有数组),高频调用需注意分配开销。

五、完整示例:通用接口验证工具

using System;
using System.Linq;
public static class InterfaceChecker
{
    public static bool HasAllInterfaces(Type type, params Type[] requiredInterfaces)
    {
        if (requiredInterfaces == null || requiredInterfaces.Length == 0)
            return true;
        var implemented = type.GetInterfaces();
        return requiredInterfaces.All(implemented.Contains);
    }
}
// 测试
interface IA { }
interface IB { }
class MyClass : IA, IB { }
class Program
{
    static void Main()
    {
        bool ok = InterfaceChecker.HasAllInterfaces(
            typeof(MyClass),
            typeof(IA),
            typeof(IB)
        );
        Console.WriteLine(ok); // True
    }
}

总结

  • params Type[] interfaceTypes 是一种灵活接收多个接口类型的写法。
  • 常用于反射、依赖注入、AOP、插件架构等需要动态处理类型的场景。
  • 调用简洁,但需注意 null、性能和类型安全。
  • 它让 API 更友好:用户无需手动构造数组。

到此这篇关于C# params基本语法及典型用法的文章就介绍到这了,更多相关c# params使用内容请搜索琼殿技术社区以前的文章或继续浏览下面的相关文章希望大家以后多多支持琼殿技术社区!

您可能感兴趣的文章:
  • C# params可变参数的使用注意详析
  • C#中Params的用法
  • c#的params参数使用示例
  • C# 运用params修饰符来实现变长参数传递的方法
  • c# 可变数目参数params实例
  • 用C#中的params关键字实现方法形参个数可变
  • 用C#的params关键字实现方法形参个数可变示例
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

相关侵权、举报、投诉及建议等,请发 E-mail:qiongdian@foxmail.com

Powered by Discuz! X5.0 © 2001-2026 Discuz! Team.

在本版发帖返回顶部