具有多个类约束的C#泛型类型

时间:2016-01-12 17:04:47

标签: c#

TL; DR

我希望在C#中编译

public SignInResponseMessage Generate(SignInRequestMessage request, (X509Certificate2 || WindowsPrincipal) principal, Uri requestUri) { SignInResponseMessage response = null; ClaimsIdentity identity = null; if (principal != null) { identity = CreateSubject(principal); response = Generate(request, requestUri, identity); } else { throw new ArgumentNullException("principal"); } return response; }

这可能吗?

完整背景

除了一个参数外,我有两个相同的方法。

DRY

我目前正在复制这种方法,它让我内心畏缩,因为我真的想做这个WindowsPrincipal - 呃。环顾四周,this documentation似乎很有希望,但它只允许我添加一个类约束。我在第二节课上得到以下错误:

  

错误1类类型约束“class2”必须先于任何其他约束

如果X509Certificate2Invoke-WebRequest -Headers $headerParams -Uri "https://graph.windows.net/$tenantdomain/reports/auditEvents?api-version=beta&`$filter=eventTime%20gt%202015-12-01T08:00:00Z%20and%20eventTime%20lt%202015-12-04T08:00:00Z" 是我编写的两个类,我可以很容易地让它们实现相同的界面,我会很高兴,但这不是一个选项。

有没有办法完成我想做的事情?

如果没有,我想更多地了解导致这种不可能的潜在机制

2 个答案:

答案 0 :(得分:7)

我担心这可能会导致出问题的方法来确定实际调用的方法。想象一下,如果其中一个类继承自另一个类,并且该方法有一个覆盖!?

Please see the "Diamond Problem" for a complete description of the reasons

如果你想解决它。您可以使用适配器设置公共共享接口,然后使用它。

interface IAdaper {
    SomeMethod();
}

class AdapterOne : IAdapter {
    TypeOneToCall  _one;

    public AdapterOne (TypeOneToCall one) {
        _one = one;
    }

    public SomeMethod() {
        return _one.SomeMethod();
    }
}

class AdapterTwo : IAdapter {
    TypeTwoToCall _two;

    public AdapterTwo (TypeTwoToCall two) {
        _two = two;
    }

    public SomeMethod() {
        return _two.SomeMethod();
    }
}

class Generic<T> where T : IAdapter {

    // Your implementation here.
}

答案 1 :(得分:6)

如果将方法作为参数传递,则T可以是任何值:

  public SignInResponseMessage Generate<T>(SignInRequestMessage request, 
                                          Func<T, ClaimsIdentity> createSubject, 
                                          T principal, 
                                          Uri requestUri)
  {
      SignInResponseMessage response = null;
      ClaimsIdentity identity = null;

      if (principal != null)
      {
          identity = createSubject(principal);
          response = Generate(request, requestUri, identity);
      }
      else
      {
          throw new ArgumentNullException("principal");
      }

      return response;
  }

所以要重用该方法:

  var r1 = Generate<X509Certificate2>(request, CreateSubject, certificate, uri);
  var r2 = Generate<WindowsPrincipal>(request, CreateSubject, principal, uri);