付録B:2018年度・定期レポートへのコメント(その2)
※ 以下のプログラムは、Julia 1.1向けに書き直した。
ネタもと
2018年12月、および、2019年1月の月間レポートでは、題材自由のレポートを課した。 うち、後者は、関数を定義することを必須の条件とした。
提出された学生レポートを少し改変して、以下に示す。
次の関数 dora1()
は、何かを描く関数の定義である。
function dora1()
ts=0:pi/18:2pi
xs=6cos.(ts)
ys=6sin.(ts)
plt.plot(xs,ys)
ts=0:pi/18:2pi
xs=1.2 .+ 1.2*cos.(ts)
ys=2.5 .+ 1.5*sin.(ts)
plt.plot(xs,ys,"k")
ts=0:pi/18:2pi
xs=-1.2 .+ 1.2cos.(ts)
ys= 2.5 .+ 1.5sin.(ts)
plt.plot(xs,ys,"k")
ts=0:pi/18:2pi
xs= 0.5cos.(ts)
ys=1.1 .+ 0.5sin.(ts)
plt.plot(xs,ys,"r")
ts=0:pi/18:2pi
xs= 5cos.(ts)
ys=-1.5 .+ 4.5sin.(ts)
plt.plot(xs,ys,"k")
ts=0:pi/18:2pi
xs=0.5 .+ 0.5cos.(ts)
ys=2.4 .+ 0.5sin.(ts)
plt.plot(xs,ys,"k")
ts=0:pi/18:2pi
xs=-0.5 .+ 0.5cos.(ts)
ys= 2.4 .+ 0.5sin.(ts)
plt.plot(xs,ys,"k")
xs=-2.3:0.5:2.3
plt.plot(xs, 1/2.5*xs.^2 .-4 , "k", label="s")
xs=-2:0.1:1
plt.plot(-4 .+ xs,-0.3*xs,"k")
plt.plot( 5 .+ xs, 0.3*xs,"k")
xs=-2:0.1:1
plt.plot(-4 .+xs,-2 .+0*xs,"k")
plt.plot( 5 .+xs,-2 .+0*xs,"k")
xs=-2:0.1:1
plt.plot(-4 .+ xs,-1.2 .+ -0.2*xs,"k")
plt.plot( 5 .+ xs,-1.2 .+ 0.2*xs,"k")
xs=-28:0.1:-5
plt.plot(0*xs,-5 .+ -0.2*xs,"k")
end
dora1 (generic function with 1 method)
この関数を呼出して、実行結果を示す。 何かのキャラクターの顔が描かれた。
using PyPlot
plt.axes().set_aspect("equal")
dora1()
plt.xlim(-7,7)
plt.ylim(-7,7)
この関数では、平面図形を描く曲線を各々設計した苦労の跡が偲ばれる。 作者の意図通りに動作し、大変結構である。
しかし、この関数を後から振り返ったときに、 各行の意図をすぐに汲み取るのは難しいだろう。 一部の数値だけ異なるが、同じようなプログラム片が並ぶのも、読みにくくしている。
下請け関数を定義する
さて、この顔は、楕円、直線、放物線の3つの図形から成り立っている。 これら3つの図形を描く関数を定義してみよう。
まず、$\left(x_c, y_c\right)$ を中心とし、$x$ 軸方向の広がりが $2a$、 $y$ 軸方向の広がりが $2b$ であるような楕円を描く関数を定義する。式で書くと $\left(\dfrac{x-x_{c}}{a}\right)^2 + \left(\dfrac{y-y_{c}}{b}\right)^2 = 1$ である。
関数の引数は、xc
, yc
, a
, b
と 色(またはスタイル)を示す文字列 c
である。 最後の引数 c
を省略したときは c="k"
(黒色) を既定値とする。
function draw_ellipse(xc,yc,a,b,c="k")
ts=0:pi/36:2pi
xs=xc .+ a*cos.(ts)
ys=yc .+ b*sin.(ts)
plt.plot(xs,ys,c)
end
draw_ellipse (generic function with 2 methods)
上の関数の動作を確認しよう。上に続けて
plt.axes().set_aspect("equal")
draw_ellipse(1,1,2,1,"g")
draw_ellipse(1,1,1,2,"r")
plt.xlim(-2,4)
plt.ylim(-2,4)
plt.axhline(1, color="k", lw=0.5)
plt.axvline(1, color="k", lw=0.5)
次に、2つの点 $(x_1, y_1)$ と $(x_2, y_2)$ とを結ぶ直線を描く関数を定義しよう。
function draw_line(x1,y1,x2,y2,c="k")
xs=[x1,x2]
ys=[y1,y2]
plt.plot(xs,ys,c)
end
draw_line (generic function with 2 methods)
上の関数の動作を確認しよう。上に続けて
plt.axes().set_aspect("equal")
draw_line(-1,-1,3,2, "b")
draw_line(-1, 3,3,0, "g")
plt.xlim(-2,4)
plt.ylim(-2,4)
plt.axhline(1, color="k", lw=0.5)
plt.axvline(1, color="k", lw=0.5)
最後に、 放物線 (2次関数) $y=ax^2+bx+c$ を、区間 $\left[x_1, x_2\right]$ の範囲で描く関数を定義しよう。
function draw_para(a,b,c, x1,x2, color="k")
xs=range(x1,x2,length=50)
ys=a*xs.^2 .+ b*xs .+ c
plt.plot(xs, ys, color)
end
draw_para (generic function with 2 methods)
上の関数の動作を確認しよう。上に続けて
plt.axes().set_aspect("equal")
draw_para( 1, 0,-1, -2,2, "b")
draw_para(-1, 0, 1, -2,2, "g")
plt.xlim(-3,3)
plt.ylim(-3,3)
plt.axhline(0, color="k", lw=0.5)
plt.axvline(0, color="k", lw=0.5)
元の関数を書き換える
これらの「下請け」関数を呼び出す形で、元の関数 dora1()
を書き直そう。 隣接する部分がまとまるように、行の順番を少し入れ替えて、コメントをつけた ( 元の描画順に意図があるなら、ご容赦願いたい )。
function dora2()
# face
draw_ellipse(0,0,6,6,"b")
# nose
draw_ellipse(0,1.1, 0.5, 0.5, "r" )
# gray line
draw_ellipse(0,-1.5, 5, 4.5 )
# eyes
draw_ellipse( 1.2,2.5, 1.2, 1.5 )
draw_ellipse(-1.2,2.5, 1.2, 1.5 )
draw_ellipse( 0.5,2.4, 0.5, 0.5 )
draw_ellipse(-0.5,2.4, 0.5, 0.5 )
# beard
draw_line(-6, 0.6, -3,-0.3 )
draw_line( 6, 0.3, 3,-0.6 )
draw_line(-6,-2,-3, -2 )
draw_line( 6,-2, 3, -2 )
draw_line(-6, -0.8, -3, -1.4 )
draw_line( 6, -1.0, 3, -1.6 )
# mouth
draw_line( 0, 0.6, 0, -4 )
draw_para(1/2.5, 0, -4, -2.3, 2.3 )
end
dora2 (generic function with 1 method)
実行してみよう。
using PyPlot
plt.axes().set_aspect("equal")
dora2()
plt.xlim(-7,7)
plt.ylim(-7,7)
リファクタリング
関数 dora1()
を dora2()
に書き直したように、 プログラムの内容を保ったまま、見通しをよくしたり、実行速度を改善する作業を、 リファクタリング (refactoring) という。
(プログラムも含めて)複数の要素 (component )が協力して働いて、 ある目的を達成するものを、システム (system)という。
システムを実装する (implement) 手法として、 ボトム・アップ的な手法 (bottom-up approach) 、 トップ・ダウン的な手法 (top-down approach) が知られている。
ボトム・アップ的な手法は、 下位の要素を作成してから (上の例では、関数 draw_ellipse
や draw_line
などを定義してから)、 それらの組合せで上位の要素 (システム)を作成する (上の例では関数 dora2()
を定義する) 手法である。
トップ・ダウン的な手法は、 先に、上位の要素 (システム) を作成してから (関数 dora2()
を定義してから)、 その下位の要素 (部品) を作成する (関数 draw_ellipse
などを定義する) 手法である。
プログラミングの初級段階では、ボトム・アップ的な手法が分かりやすいだろうが、 システムの成り立ちに習熟するにつれて、 トップ・ダウン的な手法が取れるようになるであろう。
システムの考え方では、 システムを構成する「要素」を、入力と出力の対応関係だけが決まっていて、 その中身は関知しない「ブラック・ボックス black box」とみなす。 しかしながら、現実のシステムでは、「要素」の「中身」を無視することができず、 上位のシステムの性能にも影響が及ぶ。
そのような「要素」と「要素」の界面 (interface) を上手に扱うことができる人こそ、 優れた工学者 (engineer) といえる。
諸君が優れた工学者になることを願って、この文の著者は対応しているつもりである。