summaryrefslogtreecommitdiffhomepage
path: root/vhosts/blog/content/posts/2022-08-27
diff options
context:
space:
mode:
Diffstat (limited to 'vhosts/blog/content/posts/2022-08-27')
-rw-r--r--vhosts/blog/content/posts/2022-08-27/php-conference-okinawa-code-golf.ndoc118
-rw-r--r--vhosts/blog/content/posts/2022-08-27/php-conference-okinawa-code-golf.xml122
2 files changed, 118 insertions, 122 deletions
diff --git a/vhosts/blog/content/posts/2022-08-27/php-conference-okinawa-code-golf.ndoc b/vhosts/blog/content/posts/2022-08-27/php-conference-okinawa-code-golf.ndoc
new file mode 100644
index 00000000..dae3d323
--- /dev/null
+++ b/vhosts/blog/content/posts/2022-08-27/php-conference-okinawa-code-golf.ndoc
@@ -0,0 +1,118 @@
+---
+[article]
+title = "PHP カンファレンス沖縄で出題されたコードゴルフの問題を解いてみた"
+description = "PHP カンファレンス沖縄の懇親会 LT で出題されたコードゴルフの問題を解いてみた。"
+tags = [
+ "conference",
+ "php",
+ "phpconokinawa",
+]
+
+[[article.revisions]]
+date = "2022-08-27"
+remark = "公開"
+---
+<article>
+ <section id="intro">
+ <h>はじめに</h>
+ <p>
+ 本日 <a href="https://phpcon.okinawa.jp/">PHP カンファレンス沖縄 2022</a> が開催された (らしい)。
+ </p>
+ <p>
+ カンファレンスには参加できなかったものの、懇親会の LT で出題されたコードゴルフの問題が Twitter に流れてきたので、解いてみた。
+ </p>
+ <ul>
+ <li>ツイート: <a href="https://twitter.com/m3m0r7/status/1563397620231712772">https://twitter.com/m3m0r7/status/1563397620231712772</a></li>
+ <li>スライド: <a href="https://speakerdeck.com/memory1994/php-conference-okinawa-2022-extra?slide=3">https://speakerdeck.com/memory1994/php-conference-okinawa-2022-extra?slide=3</a></li>
+ </ul>
+ </section>
+ <section id="solution">
+ <h>解</h>
+ <p>
+ 細かいレギュレーションは不明だったので、勝手に定めた。
+ </p>
+ <ul>
+ <li>コマンドライン引数の第1引数で受けとる</li>
+ <li>結果は標準出力に出す</li>
+ <li>コンマの直後にはスペースを1つ置く</li>
+ <li>末尾コンマは禁止</li>
+ <li>数字でないものは入ってこないものとする</li>
+ <li>負数は入ってこないものとする</li>
+ </ul>
+ <p>
+ 書いたものがこちら:
+ </p>
+ <codeblock language="php">
+ <![CDATA[
+ [<?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??[]);?>]
+ ]]>
+ </codeblock>
+ <p>
+ しめて 123 バイトとなった (末尾改行を含めずにカウント)。
+ </p>
+ <p>
+ こちらは改行とスペースを追加したバージョン:
+ </p>
+ <codeblock language="php">
+ <![CDATA[
+ [<?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 ?? []);
+
+ ?>]
+ ]]>
+ </codeblock>
+ </section>
+ <section id="techniques">
+ <h>使用したテクニック</h>
+ <section id="techniques--exponential-notation">
+ <h>指数表記</h>
+ <p>
+ 割と多くの言語のゴルフで使えるテクニック。
+ <code>e</code> を用いた指数表記で、大きな数を短く表す。
+ このコードでは <code>10000</code>、<code>5000</code>、<code>2000</code>、<code>1000</code> を指数表記している。
+ </p>
+ </section>
+ <section id="techniques--shorten-loop">
+ <h>foreach や for の中身を1つの文に</h>
+ <p>
+ <code>foreach</code>、<code>for</code>、<code>if</code> などの後ろには、
+ 通常 <code>{</code> を続けて複数の文を連ねるが、中身の文を1つにしてしまえば、<code>{</code> と <code>}</code> を省略できる。
+ C言語などでも使える。
+ </p>
+ </section>
+ <section id="techniques--omit-initialization">
+ <h>$r に初期値を入れない</h>
+ <p>
+ PHP では、<code>$r[] = ......</code> のような配列の末尾に追加する式を実行したとき、
+ <code>$r</code> が未定義だった場合は <code>$r</code> を勝手に定義して空の配列で初期化してくれる。
+ これを利用すると、<code>$r = [];</code> のような初期化が不要になる。
+ </p>
+ <p>
+ ただし、プログラムに 0 が渡されるとループを一度も回らないので、<code>$r</code> が未定義になってしまい、
+ <code>implode()</code> に渡すところでエラーになる。
+ それを防ぐために <code>$r ?? []</code> を使っている。
+ </p>
+ <p>
+ もし 0 が渡されたケースを無視するなら、これが不要になるので 4 バイト縮む。
+ </p>
+ </section>
+ <section id="techniques--put-text-outside-php-tag">
+ <h>PHP タグの外に文字列を置く</h>
+ <p>
+ PHP では、<code>&lt;?php</code> <code>?&gt;</code> で囲われた部分の外側にある文字列は、そのまま出力される。
+ 今回のケースでは、先頭と末尾に必ず <code>[</code> と <code>]</code> を出力するので、そのまま書いてやればよい。
+ </p>
+ </section>
+ </section>
+ <section id="outro">
+ <h>おわりに</h>
+ <p>
+ 最後になりましたが、<a href="https://twitter.com/m3m0r7">めもりー</a>さん、楽しい問題をありがとうございました。
+ </p>
+ </section>
+</article>
diff --git a/vhosts/blog/content/posts/2022-08-27/php-conference-okinawa-code-golf.xml b/vhosts/blog/content/posts/2022-08-27/php-conference-okinawa-code-golf.xml
deleted file mode 100644
index 4cbea19b..00000000
--- a/vhosts/blog/content/posts/2022-08-27/php-conference-okinawa-code-golf.xml
+++ /dev/null
@@ -1,122 +0,0 @@
-<?xml version="1.0" encoding="UTF-8"?>
-<article xmlns="http://docbook.org/ns/docbook" xmlns:xl="http://www.w3.org/1999/xlink" version="5.0">
- <info>
- <title>PHP カンファレンス沖縄で出題されたコードゴルフの問題を解いてみた</title>
- <abstract>
- PHP カンファレンス沖縄の懇親会 LT で出題されたコードゴルフの問題を解いてみた。
- </abstract>
- <keywordset>
- <keyword>conference</keyword>
- <keyword>php</keyword>
- <keyword>phpconokinawa</keyword>
- </keywordset>
- <revhistory>
- <revision>
- <date>2022-08-27</date>
- <revremark>公開</revremark>
- </revision>
- </revhistory>
- </info>
- <section xml:id="intro">
- <title>はじめに</title>
- <para>
- 本日 <link xl:href="https://phpcon.okinawa.jp/">PHP カンファレンス沖縄 2022</link> が開催された (らしい)。
- </para>
- <para>
- カンファレンスには参加できなかったものの、懇親会の LT で出題されたコードゴルフの問題が Twitter に流れてきたので、解いてみた。
- </para>
- <itemizedlist>
- <listitem>ツイート: <link xl:href="https://twitter.com/m3m0r7/status/1563397620231712772">https://twitter.com/m3m0r7/status/1563397620231712772</link></listitem>
- <listitem>スライド: <link xl:href="https://speakerdeck.com/memory1994/php-conference-okinawa-2022-extra?slide=3">https://speakerdeck.com/memory1994/php-conference-okinawa-2022-extra?slide=3</link></listitem>
- </itemizedlist>
- </section>
- <section xml:id="solution">
- <title>解</title>
- <para>
- 細かいレギュレーションは不明だったので、勝手に定めた。
- </para>
- <itemizedlist>
- <listitem>コマンドライン引数の第1引数で受けとる</listitem>
- <listitem>結果は標準出力に出す</listitem>
- <listitem>コンマの直後にはスペースを1つ置く</listitem>
- <listitem>末尾コンマは禁止</listitem>
- <listitem>数字でないものは入ってこないものとする</listitem>
- <listitem>負数は入ってこないものとする</listitem>
- </itemizedlist>
- <para>
- 書いたものがこちら:
- </para>
- <programlisting language="php" linenumbering="unnumbered">
- <![CDATA[
- [<?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??[]);?>]
- ]]>
- </programlisting>
- <para>
- しめて 123 バイトとなった (末尾改行を含めずにカウント)。
- </para>
- <para>
- こちらは改行とスペースを追加したバージョン:
- </para>
- <programlisting language="php" linenumbering="unnumbered">
- <![CDATA[
- [<?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 ?? []);
-
- ?>]
- ]]>
- </programlisting>
- </section>
- <section xml:id="techniques">
- <title>使用したテクニック</title>
- <section xml:id="techniques--exponential-notation">
- <title>指数表記</title>
- <para>
- 割と多くの言語のゴルフで使えるテクニック。
- <literal>e</literal> を用いた指数表記で、大きな数を短く表す。
- このコードでは <literal>10000</literal>、<literal>5000</literal>、<literal>2000</literal>、<literal>1000</literal> を指数表記している。
- </para>
- </section>
- <section xml:id="techniques--shorten-loop">
- <title>foreach や for の中身を1つの文に</title>
- <para>
- <literal>foreach</literal>、<literal>for</literal>、<literal>if</literal> などの後ろには、
- 通常 <literal>{</literal> を続けて複数の文を連ねるが、中身の文を1つにしてしまえば、<literal>{</literal> と <literal>}</literal> を省略できる。
- C言語などでも使える。
- </para>
- </section>
- <section xml:id="techniques--omit-initialization">
- <title>$r に初期値を入れない</title>
- <para>
- PHP では、<literal>$r[] = ......</literal> のような配列の末尾に追加する式を実行したとき、
- <literal>$r</literal> が未定義だった場合は <literal>$r</literal> を勝手に定義して空の配列で初期化してくれる。
- これを利用すると、<literal>$r = [];</literal> のような初期化が不要になる。
- </para>
- <para>
- ただし、プログラムに 0 が渡されるとループを一度も回らないので、<literal>$r</literal> が未定義になってしまい、
- <literal>implode()</literal> に渡すところでエラーになる。
- それを防ぐために <literal>$r ?? []</literal> を使っている。
- </para>
- <para>
- もし 0 が渡されたケースを無視するなら、これが不要になるので 4 バイト縮む。
- </para>
- </section>
- <section xml:id="techniques--put-text-outside-php-tag">
- <title>PHP タグの外に文字列を置く</title>
- <para>
- PHP では、<literal>&lt;?php</literal> <literal>?&gt;</literal> で囲われた部分の外側にある文字列は、そのまま出力される。
- 今回のケースでは、先頭と末尾に必ず <literal>[</literal> と <literal>]</literal> を出力するので、そのまま書いてやればよい。
- </para>
- </section>
- </section>
- <section xml:id="outro">
- <title>おわりに</title>
- <para>
- 最後になりましたが、<link xl:href="https://twitter.com/m3m0r7">めもりー</link>さん、楽しい問題をありがとうございました。
- </para>
- </section>
-</article>