一尘不染

如何将对嵌套类成员的访问限制为封闭类?

c#

是否可以指定嵌套类的成员可以被封闭的类访问,但不能访问其他类?

这是问题的说明(当然,我的实际代码要复杂一些……):

public class Journal
{
    public class JournalEntry
    {
        public JournalEntry(object value)
        {
            this.Timestamp = DateTime.Now;
            this.Value = value;
        }

        public DateTime Timestamp { get; private set; }
        public object Value { get; private set; }
    }

    // ...
}

我想防止客户端代码创建的实例JournalEntry,但Journal必须能够创建它们。如果我将构造函数公开,则任何人都可以创建实例…但是如果我将其私有化,Journal将无法!

请注意,JournalEntry该类必须是公共的,因为我希望能够将现有条目公开给客户端代码。

任何建议,将不胜感激!


更新:谢谢大家的输入,我最终选择了IJournalEntry由私有JournalEntry类实现的公共接口(尽管我的问题的最后一个要求是……)


阅读 281

收藏
2020-05-19

共1个答案

一尘不染

如果您的类不太复杂,则可以使用公共可见的接口并将实际的实现类设为私有,也可以为JornalEntry该类创建受保护的构造函数,并使用JornalEntryInstanceJornalEntry公共构造函数派生的私有类,该私有构造函数实际上是通过实例化的你的Journal

public class Journal
{
    public class JournalEntry
    {
        protected JournalEntry(object value)
        {
            this.Timestamp = DateTime.Now;
            this.Value = value;
        }

        public DateTime Timestamp { get; private set; }
        public object Value { get; private set; }
    }

    private class JournalEntryInstance: JournalEntry
    { 
        public JournalEntryInstance(object value): base(value)
        { }
    }
    JournalEntry CreateEntry(object value)
    {
        return new JournalEntryInstance(value);
    }
}

如果您的实际类太复杂而不能执行任何一种操作,并且可以避免构造函数不是完全不可见的,则可以将构造函数设置为内部,以便仅在程序集中可见。

如果那也不可行,则可以始终将构造函数设为私有,并使用反射从日记类中调用它:

typeof(object).GetConstructor(new Type[] { }).Invoke(new Object[] { value });

现在我考虑一下,另一种可能性是在包含类中使用私有委托,该委托类是由内部类设置的

public class Journal
{
    private static Func<object, JournalEntry> EntryFactory;
    public class JournalEntry
    {
        internal static void Initialize()
        {
            Journal.EntryFactory = CreateEntry;
        }
        private static JournalEntry CreateEntry(object value)
        {
            return new JournalEntry(value);
        }
        private JournalEntry(object value)
        {
            this.Timestamp = DateTime.Now;
            this.Value = value;
        }

        public DateTime Timestamp { get; private set; }
        public object Value { get; private set; }
    }

    static Journal()
    {
        JournalEntry.Initialize();
    }

    static JournalEntry CreateEntry(object value)
    {
        return EntryFactory(value);
    }
}

这应该为您提供所需的可见性级别,而无需求助于慢速反射或引入其他类/接口

2020-05-19