Last updated on
PHP で FizzBuzz問題を "条件分岐無し" でやってみよう
以前勉強会で扱ったネタで、 FizzBuzz問題を “if文を使わずに” PHPで実装する、ということをやりました。
https://speakerdeck.com/tacck/phpde-fizzbuzzwen-ti-woyatutemiyou-number-yuruwebzha-huang
この時は、 “オブジェクト指向” の考え方を織り交ぜながら進めていき、最終的にif文を消していくことで拡張性を高める、という流れにしていました。
一方で、三項演算子(?:
)は使っていて、完全に条件分岐を無くすことはできませんでした。
ということで、今回は一旦 “オブジェクト指向” の方を諦めて、完全に条件分岐を無くすコードを考えてみました。
念の為のFizzBuzz問題の説明
今更な感じですが、以下のルールを満たすプログラムを作ろう、というものです。
- 数字を1~100まで与えた時に、以下の条件に従って文字を出力する。
- 3の倍数の時に “Fizz” と出力
- 5の倍数の時に “Buzz” と出力
- 3の倍数かつ5の倍数(つまり15の倍数)の時に “FizzBuzz” と出力
- それ以外の場合には与えられた数値を出力
非常にシンプルで、そこそこに条件も織り交ぜられており、プログラミング言語の特性を掴んだり、実装方針の複雑さを考える時によく利用されるテーマですね。
条件分岐の無いFizzBuzzのコード
では、早速コードを見てみましょう。
<?php
// 基本となるタイプの定義
$fizzBuzzTypes = [
1 => 'Fizz',
2 => 'Buzz',
3 => 'FizzBuzz'
];
for ($num = 1; $num <= 100; $num++) {
// 要素0に対象の値を格納
$fizzBuzzTypes[0] = "$num";
// 対象の値を添字へ換算
// 5の倍数の剰余の有無を第二ビット、
// 3の倍数の剰余の有無を第一ビットとすることで、
// 添字の0~3を表現する。
$index = intval($num % 5 === 0) << 1 | intval($num % 3 === 0);
// 表示
echo $fizzBuzzTypes[$index] . PHP_EOL;
}
コード量はかなり少ないので、わかりやすいかと思います。
配列の $fizzBuzzTypes
に出力したい文字を格納しておき、値を添字に換算していくことで条件分岐を無くしています。
ビット演算を使って表現しているので、FizzBuzz以外に出力したい文字列が増えた場合などは、配列とビット演算の二箇所を修正しなければいけないので、拡張性は落ちますね。
また、良い方法が見つかれば記事にしてみたいと思います。