C 构造函数
C 语言为类提供的构造函数可自动完成对象的初始化任务,全局对象和静态对象的构造函数在main()函数执行之前就被调用,局部静态对象的构造函数是当程序第一次执行到相应语句时才被调用。然而给出一个外部对象的引用性声明时,并不调用相应的构造函数,因为这个外部对象只是引用在其他地方声明的对象,并没有真正地创建一个对象。
C 的构造函数定义格式为:
class <类名>
{
public:
<类名>(参数表);
//...(还可以声明其它成员函数)
};
<类名>::<函数名>(参数表)
{
//函数体
}
如以下定义是合法的:
class T
{
public:
T(int a=0){ i=a; }//构造函数允许直接写在类定义内,也允许有参数表。
private:
int i;
};
如果一个类中没有定义任何的构造函数,那么编译器只有在以下三种情况,才会提供默认的构造函数:
1、如果类有虚拟成员函数或者虚拟继承父类(即有虚拟基类)时;
2、如果类的基类有构造函数(可以是用户定义的构造函数,或编译器提供的默认构造函数);
3、在类中的所有非静态的对象数据成员,它们对应的类中有构造函数(可以是用户定义的构造函数,或编译器提供的默认构造函数)。
<类名>::<类名>(){},即不执行任何操作。
例子
#includeusingnamespacestd; classtime { public: time()//constructor.构造函数 { hour=0; minute=0; sec=0; } voidset_time(); voidshow_time(); private: inthour,minute,sec; }; intmain() { classtimet1; t1.show_time(); t1.set_time(); t1.show_time(); return0; } voidtime::set_time() { cin>>hour>>minute>>sec; } voidtime::show_time() { cout< 程序运行情况:
0:0:0
10 11 11 回车
10:11:11
任何时候,只要创建类或结构,就会调用它的构造函数。类或结构可能有多个接受不同参数的构造函数。构造函数使得程序员可设置默认值、限制实例化以及编写灵活且便于阅读的代码。
PHP的构造函数
void __construct( [mixed args [, ...]] )
php 5 允行开发者在一个类中定义一个方法作为构造函数。具有构造函数的类会在每次创建对象时先调用此方法,所以非常适合在使用对象之前做一些初始化工作。
注:如果子类中定义了构造函数则不会暗中调用其父类的构造函数。要执行父类的构造函数,需要在子类的构造函数中调用 parent::__construct()。
例子 使用新标准的构造函数
<"InBaseClassconstructor/n";}} classSubClassextendsBaseClass{ function__construct(){ parent::__construct();print"InSubClassconstructor/n";}} $obj=newBaseClass();$obj=newSubClass(); "para" label-module="para">为了实现向后兼容性,如果 php 5 在类中找不到__construct()函数,它就会尝试寻找旧式的构造函数,也就是和类同名的函数。因此唯一会产生兼容性问题的情况是:类中已有一个名为__construct()的方法,但它却又不是构造函数。
Python的构造函数
定义格式为
class <类名>:
__init__(self [,参数表]):
#函数体
#其它的方法和属性
构造函数其他特点
1.它的函数名与类名相同;
2.它可以重载;
3.不能指定返回类型,即使是void也不行;
4.虽然在一般情况下,构造函数不被显式调用,而是在创建对象时自动被调用。但是并不是不能被显式调用。有些时候是一定要显式调用的,只要是父类有带参的构造函数,在子类中就必须显式的调用父类的构造函数,因为子类的构造器在实例化时无法找到父类的构造函数(当父类有自己写的无参构造函数时,子类也不用显式调用)。
void__destruct( void )
php 5 引入了析构函数的概念,这类似于其它面向对象的语言,如 C 。析构函数会在到某个对象的所有引。
构造函数语法
Java, C , C#, ActionScript和PHP 4中的命名规范会要求构造器函数的名称与它所在类的名称相同。
PHP 5 建议的构造器函数名称为__construct。为了保证向下兼容,__construct方法无法找到时会调用任何跟类名同名的方法作为构造器。从 PHP 5.3.3 起,这种途径只对非名字空间的类有效。
在Perl里,构造器被约定俗成地命名为"new",并且会完成创建对象的大量工作。
在 Perl 的 Moose 对象系统中,构造函数(叫做 new)是自动生成的,程序员可以通过指定一个 BUILD 方法来对其进行扩充。
在 Visual Basic .NET里,构造器被命名为New,是个 Sub。
Python里构造器的被分为 __new__ 和 __init__ 两个方法。__new__方法负责为实例分配内储存空间,并接受自身的类作为参数(一般命名为 cls)。__init__方法接受被新建的实例作为参数(一般称为 self)。
Object Pascal 的构造函数用关键字 constructor 标识,并且可以起任意名字(但一般来说会被起名为 Create)。
Objective-C 的构造函数分成两个方法,alloc 和 init。alloc 方法分配内存,init 负责初始化。new 方法会调用 alloc 和 init 两者。
构造函数内存机制
在 Java, C# 和 VB .NET 里,构造器会在一种叫做堆的特殊数据结构里创建作为引用类型的实例。值类型(例如 int, double 等等)则会创建在叫做栈的有序数据结构里。VB .NET and C# 会允许用new来创建值类型的实例。然而在这些语言里,即使使用这种方法创建的对象依然只会在栈里。
在 C 里,不用 new 创建的对象会保存在栈里,使用 new 创建时则会在堆里。它们必须分别使用析构函数或者delete操作才能被删除。
构造函数语言细节
构造函数Java
在Java里,构造器和其他方法的主要差别在于:
构造器不具有任何显性返回类型。
构造器无法被直接“new” invokes them).
构造器无法被标示为synchronized, final, abstract, native, 或者static。
Java 里的构造器会按照以下顺序完成下列工作:
将类变量初始到缺省值。(byte, short, int, long, float, 和 double 变量会默认设为它们相应的0值,booleans 会被设为 false, chars 会被设为空字符(' '),对象引用会被设为 null)
引用父类的构造器,如果没有定义任何构造器。
将实例变量初始化到指定值。
执行构造器内的代码。
在 Java 中可以通过关键词super访问父类的构造器。
publicclassExample{//Definitionoftheconstructor.publicExample(){this(1);}//OverloadingaconstructorpublicExample(intinput){data=input;//Thisisanassignment}//Declarationofinstancevariable(s).privateintdata;}//Codesomewhereelse //Instantiatinganobjectwiththeaboveconstructor Examplee=newExample(42);不接收任何参数的构造器被称作“无参数构造器”。
构造函数Visual Basic .NET
在Visual Basic .NET中, 构造函数以"New"为定义方法,并且必须是个 Sub。
ClassFoobarPrivatestrDataAsString'ConstructorPublicSubNew(ByValsomeParamAsString)strData=someParamEndSubEndClass'codesomewhereelse 'instantiatinganobjectwiththeaboveconstructor DimfooAsNewFoobar(".NET")构造函数C#
publicclassMyClass{privateinta;privatestringb;//ConstructorpublicMyClass():this(42,"string"){}//OverloadingaconstructorpublicMyClass(inta,stringb){this.a=a;this.b=b;}}//Codesomewhere //Instantiatinganobjectwiththeconstructorabove MyClassc=newMyClass(42,"string");C# 静态构造函数
在C#中,静态构造函数是用来初始化任何静态数据。静态构造函数也称为“类构造函数”,由于类构造函数在生成的 MSIL 里名为“.cctor”,因此也被称为“cctor”。
静态构造函数允许复杂的静态变量初始化。
静态构造函数在该类第一次被访问时调用,任何使用该类的操作(无论是调用静态函数、属性还是访问静态变量,还是构造类的实例)都会引发静态构造函数的执行。静态构造函数是线程安全的,并且是单例的。当用在泛型类中时,静态构造函数对于泛型的每个实例化都调用一次。静态变量也同样如此。
publicclassMyClass{privatestaticint_A;//NormalconstructorstaticMyClass(){_A=32;}//StandarddefaultconstructorpublicMyClass(){}}//Codesomewhere //Instantiatinganobjectwiththeconstructorabove //rightbeforetheinstantiation //Thevariablestaticconstructorisexecutedand_Ais32 MyClassc=newMyClass();构造函数C
C 的构造函数以该类的类名为标识,且不写返回值类型也无法返回值 :
classC{public:C(void){...}};构造函数的函数体执行是在各个成员构造完之后才开始,因此要想更改成员的构造方式需要使用成员初始化列表:
classD:publicB{public:D(void):B("Hello,world!"){...}};复制构造函数接受同类对象的左值引用(一般为 const T &)、移动构造函数接受右值引用(一般为 T&&):
classE{public:E(constE&e){...}//CopyconstructorE(E&&e){...}//Moveconstructor};C 中,程序员若未对某类定义构造函数(以及赋值函数、析构函数),编译器在满足条件时会定义相应的函数。
构造函数Ruby
irb(main):001:0>classExample Classirb(main):002:1>definitialize irb(main):003:2>puts"Hellothere" irb(main):004:2>end irb(main):005:1>end=>nil irb(main):006:0>ExampleClass.new Hellothere =>#