目次
回复内容:
单例模式是指整个应用中类只有一个对象实例的设计模式。
ホームページ バックエンド開発 PHPチュートリアル 设计模式 - 关于PHP单例模式,有一点不明白,求指教!

设计模式 - 关于PHP单例模式,有一点不明白,求指教!

Jun 06, 2016 pm 08:25 PM
php シングルトン デザインパターン

首先,我定义个类,实现单例模式:(这里是简单一写,就是个最基本的单例)

<code>class Demo
{
    public static $instance;
    
    private function __Construct()
    {
        //TODO
    }
    
    public static function getInstance()
    {
        if(!self::$instance){
            self::$instance = new static();
        }
        
        return self::$instance;
    }
    
    public function call()
    {
        //其他方法
    }
}</code>
ログイン後にコピー
ログイン後にコピー

下面有两种方式实例化类:

1.在需要用的地方

<code>$aa = Demo::getInstance();
$bb = Demo::getInstance();
$cc = Demo::getInstance();</code>
ログイン後にコピー
ログイン後にコピー

这样调用肯定是没问题的,一般情况也是这样初始化。

2.定义一个普通类,写个函数初始化,保存在一个静态变量,如下:

类:

<code>class Demo{
    public function __Construct(){
        //TODO
    }
}</code>
ログイン後にコピー
ログイン後にコピー

函数:

<code>function get_obj(){
    static $obj;
    if($obj){
        return $obj;
    }else{
        $obj = new Demo;
        return $obj;
    }
}</code>
ログイン後にコピー
ログイン後にコピー

在需要调用的地方这样写:

<code>$obj = get_obj();
$obj->call();

$obj2 = get_obj();
$obj2->call();

$obj3 = get_obj();
$obj3 = get_obj();</code>
ログイン後にコピー
ログイン後にコピー

这样不是也只实例化一次这个类吗?

回复内容:

首先,我定义个类,实现单例模式:(这里是简单一写,就是个最基本的单例)

<code>class Demo
{
    public static $instance;
    
    private function __Construct()
    {
        //TODO
    }
    
    public static function getInstance()
    {
        if(!self::$instance){
            self::$instance = new static();
        }
        
        return self::$instance;
    }
    
    public function call()
    {
        //其他方法
    }
}</code>
ログイン後にコピー
ログイン後にコピー

下面有两种方式实例化类:

1.在需要用的地方

<code>$aa = Demo::getInstance();
$bb = Demo::getInstance();
$cc = Demo::getInstance();</code>
ログイン後にコピー
ログイン後にコピー

这样调用肯定是没问题的,一般情况也是这样初始化。

2.定义一个普通类,写个函数初始化,保存在一个静态变量,如下:

类:

<code>class Demo{
    public function __Construct(){
        //TODO
    }
}</code>
ログイン後にコピー
ログイン後にコピー

函数:

<code>function get_obj(){
    static $obj;
    if($obj){
        return $obj;
    }else{
        $obj = new Demo;
        return $obj;
    }
}</code>
ログイン後にコピー
ログイン後にコピー

在需要调用的地方这样写:

<code>$obj = get_obj();
$obj->call();

$obj2 = get_obj();
$obj2->call();

$obj3 = get_obj();
$obj3 = get_obj();</code>
ログイン後にコピー
ログイン後にコピー

这样不是也只实例化一次这个类吗?

完善的单例应该是这样的

class Foobar {
    static private $instance;
    
    // 禁止外部new Foobar
    private function __construct() {
    }
    
    // 禁止clone $foobar
    private function __clone() {
    }
    
    static public function getInstance() {
        retrun self::$instance
            ?: (self::$instance = new self);
    }
}
ログイン後にコピー

如果还要考虑到继承的话

class Foo {
    static private $instances = [];

    protected function __construct() {
    }

    final private function __clone() {
    }

    final static public function getInstance() {
        $class = get_called_class();

        if (!isset(self::$instances[$class])) {
            self::$instances[$class] = new static;
        }

        return self::$instances[$class];
    }
}

class Bar extends Foo {

}

$foo = Foo::getInstance();
$bar = Bar::getInstance();
ログイン後にコピー

get_obj()那种写法,也可以达到目的,但无法禁止new和clone,也就无法做到真正的单例

这问题我也有印象,我也答过一次
http://segmentfault.com/q/1010000003894638
看了一下还是题主你,为毛要问两遍……

然后你的第二种方式并非单例,请不要陷入抠概念的误区
偶尔也有你这种用法的,少见,并且不好。在一般情况下用一个特定对象就行了,特殊情况下需要新的实例的场合会用,但一般也不会这么写(早期的PHP会有一些类这么做)

= = 所以呢?
两种写法而已啊。。。
另外,你无论怎么看,都还是第一种更清晰,更独立,更好用啊。。。
第一种封装、抽象的更好,更加符合面向对象啊。
存一个静态变量或者另外弄个函数什么的,你不觉得乱嘛。。。

静态变量可以定义在函数里或者类属性,两种写法都可以,但前者封装性会比较好些。

楼主给了我启发,过程式编程中,在全局函数内用静态变量存储数据库连接实现单例:

<code><?php
// config.php
$app = array(
    'db_host'        => '127.0.0.1',
    'db_username'    => 'root',
    'db_password'    => '',
    'db_name'        => 'mysql'
);
// functions.php
function cn_db() {
    global $app;
    static $mysqli;
    if ($mysqli) {
        return $mysqli;
    } else {
        $mysqli = new mysqli($app['db_host'], $app['db_username'], $app['db_password'], $app['db_name']);
        return $mysqli;
    }
}
function cn_foo1() {
    $mysqli = cn_db();
    return $mysqli->query('select user,host from user where user = \'root\'')->fetch_all();
}
function cn_foo2() {
    $mysqli = cn_db();
    return $mysqli->query('select user,host from user')->fetch_all();
}
// controller + view
print_r(cn_foo1());
print_r(cn_foo2());</code>
ログイン後にコピー

$instance = new static(); (楼主这行代码写错了),感觉单例模式就是个思想,你第一个例子就是面向对象,第二个例子既有面向对象,又有面向过程。

单例模式是指整个应用中类只有一个对象实例的设计模式。

第二种方式是单例吗?

<code>class Demo {
    public function __construct(){
        //TODO
    }
}
function get_obj() {
    static $obj;
    if($obj){
        return $obj;
    }else{
        $obj = new Demo;
        return $obj;
    }
}

var_dump(get_obj());
var_dump(new Demo);
</code>
ログイン後にコピー

得出的是2个不同的实例

你那样写不符合面向对象思想的,而且你的单例模式也不完整,没有考虑到clone和extends的情况。可以参考下http://www.xtwind.com/design-pattern-singleton.html

单例模式,是一种设计思想实现方式

单例首先是面向对象里面的概念,
第二种就不是面向对象

get_obj()那种写法无法保证外部不new Demo对象

单例的要义第一:私有化构造方法,当然PHP还提供了clone魔术方法,也要私有化。这样做事为了保证外部实例化对象的入口只有一个,只能是我们暴露出来的静态函数getInstance()

单例模式,三私一公,其中的三私:构造方法,克隆魔术方法,实例化对象。一公:对外提供的方法

このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、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衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

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

SublimeText3 中国語版

SublimeText3 中国語版

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

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

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

CakePHP プロジェクトの構成 CakePHP プロジェクトの構成 Sep 10, 2024 pm 05:25 PM

この章では、CakePHP の環境変数、一般設定、データベース設定、電子メール設定について理解します。

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 にアップグレードする方法について説明します。

CakePHP の日付と時刻 CakePHP の日付と時刻 Sep 10, 2024 pm 05:27 PM

Cakephp4 で日付と時刻を操作するには、利用可能な FrozenTime クラスを利用します。

CakePHP ファイルのアップロード CakePHP ファイルのアップロード Sep 10, 2024 pm 05:27 PM

ファイルのアップロードを行うには、フォーム ヘルパーを使用します。ここではファイルアップロードの例を示します。

CakePHP ルーティング CakePHP ルーティング Sep 10, 2024 pm 05:25 PM

この章では、ルーティングに関連する次のトピックを学習します。

CakePHP について話し合う CakePHP について話し合う Sep 10, 2024 pm 05:28 PM

CakePHP は、PHP 用のオープンソース フレームワークです。これは、アプリケーションの開発、展開、保守をより簡単にすることを目的としています。 CakePHP は、強力かつ理解しやすい MVC のようなアーキテクチャに基づいています。モデル、ビュー、コントローラー

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 は、

CakePHP バリデータの作成 CakePHP バリデータの作成 Sep 10, 2024 pm 05:26 PM

Validator は、コントローラーに次の 2 行を追加することで作成できます。

See all articles