如何实现可返回class或nullable结构的泛型集合查找方法?

Ste*_*eil 6 c# generics

我想要一个异类查找辅助函数用于异构集合:它应该返回一个结构或类,如果找不到该项,则返回null.下面是一个使用普通集合查找的示例,但它可能是数据库调用或其他任何内容.

有没有办法通过单一方法签名实现这一目标?

    public T GetClass<T>(string key)  where T : class
    {
        object o;
        if (Contents.TryGetValue(key, out o))
        {
            return o as T;
        }
        return null;
    }

    public T? GetStruct<T>(string key) where T : struct
    {
        object o;
        if (Contents.TryGetValue(key, out o))
        {
            if (o is T)
            {
                return (T?) o;
            }
        }
        return null;
    }
Run Code Online (Sandbox Code Playgroud)

我已经尝试过的:

  • 我知道通用限制不能用于消除重载的歧义.所以我不能简单地给这两个方法赋予相同的名称.
  • 返回(Default) T不是一个选项,因为0是一个有效的int值.
  • 我试过调用<int ?>类型,但如上所述,Nullable<T>不是引用类型.

有没有办法表明我要返回一个盒装的int?

Jon*_*eet 6

有没有办法通过单一方法签名实现这一目标?

使用可选参数这是一种可怕的(真正可怕的)方式,因此在两种情况下调用代码看起来都一样.但它很蹩脚.

选项:

  • 返回a Tuple<T, bool>而不是使用nullity
  • 使用out参数(如int.TryParse等)
  • 使用不同的方法名称

请注意,通过单独指示缺少值,您可以null生成有效的"找到"结果,这有时可能很有用.或者您可能只想保证它永远不会被退回.

如果你真的想使用nullity,我会选择最后一个选项.我相信它会让你的代码更清晰.IMO,实际上只有当方法使用不同的参数表达完全相同的东西时才会使用重载- 而Nullable<T>在一种情况下T作为返回类型返回,而在另一种情况下作为返回类型则不能真正看到这种方式.


dtb*_*dtb 4

以下方法适用于类和可为 null 的结构:

public static T GetValue<T>(string key)
{
    object o;
    if (Contents.TryGetValue(key, out o))
    {
        if (o is T)
        {
            return (T)o;
        }
    }
    return default(T);
}
Run Code Online (Sandbox Code Playgroud)

用法:

int?   result1 = GetValue<int?>("someInt");
string result2 = GetValue<string>("someString");
Run Code Online (Sandbox Code Playgroud)

请注意,它?是泛型类型参数的一部分,而不是由返回类型上的方法定义的。

  • 我一半喜欢这个 - 但它 *允许* 你将其称为 `GetClass&lt;int&gt;` 并返回 0 来表示“找到值为 0”和“未找到”,无法区分。事实上,您可能会意外地错过示例用法中的第二个“?”并得到一个非常微妙的错误。(“result1”*永远不会*为空,但您会期望它是) (3认同)