声明抽象泛型类型变量

Fla*_*980 5 c# generics abstract-class

我的问题如下.我为一个显然不起作用的家庭项目做了代码设计.也许你可以帮我弄清楚"代码味道"来自哪里.

好吧,让我们开始:我已经定义了一些类来包装不同类型的存档类型:

public abstract class Archive { }
public class ZipArchive : Archive { }
public class TarArchive : Archive { }
Run Code Online (Sandbox Code Playgroud)

为了处理这些档案,我定义了Manager类.一个定义所需行为的抽象概念,

public abstract class ArchiveManager<T> where T : Archive
{ 
    public abstract void OpenArchive(T archive);
}
Run Code Online (Sandbox Code Playgroud)

具体实现特定的behaiour的具体的:

public class ZipArchiveManager : ArchiveManager<ZipArchive>
{
    public override void OpenArchive(ZipArchive archive) {  /* .. */ }
}

public class TarArchiveManager : ArchiveManager<TarArchive>
{
    public override void OpenArchive(TarArchive archive) {  /* .. */ }
}
Run Code Online (Sandbox Code Playgroud)

现在发生的事情是,在编译期间,我不知道我将处理哪种档案,所以我尝试了以下方法:

class Program
{
    static void Main(string[] args)
    {
        ArchiveManager<Archive> archiveManager = null;

        if (/*some condition*/) {            
            archiveManager = new ZipArchiveManager();
        }
        else {
            archiveManager = new TarArchiveManager();
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

最终出现以下错误:

无法将类型'ZipArchiveManager'隐式转换为'ArchiveManager'

据我所知,泛型参数不能被隐含地转换.有没有办法解决这个问题?这个代码/设计"闻"了吗?

非常感谢你提前.

Mic*_*ael 2

您可以使用逆变接口来代替不实现任何功能的抽象类。在这种情况下,您只能使用类型参数作为方法的返回值,而不能作为参数:

public interface IArchiveManager<out T>
    where T : Archive
{
    T OpenArchive(Stream stream);
}
Run Code Online (Sandbox Code Playgroud)

然后,只需在管理器类中实现该接口:

public class ZipArchiveManager : IArchiveManager<ZipArchive>
{
    public ZipArchive OpenArchive(Stream stream)
    {
        // ...
    }
}

public class TarArchiveManager : IArchiveManager<TarArchive>
{
    public TarArchive OpenArchive(Stream stream)
    {
        // ...
    }
}
Run Code Online (Sandbox Code Playgroud)