C#如何创建类的特殊实例?

时间:2013-06-24 23:03:39

标签: c# class instance named

对于某些类,理想情况下,我想创建特殊的命名实例,类似于“null”。据我所知,这是不可能的,所以相反,我使用静态构造函数创建类的静态实例,类似于:

public class Person
{
    public static Person Waldo;  // a special well-known instance of Person
    public string name;
    static Person()  // static constructor
    {
        Waldo = new Person("Waldo");
    }
    public Person(string name)
    {
        this.name = name;
    }
}

正如您所看到的,Person.Waldo是Person类的一个特殊实例,我创建它是因为在我的程序中,还有很多其他类可能想要引用这个特殊的着名实例。

实现这种方式的缺点是我不知道如何使Person.Waldo的所有属性都是不可变的,而“普通”Person实例的所有属性都应该是可变的。每当我不小心有一个Person对象指向Waldo时,我不小心不检查它是否指的是Waldo,那么我不小心破坏了Waldo的属性。

是否有更好的方法,甚至是一些其他替代方法来定义类的特殊知名实例?

我现在知道的唯一解决方案是实现get& amp;设置访问器,并在每一组上检查“if(this == Waldo)throw new ...”。虽然这有效,但我认为C#可以比我实现它做得更好。如果只有我能找到一些C#方式来使得Waldo的所有属性只读(除非在静态构造函数中。)

4 个答案:

答案 0 :(得分:4)

在继承Person ImmutablePerson : Person

的Person中创建一个私有类

使所有属性设置器都被锁定:例如,使用NotImplementedException覆盖它们。

您的静态Person初始化变为: public static readonly Person Waldo = new ImmutablePerson("Waldo");

也可以删除静态构造函数。

答案 1 :(得分:1)

也许你可以拥有以下层次结构:

class Person
{
    protected string _name;
    public virtual string Name{
        get{
            return _name;
        }
    }
}

class EditablePerson:Person
{
    public new string Name{
        get{
            return _name;
        }
        set{
            _name=value;
        }
    }
    public Person AsPerson()
    {
        //either return this (and simply constrain by interface)
        //or create an immutable copy
    }
}

答案 2 :(得分:0)

在我看来,最好是在Waldo上总是返回一个新实例。这样原来的Waldo永远不会改变。但这会阻止您使用引用相等进行比较,因此您必须重写Equals和GetHashCode。

答案 3 :(得分:0)

感谢您的所有建议 - 以下是解决方案。我需要创建字符串虚拟,覆盖公共派生类中的访问器,并使用bool标志来允许修改。

重要的是要注意,“name”是一个引用类型,虽然我已经阻止了更改“name”所指的内容,但它是否是字符串以外的内容,例如包含自我修改的类方法,尽管已经阻止修改对象引用,仍然可以修改对象的内容。

class Program
{
    static void Main(string[] args)
    {
        Person myPerson = new Person("Wenda");
        System.Console.WriteLine("myPerson is " + myPerson.name);       // Prints "myPerson is Wenda"

        if (myPerson == Person.Waldo)
            System.Console.WriteLine("Found Waldo (first attempt)");    // doesn't happen
        else
            System.Console.WriteLine("Still trying to find Waldo...");  // Prints "Still trying to find Waldo..."

        myPerson.name = "Bozo";
        System.Console.WriteLine("myPerson is now " + myPerson.name);   // Prints "myPerson is now Bozo"

        myPerson = Person.Waldo;

        if (myPerson == Person.Waldo)
            System.Console.WriteLine("Found Waldo (second attempt)");   // Prints "Found Waldo (second attempt)"

        System.Console.WriteLine("myPerson is " + myPerson.name);       // Prints "myPerson is Waldo"
        System.Console.WriteLine("Now changing to The Joker...");       // Prints "Now changing to The Joker"
        try
        {
            myPerson.name = "The Joker";                                // throws ImmutablePersonModificationAttemptException
        }
        catch (ImmutablePersonModificationAttemptException)
        {
            System.Console.WriteLine("Failed to change");               // Prints "Failed to change"
        }
        System.Console.WriteLine("myPerson is now " + myPerson.name);   // Prints "myPerson is now Waldo"

        Thread.Sleep(int.MaxValue); // keep the console alive long enough for me to see the result.
    }
}
public class Person
{
    public static readonly ImmutablePerson Waldo = new ImmutablePerson("Waldo");
    public virtual string name { get; set; }
    public Person() // empty base constructor required by ImmutablePerson(string) constructor
    { }
    public Person(string name)
    {
        this.name = name;
    }
}
public class ImmutablePersonModificationAttemptException : Exception
{ }
public class ImmutablePerson : Person
{
    private bool allowMutation;
    protected string _name;
    public override string name
    {
        get
        {
            return _name;
        }
        set
        {
            if (allowMutation)
                _name = value;
            else
                throw new ImmutablePersonModificationAttemptException();
        }
    }
    public ImmutablePerson(string name)
        : base()
    {
        allowMutation = true;
        this.name = name;
        allowMutation = false;
    }
}