aboutsummaryrefslogtreecommitdiffhomepage
path: root/content/posts/2022-08-27/php-conference-okinawa-code-golf.md
blob: 6042b651ff0da2cbf6f2248112c37208bcf78f18 (plain)
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
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
---
title: "PHP カンファレンス沖縄で出題されたコードゴルフの問題を解いてみた"
date: 2022-08-27T18:55:28+09:00
tags: ["conference", "php", "phpcon"]
summary: |
    PHP カンファレンス沖縄の懇親会 LT で出題されたコードゴルフの問題を解いてみた。
changelog:
    2022-08-27: 公開
---

# はじめに

本日 [PHP カンファレンス沖縄 2022](https://phpcon.okinawa.jp/) が開催された (らしい)。

カンファレンスには参加できなかったものの、懇親会の LT で出題されたコードゴルフの問題が Twitter に流れてきたので、解いてみた。

ツイート: https://twitter.com/m3m0r7/status/1563397620231712772 \
スライド: https://speakerdeck.com/memory1994/php-conference-okinawa-2022-extra?slide=3


# 解

細かいレギュレーションは不明だったので、勝手に定めた。

* コマンドライン引数の第1引数で受けとる
* 結果は標準出力に出す
* コンマの直後にはスペースを1つ置く
* 末尾コンマは禁止
* 数字でないものは入ってこないものとする
* 負数は入ってこないものとする

書いたものがこちら:

```php
[<?php $n=$argv[1];foreach([1e4,5e3,2e3,1e3,500,100,50,10,5,1]as$x)for(;$n>=$x;$n-=$x)$r[]=$x;echo implode(', ',$r??[]);?>]
```

しめて 123 バイトとなった (末尾改行を含めずにカウント)。

こちらは改行とスペースを追加したバージョン:

```php
[<?php

$n = $argv[1];
foreach ([1e4, 5e3, 2e3, 1e3, 500, 100, 50, 10, 5, 1] as $x)
  for (; $n >= $x; $n -= $x)
    $r[] = $x;
echo implode(', ', $r ?? []);

?>]
```

# 使用したテクニック

## 指数表記

割と多くの言語のゴルフで使えるテクニック。`e` を用いた指数表記で、大きな数を短く表す。このコードでは `10000`、`5000`、`2000`、`1000` を指数表記している。

## foreach や for の中身を1つの文に

`foreach`、`for`、`if` などの後ろには、通常 `{` を続けて複数の文を連ねるが、中身の文を1つにしてしまえば、`{` と `}` を省略できる。C言語などでも使える。

## $r に初期値を入れない

PHP では、`$r[] = ...` のような配列の末尾に追加する式を実行したとき、`$r` が未定義だった場合は `$r` を勝手に定義して空の配列で初期化してくれる。これを利用すると、`$r = [];` のような初期化が不要になる。

ただし、プログラムに 0 が渡されるとループを一度も回らないので、`$r` が未定義になってしまい、`implode()` に渡すところでエラーになる。それを防ぐために `$r ?? []` を使っている。

もし 0 が渡されたケースを無視するなら、これが不要になるので 4 バイト縮む。

## PHP タグの外に文字列を置く

PHP では、`<?php` `?>` で囲われた部分の外側にある文字列は、そのまま出力される。今回のケースでは、先頭と末尾に必ず `[` と `]` を出力するので、そのまま書いてやればよい。


# おわりに

最後になりましたが、[めもりー](https://twitter.com/m3m0r7) さん、楽しい問題をありがとうございました。