如果属性访问,C#Struct方法不会保存值

我需要创建一个看起来像int的结构(但是我需要一个额外的字段……),所以我创建了一个名为TestStruct的新结构,添加了一个我需要的方法(test())并重载了一些运算符,它看起来运作良好……

下面的示例显示了问题。 如果从Val属性执行结构test()方法,则Val属性似乎丢失了值,但是如果在Val2变量上执行该方法,则该值似乎保持正确的值…

为什么会这样?

static class Program { ///  /// The main entry point for the application. ///  [STAThread] static void Main() { new TestClass(); } } public class TestClass { public TestStruct Val { get; set; } private TestStruct Val2; public TestClass() { Val.test(); Console.WriteLine(Val + "-> why is it not 10?"); //Direct assignment works well... Val = 123; Console.WriteLine(Val + "-> direct assingment works.."); //This way works too. Why doesn't it work with "get;" and "set;"? Val2.test(); Console.WriteLine(Val2 + "-> it works this way"); } } public struct TestStruct { private Int32 _Value; public long Offset { get; set; } public static implicit operator TestStruct(Int32 value) { return new TestStruct { _Value = value }; } public static implicit operator Int32(TestStruct value) { return value._Value; } public void test() { _Value = 10; } } 

你的struct是错误的。

由于很多原因,你永远不应该制作一个可变的结构。

就像intDateTime值是不可变的并且永远不会改变一样, struct的特定值也绝不能改变。

相反,您可以创建返回新的不同值的函数。

以下是可变结构是邪恶的一些原因:

  1. http://ericlippert.com/2008/05/14/mutating-readonly-structs/
  2. http://blog.slaks.net/2010/12/when-shouldnt-you-write-ref-this.html
  3. http://codeblog.jonskeet.uk/2010/07/27/iterate-damn-you/
  4. http://philosopherdeveloper.wordpress.com/2011/03/31/how-i-discovered-a-bug-in-the-c-compiler-part-1/

要回答这个问题, Val.test()等同于get_Val().test()
由于结构是值类型,`get_Val()(自动生成的属性getter)返回结构的副本
私有支持字段中的原始结构不受影响。

它是一个结构,所以它是一个值类型 – 它是按值复制的,而不是通过引用复制的。

属性只是方法调用的语法糖,Val编译为:

 private TestStruct _valBackingField; public TestStruct get_Val { return _valBackingField; } 

通过属性访问结构时,每次访问Getter时都会获得一个新的副本。 通过字段访问结构时,每次都会获得相同的对象。

这就是可变价值类型被认为是邪恶的原因。

当您使用Val时,该属性将返回一个副本,它是您修改的副本,然后您再次访问该属性,从而获得另一个副本。

如果将结构更改为类,则此问题应该消失,因为如果处理类和结构的方式不同。

看完一些链接后,我明白了问题..

我通过改变test()方法来修复它,以返回一个带有新值的新结构,然后代替Val.test()我做Val = Val.test();

我认为当我们想要替换字符串中的一些字符时会发生类似的情况,我们必须做类似str = str.replace("xx","yy");

谢谢大家帮帮我。 我没想到会如此迅速地得到所有这些回复