Help us understand the problem. What is going on with this article?

PHPのオブジェクトと変数リファレンスは全然別物

More than 1 year has passed since last update.

まだ参照渡しで消耗してたの? 僕はPHPerなのでPHPのことしかわからないけど、だいたいPHPで参照渡しのことを気にしても時間の無駄だし積極的に利用するべき場面が極めて限定されることはPHPのリファレンス(参照&)の傾向と対策、あるいはさよならに書いたので読んでほしい。

それとPHPマニュアルでは「参照渡し」ではなく「リファレンス渡し」と呼ぶが、どちらにせよ、わざわざ「渡し (call by-)」を付けて呼ぶ意義は乏しいので、これからは「変数リファレンス」として覚えてほしい。もうちょっと具体的なことはPHP: リファレンスとは? - Manualを読んで。

ふしぎなふしぎなオブジェクト

PHPの配列とオブジェクトは、似たような書きかたをしてもまったく異なる挙動をします。

以下のコードはarray(連想配列)とオブジェクトで異なる挙動をします。

function setPrice1(array $a, int $tax): array
{
    if (!isset($a['prime_cost'])) {
        $a['prime_cost'] = $a['price'];
        $a['tax'] = $tax;
        $a['price'] = $a['prime_cost'] + $a['prime_cost'] * $tax;
    }

    return $a;
}

function setPrice2(stdClass $a, int $tax): stdClass
{
    if (!isset($a->prime_cost)) {
        $a->prime_cost = $a->price;
        $a->tax = $tax;
        $a->price = $a->prime_cost + $a->prime_cost * $tax;
    }

    return $a;
}

動かしてみようか (練習問題)

<?php

$ary = ['name' => 'りんご', 'price' => 100];

echo "before:";
var_dump($ary);

$bry = setPrice1($ary, 0.03);

echo "after:";
var_dump($bry);

var_dump($ary === $bry);

echo PHP_EOL;

////////////////////

$obj = (object)['name' => 'りんご', 'price' => 100];

echo "before:";
var_dump($obj);

$obj2 = setPrice2($obj, 0.03);

echo "after:";
var_dump($obj2);

var_dump($obj === $obj2);

だいじなのは $ary === $bry$obj === $obj2 のところです。おんなじように書いても別の結果になりましたよね?

なんでこうなるかと言ったらこうなるようにPHPが設計されたからと表現するしかないのですが、実際この挙動の差は便利なことの方が多いです。この話はオブジェクトをいい感じに複製(クローン)する [myclabs/deep-copy] - 超PHPerになろうにも書きました。

この例ではstdClassを使ってますが、これを実コードで積極的に活用することはおすすめしません。クラスを定義してください。

問題は、この両者の差異を「PHPは参照渡しだから」「オブジェクトは参照渡しされるから」のような意味不明な謎解説をして初心者を騙すひとが居ることです。繰り返しますが、このオブジェクトの挙動は「変数リファレンス」とは無関係です。PHP4のことはしらない。

それでも疑り深いひとのためのアレ

ここでは「日時オブジェクトを日本時間の正午にセットする」3種類の函数を用意してみる。PHP 7.1。

function setdate1_オブジェクト(DateTime $dt): void
{
    $dt->setTimezone(new DateTimeZone('Asia/Tokyo'));
    $dt->setTime(12, 0);
}

function setdate2_変数リファレンス(DateTimeInterface &$dt): void
{
    $dt = $dt->setTimezone(new DateTimeZone('Asia/Tokyo'));
    $dt = $dt->setTime(12, 0);
}

function setdate3_イミュータブル(DateTimeImmutable $dt): DateTimeImmutable
{
    $dt = $dt->setTimezone(new DateTimeZone('Asia/Tokyo'));
    $dt = $dt->setTime(12, 0);

    return $dt;
}

課題

この三種類の実装を比較し、それぞれの利点と欠点を挙げて考察せよ。

レポートは来週の講義開始前までに提出のこと

参考にするとよい資料

ヒント

このような.phpファイルを用意し、呼び出し側と実装コードをそれぞれ変更し出力を観察するとよい。

datetime-jikken.php
<?php

echo "---------------------------------", PHP_EOL;
echo "[setdate1_オブジェクト] (DateTime)", PHP_EOL;
echo "---------------------------------", PHP_EOL, PHP_EOL;
$dt1 = new DateTime;

echo "before:";
var_dump($dt1);

setdate1_オブジェクト($dt1);

echo "after:";
var_dump($dt1);
echo PHP_EOL;

////////////////////

echo "-------------------------------------", PHP_EOL;
echo "[setdate2_変数リファレンス] (DateTime)", PHP_EOL;
echo "-------------------------------------", PHP_EOL, PHP_EOL;
$dt2_1 = new DateTime;

echo "before:";
var_dump($dt2_1);

setdate2_変数リファレンス($dt2_1);

echo "after:";
var_dump($dt2_1);
echo PHP_EOL;

////////////////////

echo "----------------------------------------------", PHP_EOL;
echo "[setdate2_変数リファレンス] (DateTimeImmutable)", PHP_EOL;
echo "----------------------------------------------", PHP_EOL, PHP_EOL;
$dt2_2 = new DateTimeImmutable;

echo "before:";
var_dump($dt2_2);

setdate2_変数リファレンス($dt2_2);

echo "after:";
var_dump($dt2_2);
echo PHP_EOL;

////////////////////

echo "----------------------------------------------", PHP_EOL;
echo "[setdate3_イミュータブル] (DateTimeImmutable)", PHP_EOL;
echo "----------------------------------------------", PHP_EOL, PHP_EOL;
$dt3 = new DateTimeImmutable;

echo "before:";
var_dump($dt3);

$dt3 = setdate3_イミュータブル($dt3);

echo "after:";
var_dump($dt3);
echo PHP_EOL;


function setdate1_オブジェクト(DateTime $dt): void
{
    $dt->setTimezone(new DateTimeZone('Asia/Tokyo'));
    $dt->setTime(12, 0);
}

function setdate2_変数リファレンス(DateTimeInterface &$dt): void
{
    $dt = $dt->setTimezone(new DateTimeZone('Asia/Tokyo'));
    $dt = $dt->setTime(12, 0);
}

function setdate3_イミュータブル(DateTimeImmutable $dt): DateTimeImmutable
{
    $dt = $dt->setTimezone(new DateTimeZone('Asia/Tokyo'));
    $dt = $dt->setTime(12, 0);

    return $dt;
}
tadsan
僕に警備する自宅をください。Emacs初心者。Rubyist。 全ての投稿された記事は別段の表記がない限りはCC 3.0 BY-SA https://creativecommons.org/licenses/by-sa/3.0/deed.ja で二次利用できます。 記事中に含まれる全てのコードスニペットの著作権は抛棄するので、煮るなり焼くなりお好きにどうぞ。
https://tadsan.github.io/
Why not register and get more from Qiita?
  1. We will deliver articles that match you
    By following users and tags, you can catch up information on technical fields that you are interested in as a whole
  2. you can read useful information later efficiently
    By "stocking" the articles you like, you can search right away
Comments
No comments
Sign up for free and join this conversation.
If you already have a Qiita account
Why do not you register as a user and use Qiita more conveniently?
You need to log in to use this function. Qiita can be used more conveniently after logging in.
You seem to be reading articles frequently this month. Qiita can be used more conveniently after logging in.
  1. We will deliver articles that match you
    By following users and tags, you can catch up information on technical fields that you are interested in as a whole
  2. you can read useful information later efficiently
    By "stocking" the articles you like, you can search right away
ユーザーは見つかりませんでした