为什么不在此代码中使用类型推断?

Jac*_*cob 7 c# generics type-inference c#-4.0

假设我有一个如下所示的服务接口:

public interface IFooService
{
    FooResponse Foo(FooRequest request);
}
Run Code Online (Sandbox Code Playgroud)

在调用像这样的服务方法时,我想解决一些贯穿各领域的问题; 例如,我想要统一的请求记录,性能记录和错误处理.我的方法是有一个共同的基础"Repository"类,其中包含一个Invoke方法,该方法负责调用该方法并围绕它执行其他操作.我的基类看起来像这样:

public class RepositoryBase<TService>
{
    private Func<TService> serviceFactory;

    public RepositoryBase(Func<TService> serviceFactory)
    {
        this.serviceFactory = serviceFactory;
    }

    public TResponse Invoke<TRequest, TResponse>(
        Func<TService, Func<TRequest, TResponse>> methodExpr,
        TRequest request)
    {
        // Do cross-cutting code

        var service = this.serviceFactory();
        var method = methodExpr(service);
        return method(request);
    }
}
Run Code Online (Sandbox Code Playgroud)

这很好用.但是,我使代码更清晰的整个目标受到类型推断无法按预期工作这一事实的阻碍.例如,如果我写一个这样的方法:

public class FooRepository : BaseRepository<IFooService>
{
    // ...

    public BarResponse CallFoo(...)
    {
        FooRequest request = ...;
        var response = this.Invoke(svc => svc.Foo, request);
        return response;
    }
}
Run Code Online (Sandbox Code Playgroud)

我得到这个编译错误:

方法的类型参数...不能从用法中推断出来.尝试显式指定类型参数.

显然,我可以通过改变我的调用来修复它:

var response = this.Invoke<FooRequest, FooResponse>(svc => svc.Foo, request);
Run Code Online (Sandbox Code Playgroud)

但我想避免这种情况.有没有办法重做代码,以便我可以利用类型推断?

编辑:

我还要提一下,早期的方法是使用扩展方法; 这种工作的类型推断:

public static class ServiceExtensions
{
    public static TResponse Invoke<TRequest, TResponse>(
        this IService service, 
        Func<TRequest, TResponse> method, 
        TRequest request)
    {
        // Do other stuff
        return method(request);
    }
}

public class Foo
{
    public void SomeMethod()
    {
        IService svc = ...;
        FooRequest request = ...;
        svc.Invoke(svc.Foo, request);
    }
}
Run Code Online (Sandbox Code Playgroud)

Eri*_*ert 12

问题的标题是"为什么不在此代码中使用类型推断?" 我们只是有问题的代码.情景的核心是:

class Bar { }

interface I
{
    int Foo(Bar bar);
}

class C
{
    public static R M<A, R>(A a, Func<I, Func<A, R>> f) 
    { 
        return default(R); 
    }
}
Run Code Online (Sandbox Code Playgroud)

通话网站是

C.M(new Bar(), s => s.Foo);
Run Code Online (Sandbox Code Playgroud)

我们必须确定两个事实:什么是AR?我们还需要了解哪些信息?这new Bar()对应于As=>s.Foo对应于Func<I, Func<A, R>>.

显然,我们可以确定A必须Bar来自第一个事实.显然,我们可以确定s必须这样做I.所以我们现在知道(I s)=>s.Foo对应了Func<I, Func<Bar, R>>.

现在的问题是:我们R可以int通过s.Foo在lambda体中进行重载解析来推断它吗?

可悲的是,答案是否定的.你和我可以做那个推理,但编译器没有.当我们设计类型推理算法时,我们考虑添加这种"多级"lambda/delegate /方法组推理,但是认为它的成本太高而不能给它带来好处.

对不起,你在这里运气不好; 一般而言,在C#方法类型推断中不会进行需要"挖掘"多个功能抽象级别的推论.

为什么这在使用扩展方法时有用呢?

因为扩展方法没有多个级别的功能抽象.扩展方法案例是:

class C
{
    public static R M<A, R>(I i, A a, Func<A, R> f) { ... }
}
Run Code Online (Sandbox Code Playgroud)

有一个电话网站

I i = whatever;
C.M(i, new Bar(), i.Foo);
Run Code Online (Sandbox Code Playgroud)

现在我们有什么信息?我们推断,ABar如前.现在我们必须推断出R知道i.Foo映射的内容Func<Bar, R>.这是一个简单的过载解决问题; 我们假装有一个调用i.Foo(Bar),让重载解析完成它的工作.重载解析回来,说i.Foo(Bar)回报int,因此Rint.

请注意,这种推理 - 涉及方法组 - 旨在添加到C#3,但我搞砸了,我们没有及时完成.我们最终在C#4中添加了这种推断.

另请注意,要使此类推断成功,必须已推断出所有参数类型.我们必须推断只有返回类型,因为为了知道返回类型,我们必须能够做重载决议,以及做重载决议,我们必须知道所有的参数类型.我们不做任何废话,比如"哦,方法组中只有一个方法,所以让我们跳过重载决策,让这个方法自动获胜".