<?php
include_once $_SERVER['DOCUMENT_ROOT'] . '/include/shared-manual.inc';
$TOC = array();
$TOC_DEPRECATED = array();
$PARENTS = array();
include_once dirname(__FILE__) ."/toc/language.types.inc";
$setup = array (
  'home' => 
  array (
    0 => 'index.php',
    1 => 'PHP Manual',
  ),
  'head' => 
  array (
    0 => 'UTF-8',
    1 => 'ja',
  ),
  'this' => 
  array (
    0 => 'language.types.declarations.php',
    1 => '型宣言',
    2 => '型宣言',
  ),
  'up' => 
  array (
    0 => 'language.types.php',
    1 => '型',
  ),
  'prev' => 
  array (
    0 => 'language.types.iterable.php',
    1 => 'Iterable',
  ),
  'next' => 
  array (
    0 => 'language.types.type-juggling.php',
    1 => '型の相互変換',
  ),
  'alternatives' => 
  array (
  ),
  'source' => 
  array (
    'lang' => 'ja',
    'path' => 'language/types/declarations.xml',
  ),
  'history' => 
  array (
  ),
);
$setup["toc"] = $TOC;
$setup["toc_deprecated"] = $TOC_DEPRECATED;
$setup["parents"] = $PARENTS;
manual_setup($setup);

contributors($setup);

?>
<div id="language.types.declarations" class="sect1">
 <h2 class="title">型宣言</h2>

 <p class="para">
  関数のパラメータや戻り値、
  クラスのプロパティ (PHP 7.4.0 以降)、クラス定数 (PHP 8.3.0 以降) に対して型を宣言することができます。
  これによって、その値がコール時に特定の型であることを保証できます。
  その型でない場合は、<span class="classname"><a href="class.typeerror.php" class="classname">TypeError</a></span> がスローされます。
 </p>

 <p class="para">
  PHP がサポートしている単一の型それぞれを、
  ユーザーが行う型宣言の中で使うことができます。
  但し、<span class="type"><a href="language.types.resource.php" class="type resource">resource</a></span> 型を除きます。
  このページでは、それぞれの型がいつ利用可能になったかの変更履歴や、
  型宣言におけるそれらの使い方について記しています。
 </p>

 <blockquote class="note"><p><strong class="note">注意</strong>: 
  <p class="para">
   クラスがインターフェイスのメソッドを実装したり、
   親クラスで既に定義されているメソッドを再実装する場合、
   そのメソッドは、既に存在するメソッドと互換性がなければなりません。
   <a href="language.oop5.variance.php" class="link">共変性と反変性</a>
   のルールに従っている場合、メソッドには互換性があります。
  </p>
 </p></blockquote>

 <div class="sect2">
  <h3 class="title">変更履歴</h3>
  <table class="doctable informaltable">
   
    <thead>
     <tr>
      <th>バージョン</th>
      <th>説明</th>
     </tr>

    </thead>

    <tbody class="tbody">
     <tr>
      <td>8.3.0</td>
      <td>
       クラス、インターフェイス、トレイト、そして列挙型の定数は、新たに型宣言をサポートするようになりました。
      </td>
     </tr>

     <tr>
      <td>8.2.0</td>
      <td>
       <abbr title="Disjunctive Normal Form">DNF</abbr> 型のサポートが追加されました。
      </td>
     </tr>

     <tr>
      <td>8.2.0</td>
      <td>
       リテラル型 <span class="type"><a href="language.types.singleton.php" class="type true">true</a></span> のサポートが追加されました。
      </td>
     </tr>

     <tr>
      <td>8.2.0</td>
      <td>
       <span class="type"><a href="language.types.null.php" class="type null">null</a></span> と <span class="type"><a href="language.types.singleton.php" class="type false">false</a></span> 型が、独立した型として使えるようになりました。
      </td>
     </tr>

     <tr>
      <td>8.1.0</td>
      <td>
       交差型のサポートが追加されました。
      </td>
     </tr>

     <tr>
      <td>8.1.0</td>
      <td>
       戻り値を <span class="type"><span class="type"><a href="language.types.void.php" class="type void">void</a></span></span> とした関数からリファレンスを返すことは、
       推奨されなくなりました。
      </td>
     </tr>

     <tr>
      <td>8.1.0</td>
      <td>
       戻り値にのみ指定できる型として、
       <span class="type"><a href="language.types.never.php" class="type never">never</a></span> 型のサポートが追加されました。
      </td>
     </tr>

     <tr>
      <td>8.0.0</td>
      <td>
       <span class="type"><a href="language.types.mixed.php" class="type mixed">mixed</a></span> 型のサポートが追加されました。
      </td>
     </tr>

     <tr>
      <td>8.0.0</td>
      <td>
       戻り値にのみ指定できる型として、
       <span class="type">static</span> 型のサポートが追加されました。
      </td>
     </tr>

     <tr>
      <td>8.0.0</td>
      <td>
       union 型のサポートが追加されました。
      </td>
     </tr>

     <tr>
      <td>7.4.0</td>
      <td>
       クラスのプロパティに、型宣言のサポートが追加されました。
      </td>
     </tr>

     <tr>
      <td>7.2.0</td>
      <td>
       <span class="type"><a href="language.types.object.php" class="type object">object</a></span> 型のサポートが追加されました。
      </td>
     </tr>

     <tr>
      <td>7.1.0</td>
      <td>
       <span class="type"><a href="language.types.iterable.php" class="type iterable">iterable</a></span> 型のサポートが追加されました。
      </td>
     </tr>

     <tr>
      <td>7.1.0</td>
      <td>
       <span class="type"><span class="type"><a href="language.types.void.php" class="type void">void</a></span></span> 型のサポートが追加されました。
      </td>
     </tr>

     <tr>
      <td>7.1.0</td>
      <td>
       nullable な型のサポートが追加されました。
      </td>
     </tr>

    </tbody>
   
  </table>

 </div>

 <div class="sect2" id="language.types.declarations.base">
  <h3 class="title">基本型を使うときの注意</h3>

  <p class="simpara">
   基本型は、ここで説明する小さな注意事項はいくつかあるものの、
   わかりやすい振る舞いをします。
  </p>

  <div class="sect3" id="language.types.declarations.base.scalar">
   <h4 class="title">スカラー型</h4>
   <div class="warning"><strong class="warning">警告</strong>
    <p class="para">
     スカラー型(<span class="type"><a href="language.types.boolean.php" class="type bool">bool</a></span>, <span class="type"><a href="language.types.integer.php" class="type int">int</a></span>,
     <span class="type"><a href="language.types.float.php" class="type float">float</a></span>, <span class="type"><a href="language.types.string.php" class="type string">string</a></span>) のエイリアスはサポートされていません。
     つまり、これらはクラスやインターフェイスの名前として扱われているということです。
     たとえば、型の宣言に <code class="literal">boolean</code> を使った場合、
     値が <code class="literal">boolean</code>
     クラスまたはインターフェイスのインスタンスであることが要求されます。
     <span class="type"><a href="language.types.boolean.php" class="type bool">bool</a></span> 型ではありません。
    </p>
    <div class="informalexample">
     <div class="example-contents">
<div class="annotation-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br />    </span><span style="color: #007700">function </span><span style="color: #0000BB">test</span><span style="color: #007700">(</span><span style="color: #0000BB">boolean $param</span><span style="color: #007700">) {}<br />    </span><span style="color: #0000BB">test</span><span style="color: #007700">(</span><span style="color: #0000BB">true</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
     </div>

     <p class="para">上の例の PHP 8 での出力は、このようになります。:</p>
     <div class="example-contents screen">
<div class="annotation-interactive cdata"><pre>
Warning: &quot;boolean&quot; will be interpreted as a class name. Did you mean &quot;bool&quot;? Write &quot;\boolean&quot; to suppress this warning in /in/9YrUX on line 2

Fatal error: Uncaught TypeError: test(): Argument #1 ($param) must be of type boolean, bool given, called in - on line 3 and defined in -:2
Stack trace:
#0 -(3): test(true)
#1 {main}
  thrown in - on line 2
</pre></div>
     </div>
    </div>
   </div>
  </div>

  <div class="sect3" id="language.types.declarations.void">
   <h4 class="title">void</h4>
   <blockquote class="note"><p><strong class="note">注意</strong>: 
    <p class="para">
     <span class="type"><span class="type"><a href="language.types.void.php" class="type void">void</a></span></span> を返す関数からリファレンスを返すことは、PHP 8.1.0 以降は推奨されなくなりました。
     なぜなら、関数の定義そのものが矛盾しているからです。
     PHP 8.1.0 より前のバージョンでは、
     関数をコールした際に次のような <strong><code><a href="errorfunc.constants.php#constant.e-notice">E_NOTICE</a></code></strong> が発生していました:
     <span class="computeroutput">Only variable references should be returned by reference</span>
     <div class="informalexample">
      <div class="example-contents">
<div class="annotation-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">function &amp;</span><span style="color: #0000BB">test</span><span style="color: #007700">(): </span><span style="color: #0000BB">void </span><span style="color: #007700">{}<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
      </div>

     </div>
    </p>
   </p></blockquote>
  </div>

  <div class="sect3" id="language.types.declarations.base.function">
   <h4 class="title">Callable</h4>
   <p class="para">
    この型は、クラスのプロパティの型宣言では使うことができません。
   </p>

   <blockquote class="note"><p><strong class="note">注意</strong>: 
    <span class="simpara">
     関数のシグネチャに指定することもできません。
    </span>
   </p></blockquote>
  </div>

  <div class="sect3" id="language.types.declarations.references">
   <h4 class="title">リファレンス渡しのパラメータに対する型宣言</h4>

   <p class="simpara">
    リファレンス渡しのパラメータに対して宣言される型は、
    関数の入り口で <em>だけ</em> チェックされます。
    しかし、関数から返される時はチェックされません。
    これは、変数のリファレンスについては、関数が型を変更できるということです。
   </p>
   <div class="example" id="example-1">
    <p><strong>例1 リファレンス渡しのパラメータに対する型宣言</strong></p>
    <div class="example-contents">
<div class="annotation-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">function </span><span style="color: #0000BB">array_baz</span><span style="color: #007700">(array &amp;</span><span style="color: #0000BB">$param</span><span style="color: #007700">)<br />{<br />    </span><span style="color: #0000BB">$param </span><span style="color: #007700">= </span><span style="color: #0000BB">1</span><span style="color: #007700">;<br />}<br /></span><span style="color: #0000BB">$var </span><span style="color: #007700">= [];<br /></span><span style="color: #0000BB">array_baz</span><span style="color: #007700">(</span><span style="color: #0000BB">$var</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">var_dump</span><span style="color: #007700">(</span><span style="color: #0000BB">$var</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">array_baz</span><span style="color: #007700">(</span><span style="color: #0000BB">$var</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
    </div>

    <div class="example-contents"><p>上の例の出力は、
たとえば以下のようになります。</p></div>
    <div class="example-contents screen">
<div class="annotation-interactive cdata"><pre>
int(1)

Fatal error: Uncaught TypeError: array_baz(): Argument #1 ($param) must be of type array, int given, called in - on line 9 and defined in -:2
Stack trace:
#0 -(9): array_baz(1)
#1 {main}
  thrown in - on line 2
</pre></div>
    </div>
   </div>
  </div>
 </div>

 <div class="sect2" id="language.types.declarations.composite">
  <h3 class="title">複合型を使うときの注意</h3>
  <p class="para">
   複合型を宣言する場合、制限がいくつか存在します。
   また、簡単なバグを防ぐために、型の冗長チェックがコンパイル時に行われます。
  </p>

  <div class="caution"><strong class="caution">警告</strong>
   <p class="simpara">
    PHP 8.2.0 より前のバージョン、
    つまり <abbr title="Disjunctive Normal Form">DNF</abbr> 型がサポートされる前は、
    交差型とunion型を組み合わせることはできませんでした。
   </p>
  </div>

  <div class="sect3" id="language.types.declarations.composite.union">
   <h4 class="title">union 型</h4>
   <div class="warning"><strong class="warning">警告</strong>
    <p class="simpara">
     union 型の中で、シングルトン型 <span class="type"><a href="language.types.singleton.php" class="type true">true</a></span> と
     <span class="type"><a href="language.types.singleton.php" class="type false">false</a></span> を同時に組み合わせて使うことはできません。
     <span class="type"><a href="language.types.boolean.php" class="type bool">bool</a></span> 型を使ってください。
    </p>
   </div>

   <div class="caution"><strong class="caution">警告</strong>
    <p class="simpara">
     PHP 8.2.0 より前のバージョンでは、
     <strong><code><a href="reserved.constants.php#constant.false">false</a></code></strong> と <span class="type"><a href="language.types.null.php" class="type null">null</a></span> 型は
     独立した型として使えず、
     union 型でそれらだけを指定することも許されませんでした。
     つまり、<strong><code><a href="reserved.constants.php#constant.false">false</a></code></strong>, <code class="literal">false|null</code>,
     <code class="literal">?false</code> のような型はいずれも許されませんでした。
    </p>
   </div>

   <div class="sect4" id="language.types.declarations.nullable">
    <h5 class="title">nullable な型とシンタックスシュガー</h5>

    <p class="para">
     単一の基本型を宣言した場合、
     型の名前の前にクエスチョンマーク (<code class="literal">?</code>) を付けることで、nullable であるという印を付けることができます。
     よって、<code class="literal">?T</code> と <code class="literal">T|null</code> は同じ意味です。
    </p>

    <blockquote class="note"><p><strong class="note">注意</strong>: 
     <span class="simpara">
      この文法は、PHP 7.1.0 以降でサポートされており、
      一般化された union 型がサポートされる前から存在します。
     </span>
    </p></blockquote>

    <blockquote class="note"><p><strong class="note">注意</strong>: 
     <p class="para">
      デフォルト値に <code class="literal">null</code> を指定することで、
      null を許容するパラメータを指定することができます。
      これは、子クラスでデフォルト値が変更された場合にクラスの互換性が壊れ、
      型宣言で <span class="type"><a href="language.types.null.php" class="type null">null</a></span>
      型を追加しなければならなくなるため、おすすめできません。
      この振る舞いは、PHP 8.4 以降は推奨されなくなっています。
     </p>
     <div class="example" id="example-2">
      <p><strong>例2 引数にnullを許容する古いやり方</strong></p>
      <div class="example-contents">
<div class="annotation-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">class </span><span style="color: #0000BB">C </span><span style="color: #007700">{}<br /><br />function </span><span style="color: #0000BB">f</span><span style="color: #007700">(</span><span style="color: #0000BB">C $c </span><span style="color: #007700">= </span><span style="color: #0000BB">null</span><span style="color: #007700">) {<br />    </span><span style="color: #0000BB">var_dump</span><span style="color: #007700">(</span><span style="color: #0000BB">$c</span><span style="color: #007700">);<br />}<br /><br /></span><span style="color: #0000BB">f</span><span style="color: #007700">(new </span><span style="color: #0000BB">C</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">f</span><span style="color: #007700">(</span><span style="color: #0000BB">null</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
      </div>

      <div class="example-contents"><p>上の例の出力は以下となります。</p></div>
      <div class="example-contents screen">
<div class="annotation-interactive cdata"><pre>
object(C)#1 (0) {
}
NULL
</pre></div>
      </div>
     </div>
    </p></blockquote>
   </div>
  </div>

  <div class="sect3" id="language.types.declarations.composite.redundant">
   <h4 class="title">重複した冗長な型</h4>
   <p class="para">
    複合型の宣言に関する単純なバグを見つけるため、
    クラスの読み込みを行わずに検出できる冗長な型はコンパイル時にエラーになります。 たとえば、以下のような場合です:

    <ul class="itemizedlist">
     <li class="listitem">
      <span class="simpara">
       名前が解決された型は、一度しか現れることができません。
       <code class="literal">int|string|INT</code> や
       <code class="literal">Countable&amp;Traversable&amp;COUNTABLE</code>
       のような型はエラーになります。
      </span>
     </li>
     <li class="listitem">
      <span class="simpara">
       <span class="type"><a href="language.types.mixed.php" class="type mixed">mixed</a></span> 型や、
       <span class="type"><a href="language.types.never.php" class="type never">never</a></span> 型を複合型で使うとエラーになります。
      </span>
     </li>
     <li class="listitem">
      <span class="simpara">union 型に適用される制限:</span>
      <ul class="itemizedlist">
       <li class="listitem">
        <span class="simpara">
         <span class="type"><a href="language.types.boolean.php" class="type bool">bool</a></span> 型が使われている場合、<span class="type"><a href="language.types.singleton.php" class="type false">false</a></span> や
         <span class="type"><a href="language.types.singleton.php" class="type true">true</a></span> 型は追加で使えません。  
        </span>
       </li>
       <li class="listitem">
        <span class="simpara">
         <span class="type"><a href="language.types.object.php" class="type object">object</a></span> 型が使われている場合、クラス型は追加で使えません。
        </span>
       </li>
       <li class="listitem">
        <span class="simpara">
         <span class="type"><a href="language.types.iterable.php" class="type iterable">iterable</a></span> 型が使われている場合、
         <span class="type"><a href="language.types.array.php" class="type array">array</a></span> と <span class="classname"><a href="class.traversable.php" class="classname">Traversable</a></span>
         は追加で使えません。
        </span>
       </li>
      </ul>
     </li>
     <li class="listitem">
      <span class="simpara">交差型に適用される制限:</span>
      <ul class="itemizedlist">
       <li class="listitem">
        <span class="simpara">
         クラス型でない型を、交差型で使うとエラーになります。
        </span>
       </li>
       <li class="listitem">
        <span class="simpara">
         <span class="type">self</span>, <span class="type">parent</span>,
         <span class="type">static</span> のいずれかを、交差型で使うとエラーになります。
        </span>
       </li>
      </ul>
     </li>
     <li class="listitem">
      <span class="simpara"><abbr title="Disjunctive Normal Form">DNF</abbr> 型に適用される制限:</span>
      <ul class="itemizedlist">
       <li class="listitem">
        <span class="simpara">
         より広い型をひとつ使った場合、
         それより狭い型は冗長とみなされます。
        </span>
       </li>
       <li class="listitem">
        <span class="simpara">
         ふたつの等しい交差型を使った場合、冗長とみなされます。
        </span>
       </li>
      </ul>
     </li>
    </ul>
   </p>

   <blockquote class="note"><p><strong class="note">注意</strong>: 
    <span class="simpara">
     これによって、型が &quot;最低限&quot; であることは保証しません。
     なぜなら、最低限であることを保証するためには、
     使われている全てのクラスの型を読み込まなければならないからです。
    </span>
   </p></blockquote>

   <p class="para">
    たとえば、<code class="literal">A</code> と <code class="literal">B</code>
    がクラスのエイリアスだったとします。
    この場合、<code class="literal">A|B</code> は <code class="literal">A</code>
    または <code class="literal">B</code> のみに縮めることができますが、
    正しい union 型です。 同様に、<code class="code">B extends A {}</code>
    というクラスがあった場合、 <code class="literal">A|B</code>
    は <code class="literal">A</code> のみに縮めることができますが、
    正しい union 型です。

    <div class="informalexample">
     <div class="example-contents">
<div class="annotation-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">function </span><span style="color: #0000BB">foo</span><span style="color: #007700">(): </span><span style="color: #0000BB">int</span><span style="color: #007700">|</span><span style="color: #0000BB">INT </span><span style="color: #007700">{} </span><span style="color: #FF8000">// 許されません<br /></span><span style="color: #007700">function </span><span style="color: #0000BB">foo</span><span style="color: #007700">(): </span><span style="color: #0000BB">bool</span><span style="color: #007700">|</span><span style="color: #0000BB">false </span><span style="color: #007700">{} </span><span style="color: #FF8000">// 許されません<br /></span><span style="color: #007700">function </span><span style="color: #0000BB">foo</span><span style="color: #007700">(): </span><span style="color: #0000BB">int</span><span style="color: #007700">&amp;</span><span style="color: #0000BB">Traversable </span><span style="color: #007700">{} </span><span style="color: #FF8000">// 許されません<br /></span><span style="color: #007700">function </span><span style="color: #0000BB">foo</span><span style="color: #007700">(): </span><span style="color: #0000BB">self</span><span style="color: #007700">&amp;</span><span style="color: #0000BB">Traversable </span><span style="color: #007700">{} </span><span style="color: #FF8000">// 許されません<br /><br /></span><span style="color: #007700">use </span><span style="color: #0000BB">A </span><span style="color: #007700">as </span><span style="color: #0000BB">B</span><span style="color: #007700">;<br />function </span><span style="color: #0000BB">foo</span><span style="color: #007700">(): </span><span style="color: #0000BB">A</span><span style="color: #007700">|</span><span style="color: #0000BB">B </span><span style="color: #007700">{} </span><span style="color: #FF8000">// 許されません ("use" は名前解決の一部です)<br /></span><span style="color: #007700">function </span><span style="color: #0000BB">foo</span><span style="color: #007700">(): </span><span style="color: #0000BB">A</span><span style="color: #007700">&amp;</span><span style="color: #0000BB">B </span><span style="color: #007700">{} </span><span style="color: #FF8000">// 許されません ("use" は名前解決の一部です)<br /><br /></span><span style="color: #0000BB">class_alias</span><span style="color: #007700">(</span><span style="color: #DD0000">'X'</span><span style="color: #007700">, </span><span style="color: #DD0000">'Y'</span><span style="color: #007700">);<br />function </span><span style="color: #0000BB">foo</span><span style="color: #007700">(): </span><span style="color: #0000BB">X</span><span style="color: #007700">|</span><span style="color: #0000BB">Y </span><span style="color: #007700">{} </span><span style="color: #FF8000">// 問題ありません (冗長かどうかは、実行時にだけわかります)<br /></span><span style="color: #007700">function </span><span style="color: #0000BB">foo</span><span style="color: #007700">(): </span><span style="color: #0000BB">X</span><span style="color: #007700">&amp;</span><span style="color: #0000BB">Y </span><span style="color: #007700">{} </span><span style="color: #FF8000">// 問題ありません (冗長かどうかは、実行時にだけわかります)<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
     </div>

    </div>
   </p>
  </div>
 </div>

 <div class="sect2" id="language.types.declarations.examples">
  <h3 class="title">例</h3>
  <div class="example" id="example-3">
   <p><strong>例3 クラスによる型宣言の基本</strong></p>
   <div class="example-contents">
<div class="annotation-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">class </span><span style="color: #0000BB">C </span><span style="color: #007700">{}<br />class </span><span style="color: #0000BB">D </span><span style="color: #007700">extends </span><span style="color: #0000BB">C </span><span style="color: #007700">{}<br /><br /></span><span style="color: #FF8000">// このクラスは、C を継承していません<br /></span><span style="color: #007700">class </span><span style="color: #0000BB">E </span><span style="color: #007700">{}<br /><br />function </span><span style="color: #0000BB">f</span><span style="color: #007700">(</span><span style="color: #0000BB">C $c</span><span style="color: #007700">) {<br />    echo </span><span style="color: #0000BB">get_class</span><span style="color: #007700">(</span><span style="color: #0000BB">$c</span><span style="color: #007700">).</span><span style="color: #DD0000">"\n"</span><span style="color: #007700">;<br />}<br /><br /></span><span style="color: #0000BB">f</span><span style="color: #007700">(new </span><span style="color: #0000BB">C</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">f</span><span style="color: #007700">(new </span><span style="color: #0000BB">D</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">f</span><span style="color: #007700">(new </span><span style="color: #0000BB">E</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

   <div class="example-contents"><p>上の例の PHP 8 での出力は、このようになります。:</p></div>
   <div class="example-contents screen">
<div class="annotation-interactive cdata"><pre>
C
D

Fatal error: Uncaught TypeError: f(): Argument #1 ($c) must be of type C, E given, called in /in/gLonb on line 14 and defined in /in/gLonb:8
Stack trace:
#0 -(14): f(Object(E))
#1 {main}
  thrown in - on line 8
</pre></div>
   </div>
  </div>

  <div class="example" id="example-4">
   <p><strong>例4 インターフェイスによる型宣言の基本</strong></p>
   <div class="example-contents">
<div class="annotation-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">interface </span><span style="color: #0000BB">I </span><span style="color: #007700">{ public function </span><span style="color: #0000BB">f</span><span style="color: #007700">(); }<br />class </span><span style="color: #0000BB">C </span><span style="color: #007700">implements </span><span style="color: #0000BB">I </span><span style="color: #007700">{ public function </span><span style="color: #0000BB">f</span><span style="color: #007700">() {} }<br /><br /></span><span style="color: #FF8000">// このクラスは、インターフェイス I を実装していません<br /></span><span style="color: #007700">class </span><span style="color: #0000BB">E </span><span style="color: #007700">{}<br /><br />function </span><span style="color: #0000BB">f</span><span style="color: #007700">(</span><span style="color: #0000BB">I $i</span><span style="color: #007700">) {<br />    echo </span><span style="color: #0000BB">get_class</span><span style="color: #007700">(</span><span style="color: #0000BB">$i</span><span style="color: #007700">).</span><span style="color: #DD0000">"\n"</span><span style="color: #007700">;<br />}<br /><br /></span><span style="color: #0000BB">f</span><span style="color: #007700">(new </span><span style="color: #0000BB">C</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">f</span><span style="color: #007700">(new </span><span style="color: #0000BB">E</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

   <div class="example-contents"><p>上の例の PHP 8 での出力は、このようになります。:</p></div>
   <div class="example-contents screen">
<div class="annotation-interactive cdata"><pre>
C

Fatal error: Uncaught TypeError: f(): Argument #1 ($i) must be of type I, E given, called in - on line 13 and defined in -:8
Stack trace:
#0 -(13): f(Object(E))
#1 {main}
  thrown in - on line 8
</pre></div>
   </div>
  </div>

  <div class="example" id="example-5">
   <p><strong>例5 基本的な戻り値の型宣言</strong></p>
   <div class="example-contents">
<div class="annotation-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">function </span><span style="color: #0000BB">sum</span><span style="color: #007700">(</span><span style="color: #0000BB">$a</span><span style="color: #007700">, </span><span style="color: #0000BB">$b</span><span style="color: #007700">): </span><span style="color: #0000BB">float </span><span style="color: #007700">{<br />    return </span><span style="color: #0000BB">$a </span><span style="color: #007700">+ </span><span style="color: #0000BB">$b</span><span style="color: #007700">;<br />}<br /><br /></span><span style="color: #FF8000">// float が返される点に注意<br /></span><span style="color: #0000BB">var_dump</span><span style="color: #007700">(</span><span style="color: #0000BB">sum</span><span style="color: #007700">(</span><span style="color: #0000BB">1</span><span style="color: #007700">, </span><span style="color: #0000BB">2</span><span style="color: #007700">));<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

   <div class="example-contents"><p>上の例の出力は以下となります。</p></div>
   <div class="example-contents screen">
<div class="annotation-interactive cdata"><pre>
float(3)
</pre></div>
   </div>
  </div>

  <div class="example" id="example-6">
   <p><strong>例6 オブジェクトを返す</strong></p>
   <div class="example-contents">
<div class="annotation-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">class </span><span style="color: #0000BB">C </span><span style="color: #007700">{}<br /><br />function </span><span style="color: #0000BB">getC</span><span style="color: #007700">(): </span><span style="color: #0000BB">C </span><span style="color: #007700">{<br />    return new </span><span style="color: #0000BB">C</span><span style="color: #007700">;<br />}<br /><br /></span><span style="color: #0000BB">var_dump</span><span style="color: #007700">(</span><span style="color: #0000BB">getC</span><span style="color: #007700">());<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

   <div class="example-contents"><p>上の例の出力は以下となります。</p></div>
   <div class="example-contents screen">
<div class="annotation-interactive cdata"><pre>
object(C)#1 (0) {
}
</pre></div>
   </div>
  </div>

  <div class="example" id="example-7">
   <p><strong>例7 Null を許容する型宣言</strong></p>
   <div class="example-contents">
    <div class="annotation-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">class </span><span style="color: #0000BB">C </span><span style="color: #007700">{}<br /><br />function </span><span style="color: #0000BB">f</span><span style="color: #007700">(?</span><span style="color: #0000BB">C $c</span><span style="color: #007700">) {<br />    </span><span style="color: #0000BB">var_dump</span><span style="color: #007700">(</span><span style="color: #0000BB">$c</span><span style="color: #007700">);<br />}<br /><br /></span><span style="color: #0000BB">f</span><span style="color: #007700">(new </span><span style="color: #0000BB">C</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">f</span><span style="color: #007700">(</span><span style="color: #0000BB">null</span><span style="color: #007700">);<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

   <div class="example-contents"><p>上の例の出力は以下となります。</p></div>
   <div class="example-contents screen">
    <div class="annotation-interactive cdata"><pre>
object(C)#1 (0) {
}
NULL
</pre></div>
   </div>
  </div>

  <div class="example" id="example-8">
   <p><strong>例8 Null を許容する戻り値の型宣言</strong></p>
   <div class="example-contents">
    <div class="annotation-non-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">function </span><span style="color: #0000BB">get_item</span><span style="color: #007700">(): ?</span><span style="color: #0000BB">string </span><span style="color: #007700">{<br />    if (isset(</span><span style="color: #0000BB">$_GET</span><span style="color: #007700">[</span><span style="color: #DD0000">'item'</span><span style="color: #007700">])) {<br />        return </span><span style="color: #0000BB">$_GET</span><span style="color: #007700">[</span><span style="color: #DD0000">'item'</span><span style="color: #007700">];<br />    } else {<br />        return </span><span style="color: #0000BB">null</span><span style="color: #007700">;<br />    }<br />}<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

  </div>

  <div class="example" id="example-9">
   <p><strong>例9 クラスのプロパティで型宣言</strong></p>
   <div class="example-contents">
<div class="annotation-non-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">class </span><span style="color: #0000BB">User </span><span style="color: #007700">{<br />    public static </span><span style="color: #0000BB">string $foo </span><span style="color: #007700">= </span><span style="color: #DD0000">'foo'</span><span style="color: #007700">;<br /><br />    public </span><span style="color: #0000BB">int $id</span><span style="color: #007700">;<br />    public </span><span style="color: #0000BB">string $username</span><span style="color: #007700">;<br /><br />    public function </span><span style="color: #0000BB">__construct</span><span style="color: #007700">(</span><span style="color: #0000BB">int $id</span><span style="color: #007700">, </span><span style="color: #0000BB">string $username</span><span style="color: #007700">) {<br />        </span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">id </span><span style="color: #007700">= </span><span style="color: #0000BB">$id</span><span style="color: #007700">;<br />        </span><span style="color: #0000BB">$this</span><span style="color: #007700">-&gt;</span><span style="color: #0000BB">username </span><span style="color: #007700">= </span><span style="color: #0000BB">$username</span><span style="color: #007700">;<br />    }<br />}<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

  </div>
 </div>

  
 <div class="sect2" id="language.types.declarations.strict">
  <h3 class="title">厳密な型付け</h3>

  <p class="para">
   デフォルトでは、PHP は誤った型の値を
   可能であれば期待されたスカラー型の宣言に従うよう自動的に変換します(coercive モード)。
   たとえば、関数に <span class="type"><a href="language.types.integer.php" class="type int">int</a></span> が与えられたが、
   パラメータで文字列が期待されていた場合、文字列型の値を取得します。
  </p>

  <p class="para">
   ファイルごとに strict モードを有効にすることができます。
   strict モードでは、型宣言に正確に対応する値のみを受け入れ、
   それ以外の値の場合、<span class="classname"><a href="class.typeerror.php" class="classname">TypeError</a></span> がスローされます。
   このルールに関する唯一の例外は、<span class="type"><a href="language.types.integer.php" class="type int">int</a></span> の値が
   <span class="type"><a href="language.types.float.php" class="type float">float</a></span> 型の宣言に渡せることだけです。
  </p>

  <div class="warning"><strong class="warning">警告</strong>
   <p class="simpara">
    内部関数の中からの関数呼び出しは、
    <code class="literal">strict_types</code> 宣言の影響を受けません。
   </p>
  </div>

  <p class="para">
   strict モードを有効にするには、<a href="control-structures.declare.php" class="link"><code class="literal">declare</code></a> 文を <code class="literal">strict_types</code> 宣言と一緒に使います。 
  </p>

  <blockquote class="note"><p><strong class="note">注意</strong>: 
   <p class="para">
    厳密な型付けは、strict モードが有効になったファイルの
    <em>内部</em> から行われる関数呼び出しに適用されます。
    そのファイルで宣言された関数への呼び出しに対して適用されるわけではありません。
    厳密な型付けが有効になっていないファイルから、
    厳密な型付けが有効になっているファイルで定義された関数を呼び出した場合は、
    呼び出し側の好み(型の自動変換)が尊重され、値は型変換されます。
   </p>
  </p></blockquote>

  <blockquote class="note"><p><strong class="note">注意</strong>: 
   <p class="para">
    厳密な型付けは、スカラー型の宣言に対してのみ定義されます。
   </p>
  </p></blockquote>

  <div class="example" id="example-10">
   <p><strong>例10 引数の値に対する厳密な型付け</strong></p>
   <div class="example-contents">
<div class="annotation-non-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">declare(</span><span style="color: #0000BB">strict_types</span><span style="color: #007700">=</span><span style="color: #0000BB">1</span><span style="color: #007700">);<br /><br />function </span><span style="color: #0000BB">sum</span><span style="color: #007700">(</span><span style="color: #0000BB">int $a</span><span style="color: #007700">, </span><span style="color: #0000BB">int $b</span><span style="color: #007700">) {<br />    return </span><span style="color: #0000BB">$a </span><span style="color: #007700">+ </span><span style="color: #0000BB">$b</span><span style="color: #007700">;<br />}<br /><br /></span><span style="color: #0000BB">var_dump</span><span style="color: #007700">(</span><span style="color: #0000BB">sum</span><span style="color: #007700">(</span><span style="color: #0000BB">1</span><span style="color: #007700">, </span><span style="color: #0000BB">2</span><span style="color: #007700">));<br /></span><span style="color: #0000BB">var_dump</span><span style="color: #007700">(</span><span style="color: #0000BB">sum</span><span style="color: #007700">(</span><span style="color: #0000BB">1.5</span><span style="color: #007700">, </span><span style="color: #0000BB">2.5</span><span style="color: #007700">));<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

   <div class="example-contents"><p>上の例の PHP 8 での出力は、このようになります。:</p></div>
   <div class="example-contents screen">
<div class="annotation-interactive cdata"><pre>
int(3)

Fatal error: Uncaught TypeError: sum(): Argument #1 ($a) must be of type int, float given, called in - on line 9 and defined in -:4
Stack trace:
#0 -(9): sum(1.5, 2.5)
#1 {main}
  thrown in - on line 4
</pre></div>
   </div>
  </div>

  <div class="example" id="example-11">
   <p><strong>例11 引数の値に対する型の自動変換</strong></p>
   <div class="example-contents">
<div class="annotation-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">function </span><span style="color: #0000BB">sum</span><span style="color: #007700">(</span><span style="color: #0000BB">int $a</span><span style="color: #007700">, </span><span style="color: #0000BB">int $b</span><span style="color: #007700">) {<br />    return </span><span style="color: #0000BB">$a </span><span style="color: #007700">+ </span><span style="color: #0000BB">$b</span><span style="color: #007700">;<br />}<br /><br /></span><span style="color: #0000BB">var_dump</span><span style="color: #007700">(</span><span style="color: #0000BB">sum</span><span style="color: #007700">(</span><span style="color: #0000BB">1</span><span style="color: #007700">, </span><span style="color: #0000BB">2</span><span style="color: #007700">));<br /><br /></span><span style="color: #FF8000">// これらは、整数型に変換されます: 以下の出力を参照!<br /></span><span style="color: #0000BB">var_dump</span><span style="color: #007700">(</span><span style="color: #0000BB">sum</span><span style="color: #007700">(</span><span style="color: #0000BB">1.5</span><span style="color: #007700">, </span><span style="color: #0000BB">2.5</span><span style="color: #007700">));<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

   <div class="example-contents"><p>上の例の出力は以下となります。</p></div>
   <div class="example-contents screen">
<div class="annotation-interactive cdata"><pre>
int(3)
int(3)
</pre></div>
   </div>
  </div>

  <div class="example" id="example-12">
   <p><strong>例12 戻り値に対する厳密な型付け</strong></p>
   <div class="example-contents">
<div class="annotation-non-interactive phpcode"><code><span style="color: #000000"><span style="color: #0000BB">&lt;?php<br /></span><span style="color: #007700">declare(</span><span style="color: #0000BB">strict_types</span><span style="color: #007700">=</span><span style="color: #0000BB">1</span><span style="color: #007700">);<br /><br />function </span><span style="color: #0000BB">sum</span><span style="color: #007700">(</span><span style="color: #0000BB">$a</span><span style="color: #007700">, </span><span style="color: #0000BB">$b</span><span style="color: #007700">): </span><span style="color: #0000BB">int </span><span style="color: #007700">{<br />    return </span><span style="color: #0000BB">$a </span><span style="color: #007700">+ </span><span style="color: #0000BB">$b</span><span style="color: #007700">;<br />}<br /><br /></span><span style="color: #0000BB">var_dump</span><span style="color: #007700">(</span><span style="color: #0000BB">sum</span><span style="color: #007700">(</span><span style="color: #0000BB">1</span><span style="color: #007700">, </span><span style="color: #0000BB">2</span><span style="color: #007700">));<br /></span><span style="color: #0000BB">var_dump</span><span style="color: #007700">(</span><span style="color: #0000BB">sum</span><span style="color: #007700">(</span><span style="color: #0000BB">1</span><span style="color: #007700">, </span><span style="color: #0000BB">2.5</span><span style="color: #007700">));<br /></span><span style="color: #0000BB">?&gt;</span></span></code></div>
   </div>

   <div class="example-contents"><p>上の例の出力は以下となります。</p></div>
   <div class="example-contents screen">
<div class="annotation-interactive cdata"><pre>
int(3)

Fatal error: Uncaught TypeError: sum(): Return value must be of type int, float returned in -:5
Stack trace:
#0 -(9): sum(1, 2.5)
#1 {main}
  thrown in - on line 5
</pre></div>
   </div>
  </div>
 </div>

</div><?php manual_footer($setup); ?>