定义一个Student类。
class Student
{
public int ID;
public string Name;
}
在Main()中调用。
Student stu = new Student();
Console.WriteLine(stu.ID);
Console.WriteLine(stu.Name==null);
这对括号就是在调用Student 实例 的构造器。
但是我们的类定义时:没有构造器。编译器会自动准备一个默认的构造器。
如何判断默认的构造器执行了?我们打印字段的值。默认构造函数会对字段初始化构造。从内存中直接刷0。默认构造器的作用。
如果我们想给字段初始化值,不想让默认构造器赋值,那么就需要自己写自定义构造器。
VS中在类内:输入:ctor。按两次Tab键,即可完成构造器结构。
class Student
{
public int ID;
public string Name;
public Student()
{
this.ID = 1;
this.Name = "No Name";
}
}
需求来了:有时候为了防止我们在声明学生之后,忘了给他编号,忘了给他学号ID赋值。学生必须有学号ID。所以需要强制用户在创建实例的时候必须要输入ID。
此时就需要一个带参数的构造函数。
class Student
{
public int ID;
public string Name;
public Student(int initID, string initName)
{
this.ID = initID;
this.Name = initName;
}
public Student()
{
this.ID = 1;
this.Name = "No Name";
}
}
实例构造函数内存机理:
class Program
{
static void Main(string[] args)
{
Student stu = new Student();
Console.WriteLine(stu.ID);
Console.WriteLine(stu.Name==null);
}
}
class Student
{
public int ID;
public string Name;
public Student()
{
this.ID = 1;
this.Name = "No Name";
}
}
先来看看无参构造实例化对象在内存中的操作。
ID是int类型,结构体类型,在内存中占4个字节。Name是String,是类类型,是引用类型,引用类型的变量占4个字节,里面存储的是:实例的地址。
首先给局部变量stu分配地址。stu是局部变量,分配在栈中。从栈中,分配4个字节给Student引用变量,栈内存地址在分配的时候,是从高地址往低地址分配,直到分配到栈顶为止,就溢出了。
1、分配:stu在栈中分配完之后,要等到new Student为创建对象分配空间,new Student在创建对象时,从堆中找可用空间,分配给实例。此时分析实例需要多大空间。发现类中有一个int字段,一个string字段。都是占用4个字节。总共八个字节空间,将8个字节分配给Student实例。
2、构造:分配给实例之后,此时调用构造器()。构造器将分配的空间分割成两部分,一部分是int类型字段ID的4字节,一部分是string类型字段Name的4字节。
3、初始化:上面提及的【默认构造器】:没有给任何参数,堆中默认刷为0,如果是int类型,那么默认就是0,如果是string类型,那么默认就是null。
再来看看带参构造实例化对象在内存中的操作:二次堆中开辟空间。两次内存转储。
class Program
{
static void Main(string[] args)
{
Student stu = new Student(1, "Mr.Okey");
Console.WriteLine(stu.ID);
Console.WriteLine(stu.Name);
}
}
class Student
{
public int ID;
public string Name;
public Student(int initID, string initName)
{
this.ID = initID;
this.Name = initName;
}
public Student()
{
this.ID = 1;
this.Name = "No Name";
}
}
同上面无参构造相似,不同点在于:我们使用【自定义构造器】带参数,不再采用【默认构造器】。
(1,"Mr.Okey")带参数构造器。
字段ID是int类型,直接将初始化值1在堆中转为二进制存储,注意是从高地址到低地址排。最上面代表最低位置。
字段Name是string类型,因此是引用类型,不能在存储字段的位置保存,在堆中重新开辟空间保存字段的值。字段Name中保存值Mr.Okey的(引用位置)起始位置。
因此引用类型中的引用,二级引用相当于二级指针。