-5

我正在挖掘一些代码,突然出现了一个野生错误......

好的,让我们开始吧。我创建此代码以在解析时表示脚本中的位置。

/// <summary>
/// Represents a position in script. (Used for error reporting)
/// </summary>
public sealed class Position
{
    /// <summary>
    /// Line count.
    /// </summary>
    public int Line { get; set; }
    /// <summary>
    /// Character count.
    /// </summary>
    public int Char { get; set; }
    /// <summary>
    /// Index data.
    /// </summary>
    public int Index { get; set; }

    /// <summary>
    /// Initialize new Position object to standard values.
    /// </summary>
    public Position()
    {
        this.Line = 1;
        this.Char = 1;
        this.Index = 0;
    }
    /// <summary>
    /// Copy a Position object.
    /// </summary>
    /// <param name="pos"></param>
    public Position(Position pos)
    {
        this.Line = pos.Line;
        this.Char = pos.Char;
        this.Index = pos.Index;
    }
    /// <summary>
    /// Initialize new Position object to given parameters.
    /// </summary>
    /// <param name="p_index">The index in stream.</param>
    /// <param name="p_line">The line count.</param>
    /// <param name="p_char">The character count</param>
    public Position(int p_index, int p_line, int p_char)
    {
        this.Line = p_line;
        this.Char = p_char;
        this.Index = p_index;
    }

    /// <summary>
    /// Check if 2 Position objects are equal.
    /// </summary>
    /// <param name="p1">Left operand.</param>
    /// <param name="p2">Right operand.</param>
    /// <returns>Returns true, if both position objects are equal.</returns>
    public static Boolean operator ==(Position p1, Position p2)
    {
        return
            p1.Index == p2.Index &&
            p1.Char == p2.Char &&
            p1.Line == p2.Line;
    }

    /// <summary>
    /// Check if 2 Position objects are not equal.
    /// </summary>
    /// <param name="p1">Left operand.</param>
    /// <param name="p2">Right operand.</param>
    /// <returns></returns>
    public static Boolean operator !=(Position p1, Position p2)
    {
        return !(p1 == p2);
    }

    public override int GetHashCode()
    {
        return base.GetHashCode();
    }

    /// <summary>
    /// Equals overload.
    /// </summary>
    /// <param name="obj"></param>
    /// <returns></returns>
    public override bool Equals(object obj)
    {
        if (obj is Position)
            return this == (Position)obj;
        return false;
    }

    /// <summary>
    /// ToString override.
    /// </summary>
    /// <returns></returns>
    public override string ToString()
    {
        return String.Format("{0} - {1} - {2}", this.Index, this.Line, this.Char);
    }
}

但是......当我做这样的事情时:

var pos1 = new Position();
var pos2 = pos1;
pos2.Char = 10;

然后我也改变了 pos1.Char 的值......我不知道 C# 的这种行为。其他类的行为相同。

复制构造函数没有帮助。

我使用 .NET 4.5 和 VS 2012 Update 3...

有人能告诉我是什么导致了这种行为吗?或者至少如何解决这种行为......

4

3 回答 3

3

pos1是对新对象的引用。通过设置pos2pos1,您现在拥有对同一对象的两个引用。如果你想要两个不同的对象,你应该这样做

var pos1 = new Position();  // Create a new object
var pos2 = new Position(pos1);  // Use your copy constructor
pos2.Char = 10;
于 2013-07-01T13:43:36.127 回答
1

您的“位置”类是“参考类型”当您将 pos2 等于 pos1 时,它指向相同的内存位置。

因此,改变一个对象的属性也会改变另一个对象,因为它是同一个对象

于 2013-07-01T13:43:14.607 回答
1

这不是错误;这是正确的 C# 行为。Position是 a class,它是一个引用类型。该分配var pos2 = pos1;不会复制任何内容,它只是创建对同一Position实例的另一个引用。

你想像这样调用你的复制构造函数:

var pos2 = new Position(pos1);
于 2013-07-01T13:43:22.873 回答