ホームページ バックエンド開発 PHPチュートリアル PHP の __construct() と __destory() に関する注意事項

PHP の __construct() と __destory() に関する注意事項

May 29, 2020 am 09:47 AM
php

PHP の __construct() と __destory() に関する注意事項

基本的にすべてのプログラミング言語には、クラス内にコンストラクターとデストラクターの概念があります。コンストラクターは関数インスタンスの作成時に初期化作業を行うために使用でき、デストラクターはインスタンスが破棄される前にクリーンアップ作業を行うことができます。比較的言えば、コンストラクターはよく使用されますが、デストラクターは通常、データベース リンク、ファイルの読み取りおよび書き込みハンドルなどのリソースを解放するために使用されます。

コンストラクターとデストラクターの使用法

まず、通常のコンストラクターとデストラクターの使用法を見てみましょう:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

class A

{

    public $name;

    public function __construct($name)

    {

        $this->name = $name;

        echo "A:构造函数被调用,{$this->name}", PHP_EOL;

    }

    public function __destruct()

    {

        echo "A:析构函数被调用,{$this->name}", PHP_EOL;

    }

}

$a = new A('$a');

echo '-----', PHP_EOL;

class B extends A

{

    public function __construct($name)

    {

        $this->name = $name;

        parent::__construct($name);

        echo "B:构造函数被调用,{$this->name}", PHP_EOL;

    }

    public function __destruct()

    {

        parent::__destruct();

        echo "B:析构函数被调用,{$this->name}", PHP_EOL;

    }

}

class C extends A

{

    public function __construct($name)

    {

        $this->name = $name;

        echo "C:构造函数被调用,{$this->name}", PHP_EOL;

    }

    public function __destruct()

    {

        echo "C:析构函数被调用,{$this->name}", PHP_EOL;

    }

}

class D extends A

{

}

// unset($a); // $a的析构提前

// $a = null; // $a的析构提前

$b = new B('$b');

$c = new C('$c');

$d = new D('$d');

echo '-----', PHP_EOL;exit;

// A:构造函数被调用,$a

// -----

// A:构造函数被调用,$b

// B:构造函数被调用,$b

// C:构造函数被调用,$c

// A:构造函数被调用,$d

// -----

// A:析构函数被调用,$d

// C:析构函数被调用,$c

// A:析构函数被调用,$b

// B:析构函数被调用,$b

// A:析构函数被调用,$a

ログイン後にコピー

上記の内容には何かありますか?予想したものと異なるコードはありませんか?問題はありません。1 つずつ見てみましょう。

サブクラスが親クラスのコンストラクターまたはデストラクターをオーバーライドし、親クラスのコンストラクターがparent::__constuct()を使用して明示的に呼び出されない場合、親クラスはクラスのコンストラクターは実行されません。たとえば、C サブクラスがコンストラクターまたはデストラクターをオーバーライドしない場合、親クラスのデストラクターがデフォルトで呼び出されます。変数が明示的に NULL またはデストラクターに設定されていない場合は、 unset() を使用すると、スクリプトの実行が完了してから呼び出されます。テストコード内の呼び出し順序はスタックの先入れ後出しと同様です (C->B->A、C が最初に破棄されます)となりますが、サーバー環境では必ずしもその順序が固定されるわけではありません

デストラクタの参照問題

オブジェクトに次の内容が含まれる場合独自の相互参照がある場合、NULL に設定すると、デストラクターを呼び出すときに unset( ) で問題が発生する可能性があります。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

class E

{

    public $name;

    public $obj;

    public function __destruct()

    {

        echo "E:析构函数被调用," . $this->name, PHP_EOL;

        echo '-----', PHP_EOL;

    }

}

$e1 = new E();

$e1->name = 'e1';

$e2 = new E();

$e2->name = 'e2';

$e1->obj = $e2;

$e2->obj = $e1;

ログイン後にコピー

このコードと同様に、$e1 と $e2 はどちらもクラス E のオブジェクトであり、それぞれ相互への参照を保持しています。実際、簡単に言うと、自分でリファレンスを保持している場合にも同様の問題が発生します。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

$e1 = new E();

$e1->name = 'e1';

$e2 = new E();

$e2->name = 'e2';

$e1->obj = $e2;

$e2->obj = $e1;

$e1 = null;

$e2 = null;

// gc_collect_cycles();

$e3 = new E();

$e3->name = 'e3';

$e4 = new E();

$e4->name = 'e4';

$e3->obj = $e4;

$e4->obj = $e3;

$e3 = null;

$e4 = null;

echo 'E destory', PHP_EOL;

ログイン後にコピー

gc_collect_cycles() 行のコメントをオンにしない場合、デストラクターが実行される順序は次のとおりです。

1

2

3

4

5

6

7

8

9

10

// 不使用gc回收的结果

// E destory

// E:析构函数被调用,e1

// -----

// E:析构函数被调用,e2

// -----

// E:析构函数被调用,e3

// -----

// E:析构函数被调用,e4

// -----

ログイン後にコピー

gc_collect_cycles() のコメントをオンにすると、 )、デストラクターの実行順序は次のとおりです:

1

2

3

4

5

6

7

8

9

10

// 使用gc回收后结果

// E:析构函数被调用,e1

// -----

// E:析构函数被调用,e2

// -----

// E destory

// E:析构函数被调用,e3

// -----

// E:析构函数被调用,e4

// -----

ログイン後にコピー

PHP は gc を使用して一度リサイクルする必要があり、オブジェクトへのすべての参照が解放されるまでクラスのデストラクターは実行されないことがわかります。参照が解放されない場合、デストラクターは実行されません。

低バージョンのコンストラクターとの互換性の問題

PHP5 より前の PHP のコンストラクターは、クラス名と同じ名前のメソッドでした。つまり、クラス F がある場合、関数 F(){} メソッドがそのコンストラクターになります。 PHP では下位バージョンとの互換性を保つためこの機能が残されており、PHP7 以降でもクラス名と同名のメソッドが存在する場合、タイムアウト警告が報告されますが、プログラムの実行には影響しません。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

class F

{

    public function f()

    {

        // Deprecated: Methods with the same name as their class will not be constructors in a future version of PHP; F has a deprecated constructor

        echo "F:这也是构造函数,与类同名,不区分大小写", PHP_EOL;

    }

    // function F(){

    //     // Deprecated: Methods with the same name as their class will not be constructors in a future version of PHP; F has a deprecated constructor

    //     echo "F:这也是构造函数,与类同名", PHP_EOL;

    // }

    // function __construct(){

    //     echo "F:这是构造函数,__construct()", PHP_EOL;

    // }

}

$f = new F();

ログイン後にコピー

__construc() と同じ名前のクラス メソッドが同時に存在する場合、__construct() が最初に使用されます。もう 1 つ注意すべき点は、関数名では大文字と小文字が区別されないため、F() メソッドと f() メソッドは同じであり、コンストラクターになることです。同様に、大文字と小文字が区別されないため、f() と F() は同時に存在できません。もちろん、同じ名前の関数をコンストラクターとして使用することはお勧めしません。結局のところ、これは古い機能であり、いつか廃止される可能性があります。

コンストラクターのオーバーロード

PHP はメソッドのオーバーロードを実行しません。書き換えのみサポートします。つまり、サブクラスが親クラスのメソッドをオーバーライドしますが、複数のメソッドを定義することはできません。同じ名前ですが、パラメータが異なります。 Java などの言語では、メソッドのオーバーロードは、特にクラスをインスタンス化するときに、ポリモーフィック機能を簡単に実装するために非常に便利です。

1

2

3

$r1 = new R(); // 默认构造函数

$r2 = new R('arg1'); // 默认构造函数 一个参数的构造函数重载,arg1

$r3 = new R('arg1', 'arg2'); // 默认构造函数 两个参数的构造函数重载,arg1,arg2

ログイン後にコピー

上記のコードと同様に、複数の __construct() を定義しようとすると、PHP は実行できないことを直接通知します。では、上記のコードの機能を実現する他の方法はあるのでしょうか?もちろんそれはありますが、そうでなければ書きません。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

class R

{

    private $a;

    private $b;

    public function __construct()

    {

        echo '默认构造函数', PHP_EOL;

        $argNums = func_num_args();

        $args = func_get_args();

        if ($argNums == 1) {

            $this->constructA(...$args);

        } elseif ($argNums == 2) {

            $this->constructB(...$args);

        }

    }

    public function constructA($a)

    {

        echo '一个参数的构造函数重载,' . $a, PHP_EOL;

        $this->a = $a;

    }

    public function constructB($a, $b)

    {

        echo '两个参数的构造函数重载,' . $a . ',' . $b, PHP_EOL;

        $this->a = $a;

        $this->b = $b;

    }

}

$r1 = new R(); // 默认构造函数

$r2 = new R('arg1'); // 默认构造函数 一个参数的构造函数重载,arg1

$r3 = new R('arg1', 'arg2'); // 默认构造函数 两个参数的构造函数重载,arg1,arg2

ログイン後にコピー

比較的言えば、Java などの言語よりも面倒ですが、同じ機能を実現します。

コンストラクターとデストラクターのアクセス制限

コンストラクターとデストラクターはデフォルトでパブリックであり、クラス内の他のメソッドのデフォルト値と同じです。もちろん、非公開に設定して保護することもできます。コンストラクターを非パブリックにすると、クラスをインスタンス化できなくなります。これはシングルトン モードで広く使用されているので、シングルトン モードのコードを直接見てみましょう。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

class Singleton

{

    private static $instance;

    public static function getInstance()

    {

        return self::$instance == null ? self::$instance = new Singleton() : self::$instance;

    }

    private function __construct()

    {

    }

}

$s1 = Singleton::getInstance();

$s2 = Singleton::getInstance();

echo $s1 === $s2 ? 's1 === s2' : 's1 !== s2', PHP_EOL;

// $s3 = new Singleton(); // Fatal error: Uncaught Error: Call to private Singleton::__construct() from invalid context

ログイン後にコピー

$s3 をインスタンス化したい場合、エラーが直接報告されます。シングルトン パターンによって外部からのインスタンス化が不可能になる理由については、以前のデザイン パターン システムの記事でシングルトン パターンについて説明しました。

まとめ

普段使っているコンストラクターでこんなにも色々なことができるとは思いませんでした 日々の開発で気を付けなければならないのはサブクラスの継承です. 問題は親クラスのコンストラクタをオーバーライドする際の呼び出し問題と参照時の破壊問題です。

推奨チュートリアル: 「PHP チュートリアル

以上がPHP の __construct() と __destory() に関する注意事項の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

Video Face Swap

Video Face Swap

完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

Ubuntu および Debian 用の PHP 8.4 インストールおよびアップグレード ガイド Ubuntu および Debian 用の PHP 8.4 インストールおよびアップグレード ガイド Dec 24, 2024 pm 04:42 PM

PHP 8.4 では、いくつかの新機能、セキュリティの改善、パフォーマンスの改善が行われ、かなりの量の機能の非推奨と削除が行われています。 このガイドでは、Ubuntu、Debian、またはその派生版に PHP 8.4 をインストールする方法、または PHP 8.4 にアップグレードする方法について説明します。

今まで知らなかったことを後悔している 7 つの PHP 関数 今まで知らなかったことを後悔している 7 つの PHP 関数 Nov 13, 2024 am 09:42 AM

あなたが経験豊富な PHP 開発者であれば、すでにそこにいて、すでにそれを行っていると感じているかもしれません。あなたは、運用を達成するために、かなりの数のアプリケーションを開発し、数百万行のコードをデバッグし、大量のスクリプトを微調整してきました。

PHP 開発用に Visual Studio Code (VS Code) をセットアップする方法 PHP 開発用に Visual Studio Code (VS Code) をセットアップする方法 Dec 20, 2024 am 11:31 AM

Visual Studio Code (VS Code とも呼ばれる) は、すべての主要なオペレーティング システムで利用できる無料のソース コード エディター (統合開発環境 (IDE)) です。 多くのプログラミング言語の拡張機能の大規模なコレクションを備えた VS Code は、

JSON Web Tokens(JWT)とPHP APIでのユースケースを説明してください。 JSON Web Tokens(JWT)とPHP APIでのユースケースを説明してください。 Apr 05, 2025 am 12:04 AM

JWTは、JSONに基づくオープン標準であり、主にアイデンティティ認証と情報交換のために、当事者間で情報を安全に送信するために使用されます。 1。JWTは、ヘッダー、ペイロード、署名の3つの部分で構成されています。 2。JWTの実用的な原則には、JWTの生成、JWTの検証、ペイロードの解析という3つのステップが含まれます。 3. PHPでの認証にJWTを使用する場合、JWTを生成および検証でき、ユーザーの役割と許可情報を高度な使用に含めることができます。 4.一般的なエラーには、署名検証障害、トークンの有効期限、およびペイロードが大きくなります。デバッグスキルには、デバッグツールの使用とロギングが含まれます。 5.パフォーマンスの最適化とベストプラクティスには、適切な署名アルゴリズムの使用、有効期間を合理的に設定することが含まれます。

母音を文字列にカウントするPHPプログラム 母音を文字列にカウントするPHPプログラム Feb 07, 2025 pm 12:12 PM

文字列は、文字、数字、シンボルを含む一連の文字です。このチュートリアルでは、さまざまな方法を使用してPHPの特定の文字列内の母音の数を計算する方法を学びます。英語の母音は、a、e、i、o、u、そしてそれらは大文字または小文字である可能性があります。 母音とは何ですか? 母音は、特定の発音を表すアルファベットのある文字です。大文字と小文字など、英語には5つの母音があります。 a、e、i、o、u 例1 入力:string = "tutorialspoint" 出力:6 説明する 文字列「TutorialSpoint」の母音は、u、o、i、a、o、iです。合計で6元があります

PHPでHTML/XMLを解析および処理するにはどうすればよいですか? PHPでHTML/XMLを解析および処理するにはどうすればよいですか? Feb 07, 2025 am 11:57 AM

このチュートリアルでは、PHPを使用してXMLドキュメントを効率的に処理する方法を示しています。 XML(拡張可能なマークアップ言語)は、人間の読みやすさとマシン解析の両方に合わせて設計された多用途のテキストベースのマークアップ言語です。一般的にデータストレージに使用されます

PHPでの後期静的結合を説明します(静的::)。 PHPでの後期静的結合を説明します(静的::)。 Apr 03, 2025 am 12:04 AM

静的結合(静的::) PHPで後期静的結合(LSB)を実装し、クラスを定義するのではなく、静的コンテキストで呼び出しクラスを参照できるようにします。 1)解析プロセスは実行時に実行されます。2)継承関係のコールクラスを検索します。3)パフォーマンスオーバーヘッドをもたらす可能性があります。

PHPマジックメソッド(__construct、__destruct、__call、__get、__setなど)とは何ですか? PHPマジックメソッド(__construct、__destruct、__call、__get、__setなど)とは何ですか? Apr 03, 2025 am 12:03 AM

PHPの魔法の方法は何ですか? PHPの魔法の方法には次のものが含まれます。1。\ _ \ _コンストラクト、オブジェクトの初期化に使用されます。 2。\ _ \ _リソースのクリーンアップに使用される破壊。 3。\ _ \ _呼び出し、存在しないメソッド呼び出しを処理します。 4。\ _ \ _ get、dynamic属性アクセスを実装します。 5。\ _ \ _セット、動的属性設定を実装します。これらの方法は、特定の状況で自動的に呼び出され、コードの柔軟性と効率を向上させます。

See all articles