using UnityEngine; using System.Collections; using System.Xml.Linq; using UnityEditor; using System; public class NewBehaviourScript : MonoBehaviour { struct MVec3{ public float x; public float y; public float z; } class CTest{ public MVec3 posx; public MVec3 pos { set; get; }//等同于下面的写法-> /*MVec3 dv = new MVec3 (); public MVec3 pos{ set{ dv = value; } get{ return dv;} }*/ } CTest otest; // Use this for initialization void Start () { otest = new CTest (); otest.pos.x = 10; otest.posx.x = 123; gameObject.transform.position.x = 10; Debug.Log ("ot.pos.x={0}" + otest.posx.x); } // Update is called once per frame void Update () { Vector3 vec3 = gameObject.transform.position; } }
编译时出现如下错误:
可以看到34行和36行都出现了编译错误,而35行则正确编译。原因分析:
C#中,reference类型变量存储在堆上,value类型存储在栈上。pos, posx, position都是值类型,为什么会有不同的编译结果呢。区别在于 pos, position是属性,posx是字段。具体分析如下:
32行:new了一个引用类型CTest的对像otest,这时在堆上为对象分配了一片内存,内存大小为pos的三个float和posx的三个float。
34行:由于pos是一个属性,otest.pos将调用属性的get方法,方法的调用是使用栈来进行的,get方法在栈上分配一个临时变量temp来返回pos的的值。即otest.pos返回了一个分配在栈上的临时变量的地址,而otest.pos.x = 10则是试图对栈上临时变量的X进行赋值。这个操作并非不合法,然而是没有意义的,于是编译器就阻止了我们的这个无意义操作,以避免隐患。同理36行。
明白了这个,就容易理解35行为什么是正确的了。otest.posx是一个正常的取地址操作,表示取出otest所在堆内存中的posx变量的地址,这个地址是对象的堆内存中的。
otest.posx.x = 10则是修改堆内存中posx的x的值。