言語の特徴

開発環境

Windows

Perlのインスコ

ActiveStateからActivePerlが無償配布されています。これを使うといいでしょう。インストール先(perl.exeがおいてあるところ/デフォルトはc:\usr\bin\?)を環境変数PATHに追加しておきましょう。パスが通ったことを確認するには

perl -v

としてThis is perl,...と表示されればOKです。

ソースの編集方法

メモ帳でもいいです。テキストが編集できるものなら何でもいいですが、文字コード・改行コードが指定できるものの方が何かと役に立つかもしれません。今のところ定番のIDEってのはないようで、みんな思い思いのものを使っていることが多いです。ActiveStateはVisual Perlってのを開発してたらしいんだけど…開発は打ち切られたらしい ""Engineering support for Visual Perl was discontinued as of December 15, 2005. Installation and configuration support is no longer available. とある。 ""訳:Visual Perlの設計サポートは2005年12月15日をもって打ち切られました。インストールと環境設定のサポートはもう利用できません。 他にも窓の杜Vectorでフリーのものはたくさんあるので探してみてください。

モジュールとCPAN

CPAN(Comprehensive Perl Archive Network)には、使えるモジュールがたくさんあるので、まずは探すべし! たいていのことはモジュールの組み合わせでできるはずです。標準でインストールされるモジュールも活用! 真面目に書くときはuse strictしておけ!

VIP的Perl用語解説

ワンライナー
プログラムをソースファイルにかかずコマンドライン上に直接一行で書くこと。通常より複雑な技術を必要とするため、プログラムがわからなくても「それはワンライナーいけるな」と言えばその筋の人から尊敬のまなざしでみられる。プログラム書けなくてもとりあえず言えおwww
シュワルツ変換
並び替えを効率よくやるための記法。まず相手をたたみこむために使う。「シュワルツ変換つかえよwww」と言えばいい。その筋の人には伝わる。でっ?ていう
ラクダ本
オライリー・ジャパンから出版されているLarry Wall著の本。表紙がラクダの絵だからラクダ本。Vipperに限らずPerlerは知ってる人が多い。

プログラム例

初心者用課題に載っているものです。

Hello World

#!perl
print "Hello World!";

(注意:#!行は環境依存) をhello.plとして保存し、

perl hello.pl

として実行すると

Hello World!

と表示される。

解説

1: #!perl

perl.exeというperlインタプリタを呼び出すコマンド。お約束。Perlのインストール環境によって異なる。

2: print "Hello World!";

"Hello World!"という文字列を出力するための行。printと最初の"の間にはスペースを入れることが多い。行末に;(セミコロン)を忘れずに。Rubyと違って(ブロック末を除き)省略できない。

FizzBuzz

print+(Fizz)[$_%3].(Buzz)[$_%5]||$_,$/for 1..100

・上のコードについて

これらを踏まえてもう一度コードを見てみると、

foreach $_ (1..100) {
	my $str = ("Fizz", "", "")[$_ % 3].("Buzz", "", "", "", "")[$_ % 5];
	$str = $_ if($str eq '');
	print($str."\n")
}

とほぼ同じ意味になるのが分かるだろう。

平方根

#!/usr/local/bin/perl
use strict;

my $MAX = 30; # 小数点以下30位まで求める。

my $num = <>;
$num -=0;
my $flg = 0;
if($num<0){
	$flg = 1;
	$num *= -1;
}
my $cnt = 0;
1while($cnt++**2<$num);

if(--$cnt**2 == $num){
	print $cnt;
	exit;
}
print --$cnt.".";

$num = $num - $cnt**2;
$cnt *= 2;
while($MAX--){
	$num *=100;
	my $r = 0;
	1while($num>(++$r+$cnt*10)*$r);
	if($num == ($r+$cnt*10)*$r){
		print $r;
		last;
	}
	$num -= (--$r+$cnt*10)*$r;
	$cnt = $cnt*10+$r*2;
	print $r;
}
$flg && print "i";
print$/;

虚数、小数にも対応 誰か解説してくれ。

素数判定

#!perl
$n=100;
for($i=2;$i<=sqrt($n);$i++){
    if($n%$i==0){
        print "n is not a prime number.";
        exit;
    }
}
print "n is a prime number.";

解説

2: $n=100;

変数nに100を代入する」という意味です。右の値を左に代入します。100って数字に何かを代入するのは不自然なので逆に書いてはいけません(エラーになります)。変数名の前には$(ダラー)を、代入文の後ろには;(セミコロン)を忘れないように。

3: for($i=2;$i<=sqrt($n);$i++){

プログラミング初心者の(多分)最初の壁であるループ構文です。$iは2から√n(より小さい最大の自然数)まで1ずつ増えながら{〜}の中身を実行していきます。実行順序はfor(A;B;C){D}に対してABCDBCDBCDBCDBのような感じです。

$i++

ってのは変数$iの値を1増やすためによく使われます。ここでは$i++と書いても++$iと書いても構いませんが、実際には全く同じではないです。

4: if($n%$i==0){

$n%$iは「$nを$iで割った余り」を意味しまう(数学の言葉あるいはBASICなら$n mod $i)。「数値が等しいか否か」を判断するには演算子==を使います(文字列にはeqを使ってください。また、数値には<=>、文字列にはcmpというのも用意されています)。数値の比較を行う際には=は2つ並べなければなりません。。1つだと代入文になってしまいます。()内がなら{}内を実行し、なら華麗にスルーします。

5: print "n is not a prime number.";

割り切れたということは素数じゃないということなのでその旨表示します。

6: exit;

プログラムそのものを終了する命令です。今回はループを抜けた場合は「割り切れなかった」と判断して「素数だ」と表示するように組んだので、割り切れたら即座に「素数じゃない」と判断して終了するようにします。

別解例

$n = 101;
for(2 .. int sqrt $n){
    print "$n is not a prime number.\n" and exit unless($n % $_);
}
print "$n is a prime number.\n";

調べる数が$nです。2~√nまで特殊変数$_に代入していき、$nが$_で割り切れたら「素数でない」と表示して終了(exit)しています。ループを最後まで抜ければ素数であったのでそう表示しています。

コマンドラインから引数を取るバージョン

$n = shift;
for my $i (2 .. sqrt $n){
    print "$n is not a prime number.\n" and exit unless($n % $i);
}
print "$n is a prime number.\n";

実行

perl test.pl 101

1行目のshiftで引数を変数に格納します。他は前の例とだいたい同じです。2行目の$iには、(2 .. sqrt $n)が順番に入ります。2行目のsqrtの前のintは会っても無くてもいいです。ちなみに、もう一回shiftすると、次の引数が入ります。(今回は実行コマンドに引数が一つしかないので、何も入らない。)

エラトステネスのふるい

#!perl
$N=100;
@prime = ();
@zahlen = 2 .. $N # ステップ1


do{
    push(@prime,$zahlen[0]);	# ステップ2
    for($j=$#zahlen;$j>=0;$j--){ # ステップ3(for文全体)
        if($zahlen[$j]%$prime[$#prime]==0){
            splice(@zahlen,$j,1);
        }
    }
} while ($zahlen[$#zahlen]>$prime[$#prime]**2);	# ステップ4判定
foreach $k (@prime){	#以下表示部分
    print "$k ";
}
foreach (@zahlen){
    print "$_ ";
}

解説

do〜whileを利用した例。エラトステネスの篩-wikipediaに基づいたコーディング。

3: @prime = ();

@primeには8行目でいきなりpush(要素の末尾に追加)するから空配列を用意しておきます。

*ステップ1

以下説明では$N=100の場合を使っていくことにします。$zahlen[0]=2,$zahlen[1]=3,...,$zahlen[98]=100がそれぞれ代入されます。

*ステップ2

7: do{ 〜 14| } while ($zahlen[$#zahlen]>$prime[$#prime]**2);

ループ記法の1つ、do〜while文。while(A){B}はABAB...ABAで終わるのに対し、do{B}while(A);はBABA...BAとなるところがポイント(最初のAの有無)。やりたい処理に応じてループ記法を使いわけてください。

8: push(@prime,$zahlen[0]);

pushってのは配列の末尾に配列を足すことができる命令です。足す方は今回のようにスカラーでも大丈夫。@prime = (2,3,5),$zahlen[0]=7のときにこれが実行されると@prime = (2,3,5,7),$zahlen[0]=7になります。

*ステップ3

ここでは配列要素を削除していきますが、前からその処理をやると詰めが発生するから添字の扱いが(個人的に)面倒です。ここでは簡単のため@zahlenの後ろから見ていきます。

9: for($j=$#zahlen;$j>=0;$j--){

$#zahlenは@zahlenの最後の添字(要素数-1)を指しています。ここから0番目の要素(そしてこれは必ず除かれる)までループしていきます。

9: for $j(reverse 0 .. $#zahlen){

とも書けます。

10: if($zahlen[$j]%$prime[$#prime]==0){

ステップ2で最後に加えた数は@primeの末尾にいます。$#primeは@primeの最後の添字だったので、末尾の要素は$prime[$#prime]となります。

11: splice(@zahlen,$j,1);

spliceは配列,削除を始める添字,削除する項数を受け取って削除してくれます。spliceだけでもいくつか書き方があり、これはその一例です(詳細はperldoc -f spliceを見てください)。今回は割り切れた添字$j項目の1個だけを削除するからこのように書きます。

*ステップ4

このステップはwhile文ですが、ここまで分かっていれば説明の必要はないと思います。表示の方を説明します。

15: foreach $k (@prime){

これはPerl独特の記法です。@primeの要素を$prime[0]~$prime[$#prime]まで順に$kに代入してループ内に入ります。大体次と似たように動きます(全く同じではないです)。

for($k=0;$k<=$#prime;$k++){

後者のコーディングをしたときのブロック内で$prime[$k]にあたるものが、上のコーディングでは$kとして使えるということです。読みやすさを重視するときに便利なループ処理です。前者は必ず配列の全要素についてループしようとするから無限ループに落ちる心配は(自分の意思で行わない限り)ありません。

for($k=0;$k<=$#prime;$k++){
for $k (0 .. $#prime){
foreach $k (0 .. $#prime){

は同じ動きです。

18: foreach (@zahlen){

みたいに添字を省略した場合はデフォルトの特殊変数$_にそのループの値が入ります。

19: print "$_ ";

こんな感じ

別解例

$n = 100;
@prime = ();  # 素数リスト
@zahlen = (2 .. $n);  # 探索リスト
do{
    push @prime, $zahlen[0];  # ステップ2
    @zahlen = grep { $_ % $zahlen[0] } @zahlen;  # ステップ3
} while ($zahlen[-1] > $prime[-1]**2);  # ステップ4
print "@prime @zahlen\n";

閏年判定

#!perl
$YEAR = 2007;
if(($YEAR%4==0&&$YEAR%100!=0)||$YEAR%400==0){
    print("$YEAR is a leap year.");
}else{
    print("$YEAR is not a leap year.");
}

解説

3: if(($YEAR%4==0&&$YEAR%100!=0)||$YEAR%400==0){

この行が全てです。このプログラムの価値はここだけにあります。条件判断1&&条件判断2は両方成り立てば1,成り立たない方があれば0を返します。条件判断1||条件判断2は少なくとも一方が成り立てば1,さもなくば0を返します。何でこれで大丈夫なのか、閏年-wikipediaを参照してじっくり考えてみてください。

転置行列

#!perl
@a = ( [ 1 , 2 , 3 ],
       [ 4 , 5 , 6 ],
       [ 7 , 8 , 9 ],
       [10 ,11 , 12] );
$m = $#a;
$n = @{$a[0]}-1;
for($i=0;$i<=$m;$i++){
    for($j=0;$j<=$n;$j++){
        $b[$j][$i] = $a[$i][$j];
    }
}
for($j=0;$j<=$n;$j++){
    for($i=0;$i<=$m;$i++){
        print "$b[$j][$i] ";
    }
    print "\n";
}

二次元配列を使用した例。

別解例

モジュールを使用した例。強度のMじゃなければこっちの別解(モジュール)使え。

use Math::Matrix;
$a = new Math::Matrix(
    [  1,  2,  3],
    [  4,  5,  6],
    [  7,  8,  9],
    [ 10, 11, 12] );
$b = $a->transpose;
$b->print;

最後の2行は

print $a->transpose;

ってやってもOKで、$bにオブジェクトをコピーしない分だけ早そうです。

線形合同法

#!perl
# X[n+1] = (A*X[n]+B) mod M
$A = 997;
$B = 1;
$M = 2 ** 16;
$n = 100;
$x[0] = 12345; # 配列@xの要素は整数 1つ1つの乱数は$x[$i]/$Mで得られる
$s=0;
for($i=0;$i<$n-1;$i++){
    $x[$i+1] = (($A*$x[$i]+$B) % $M);
}
for($i=0;$i<$n;$i++){
    $s += $x[$i]/$M;
}
for($i=0;$i<$n;$i++){
    printf ("%0.4f ",$x[$i]/$M);
    print "\n" if($i%10==9);
}
print "Average : " . $s/$n;

解説

5: $M = 2 ** 16;

ここでは**演算子を使いました。2 ** 16は「2の16乗(2×2×…×2=65536)」の意味です。(POSIXモジュールにpowが用意されています。)

9: for($i=0;$i<$n-1;$i++){

9: for $i(0 .. $n-2){

とも書ける。C言語とかでガリガリ書いてた人以外は、こっちの方が直感的に判りやすいと思う。

13: $s += $x[$i];

これはメジャーなプログラミング言語ではよく見られる略記法です。「$sに配列@xの$i番目を加える」操作だから、

$s = $s + $x[$i];

と書いても同じことです #でもこれ、左右が絶対等しくないからちょっと気持ち悪い式だと書いてる人は思うんだお。前者をおすすめします。

16: printf ("%5.d ",$x[$i*10+$j]);

ここは表示するだけだから大したところではないけど、一応解説を加えます。printfってのは元々言語の言葉です。5桁分のスペースをとって整数を表示して空白を出力します。Perlでもところどころで(もとい、かなりの部分で)C言語の文が使えるから、Cプログラマの人はこっちを使っても大丈夫です(Perlの記法を使った方が見た目はすっきりすることが多いと思います)。この文の詳細はC言語の項に任せるお。

17: print "\n" if($i%10==9);

見栄えのために10個ごとに改行を入れています。添字は0から始まるから、$i=9,19,29,...99のときに改行します。これらの数字の特徴は1の位が9⇔10で割ったら9余るということです。こんな風にif文を後ろに書く記法もあります。

if($i%10==9){
    print "\n";
}

と書くのと同じです。

ニュートン法で平方根を求める

#!perl

$n = 3; # $nの平方根を求める
$m = 20; # ループ回数
$x0 = 1; # 初期値

@r = ();
$r[0] = $x0;

for($i = 0; $i < $m ; $i ++ ){
    $r[$i+1] = root($r[$i]);
}
print "$r[$m-1]";

sub root{
    my($rn,$rnn);
    $rn = $_[0];
    $rnn = ($rn**2+$n)/(2*$rn);
    return $rnn;
}
exit;

解説

7: @r = ();

これは@rという空の配列を用意するお。8行目で初期値をセットしておきます。

10-12: for($i=0;$i<$m;$i++){ $r[$i+1]=root($r[$i]); }

ニュートン法を使って計算する部分は後述のrootっていうサブルーチンに○投げしてあります。とりあえず考えなくていいです。ニュートン法は数列(現行教育課程で高2)の隣接2項間漸化式を使う方法です。でも数学できなくても気にしないでください。$i番目が決まったら$i+1番目が決まるってことが分かってれば大丈夫。そんなわけで、その規則をrootサブルーチンに書いてあります。今回は$iが$mに近いほど(最大で$m-1)真の値に近づく仕組み。

13: print "$r[$m-1]";

これが計算したなかで一番近い値となる。$n=3,$m=20の場合小数点以下13桁完璧でした。

15: sub root{

言語などの「関数」にあたるのがPerlの「サブルーチン」です。多くの場合、何か値を受け取って、それに応じて何らかの値を返却します。受け取ったものは特殊な配列@_に格納されています。返却するときはreturnに引数を取ればいいです。

16: my($rn,$rnn);

変数のネーミングがちょっと気に入らないけど、ここでは$rnはn番目、$rnnはn+1番目の値を入れることにしました。myってのはその中身がローカル変数(このサブルーチンの中でのみ有効)だということを宣言するものです。

17: $rn = $_[0];

$_[0]にはn番目の値が引数として渡されているので、それをとりあえず$rnに格納しておきます。

18: $rnn = ($rn**2+$n)/(2*$rn);

ここがニュートン法の計算式で、なんでこうなるかは数学の範疇です。ニュートン法-Wikipediaの(1.1)式を理解すればOKです

数当てゲーム

#!/usr/local/bin/perl -W
use strict;
local $/;
print eval <DATA>;
__DATA__
$, = $\ = $/ = "\n";
my $flg = 1;
while($flg){
	my $number = 1000 + int rand 9000;
	while(<>){
		$number==int&&last;
		print$_>$number?'bigger...':'smaller...';
	}
	print 'hit!!','again? y/n';
	while(<>){
		/^y$/i&&last;
		/^n$/i&&$flg--&&last;
	}
}
'good bye';

何この邪悪なコード。ふざけてるの?

B:
$Ans = int rand 9000;
A:
print"入力=>";$_=<>;
printf"入力は答えより%sです\n",$_>$Ans?"大きい":"小さい";
&more if/$Ans/;goto A;
sub more{
print"正解です=>$Ans\n\nもう一回やる?y/n=>";
$_=<>;goto B if/y/;exit if/n/;
}

配列いじり

print shift @ARGV;
print " ";
print "0 " while shift @ARGV;

実行例

perl test1.pl 3 5 2 4 2

結果例

3 0 0 0 0 

↑"配列"いじりじゃないうえに、引数に0が含まれるとそこでループを抜けてしまうため、不正解。

#!/usr/local/bin/perl
use strict;
$, = ", ";
$\ = $/;
my @aa = (3,6,2,4,2);
print @aa;
splice @aa,1,$#aa,(0)x$#aa;
print @aa;

Base64

use MIME::Base64;

$encoded = encode_base64('vipper');
$decoded = decode_base64($encoded);

CPANモジュールそのまんま標準で入ってるモジュールと思われるので、追加インストール不要です。

Caesar暗号解読

use strict;

my $words = "qdq-gi.q-a ziatmxxitmdqibtqi-ustbi ri.qmoqrcxi.qbubu zir -ibtqi-qp-qaai ripmymsqkir -ibtqi-qy dmxi ri.cnxuoi rruoumxakir -ibtqiqzmobyqzbkii-q.qmxi -imyqzpyqzbi rixmeaki -puzmzoqai -i-qscxmbu zaimzpir -i btq-iymbbq-a;iz -iatmxximzgi.q-a zinqiuzimzgiemgipuao-uyuzmbqpimsmuzabir -ia. za -uzsiacotiimi.qbubu zj"; #問題文を保存
my @answers=();

for (1 .. 30){ #全部で30文字なので30回ループする
  $words =~ tr/abcdefghijklmnopqrstuvwxyz .,-/-abcdefghijklmnopqrstuvwxyz .,/;#1文字ずらす
  push (@answers, $words) if $words =~ m/person/; #personが含まれている場合は答えを保存する。
}
while (@answers){
  print shift @answers;
  print "\n";
}
#答えが1通りであることがわかってる場合は、8行目のpushのところで、printして、exit しちゃえばOK。

別解

#!/usr/local/bin/perl
use strict;
$\ = $/;
$_ = "qdq-gi.q-a ziatmxxitmdqibtqi-ustbi ri.qmoqrcxi.qbubu zir -ibtqi-qp-qaai ripmymsqkir -ibtqi-qy dmxi ri.cnxuoi rruoumxakir -ibtqiqzmobyqzbkii-q.qmxi -imyqzpyqzbi rixmeaki -puzmzoqai -i-qscxmbu zaimzpir -i btq-iymbbq-a;iz -iatmxximzgi.q-a zinqiuzimzgiemgipuao-uyuzmbqpimsmuzabir -ia. za -uzsiacotiimi.qbubu zj";
y/abcdefghijklmnopqrstuvwxyz .,\-/bcdefghijklmnopqrstuvwxyz .,\-a/while(!/person/);
print;

解が複数あると仮定するなら、

#!/usr/local/bin/perl
use strict;
$\ = $/;
my $c = "qdq-gi.q-a ziatmxxitmdqibtqi-ustbi ri.qmoqrcxi.qbubu zir -ibtqi-qp-qaai ripmymsqkir -ibtqi-qy dmxi ri.cnxuoi rruoumxakir -ibtqiqzmobyqzbkii-q.qmxi -imyqzpyqzbi rixmeaki -puzmzoqai -i-qscxmbu zaimzpir -i btq-iymbbq-a;iz -iatmxximzgi.q-a zinqiuzimzgiemgipuao-uyuzmbqpimsmuzabir -ia. za -uzsiacotiimi.qbubu zj";
($_ = $c) && y/abcdefghijklmnopqrstuvwxyz .,\-/bcdefghijklmnopqrstuvwxyz .,\-a/ && ($c = $_) && /person/ && print for(0..29);

フィボナッチ数列

Math:Bigを見ろ。

真と偽

とほほのperl入門(概要編) - ◆ 真(true)と偽(false)

スタック

#!/usr/local/bin/perl -W
use strict;
my %calc = (
	'+' => sub{pop(@{$_[0]})+pop@{$_[0]};},
	'-' => sub{splice(@{$_[0]},$#{$_[0]}-1,1)-pop@{$_[0]};},
	'*' => sub{pop(@{$_[0]})*pop@{$_[0]};},
	'/' => sub{splice(@{$_[0]},$#{$_[0]}-1,1)/pop@{$_[0]};},
);
my @stack = ();
defined$calc{$_}?push@stack,&{$calc{$_}}(\@stack):sub{
	my @x = @_;
	$x[0] =~ y/[0-9\-]//c;
	push@{$x[1]}, int$x[0] if $x[0] eq int$x[0];
}->($_,\@stack)for@ARGV;
print$stack[0];

参考リンク

Perl-users.jp - 日本のPerlユーザのためのハブサイト
モダンなPerl : 日本のPerlユーザのためのハブサイト
perl-mongers.org
モダンなPerl : OpenIDでログイン出来るPerlユーザblog。Perler な人達が初心者向けの情報や Tips をブログ形式で。
Perl「言語」初心者がモダンなPerlを高速に学ぶ方法 - dann@webdev - dann's portal
plaggerのソースを読もう
プログラミング言語Perl 詳細リファレンス - Perl入門~サンプルコードによるPerl入門~
詳細なリファレンス、サンプルコード付き
Perlという言語について - Perl入門~サンプルコードによるPerl入門~
Perlのバージョン5以降であればPerlできれいにプログラムを書くことができます。
Perlプログラマーの皆さん - Perl入門~サンプルコードによるPerl入門~
Perlをやっている人のブログへのリンク

トップ   編集 凍結 差分 履歴 添付 複製 名前変更 リロード   新規 一覧 検索 最終更新   ヘルプ   最終更新のRSS
Last-modified: 2023-02-23 (木) 23:33:34