summaryrefslogtreecommitdiffhomepage
path: root/services/blog/content/posts/2022-08-27/php-conference-okinawa-code-golf.dj
blob: 5701fe4d3efb9b21b70ec5b093f6abc4fe00e9d5 (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
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
---
[article]
uuid = "bb71bb5d-361b-44cb-9753-81d14583d860"
title = "PHP カンファレンス沖縄で出題されたコードゴルフの問題を解いてみた"
description = "PHP カンファレンス沖縄の懇親会 LT で出題されたコードゴルフの問題を解いてみた。"
tags = [
  "conference",
  "php",
  "phpconokinawa",
]

[[article.revisions]]
date = "2022-08-27"
remark = "公開"
---
{#intro}
# はじめに

本日 [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

{#solution}
# 解

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

* コマンドライン引数の第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 ?? []);

?>]
```

{#techniques}
# 使用したテクニック

{#exponential-notation}
## 指数表記

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

{#shorten-loop}
## foreach や for の中身を1つの文に

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

{#omit-initialization}
## $r に初期値を入れない

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

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

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

{#put-text-outside-php-tag}
## PHP タグの外に文字列を置く

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

{#outro}
# おわりに

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