Movatterモバイル変換


[0]ホーム

URL:


コンテンツにスキップ
Wikipedia
検索

Tcl

出典: フリー百科事典『ウィキペディア(Wikipedia)』
曖昧さ回避この項目では、プログラミング言語について説明しています。その他の用法については「TCL」をご覧ください。
この記事は検証可能参考文献や出典が全く示されていないか、不十分です。出典を追加して記事の信頼性向上にご協力ください。このテンプレートの使い方
出典検索?"Tcl" – ニュース ·書籍 ·スカラー ·CiNii ·J-STAGE ·NDL ·dlib.jp ·ジャパンサーチ ·TWL
(2023年11月)

プログラミング言語

カテゴリ /テンプレート

Tcl
Tclのロゴ
パラダイム手続き型
登場時期1988年
設計者John Ousterhout
開発者Tclコアチーム[1]
最新リリース9.0.1[2]/2024年12月20日 (13か月前) (2024-12-20)
型付け動的型付け
主な処理系ActiveTcl
影響を受けた言語AWKLISP
影響を与えた言語PowerShell[3]、Tea
ライセンス Tcl/Tk License[4]
ウェブサイトwww.tcl-lang.org
拡張子.tcl
テンプレートを表示

Tclティクルスクリプト言語の一つであり、コマンド行のみで構造化文法をフォローしてしまう非常にシンプルな文法を特徴とする。

Tclはその非常に強力なGUIツールキットであるTkにより、GUIツールを素早く作り上げるのに適した強力なスクリプティング環境を構築できる。この両者を組み合わせTcl/Tkティクル・ティーケーと呼ばれる。ただしTkはクロスプラットフォームなGUI環境としても有名で、Tclに限らずPerlPythonRubyなどの言語環境からも利用できる。

Tcl/Tkは、スクリプト言語TclとGUIツールキットTkからなる 非常に強力なGUIスクリプティング環境である。現在、各種オペレーティングシステム(UNIXWindowsMacintosh)上で動作する。他にもウェブブラウザ上でTcl/Tkを動作させるプラグインTcletティクレットがある。

背景

[編集]

Tclは1988年にカリフォルニア大学バークレー校教授のジョン・オースターハウトによって開発された。当時アプリケーションプログラムに組み込まれる拡張用スクリプト言語には標準がなく、アプリケーション毎に独自の言語が実装されていた。そのためアプリケーション使用者はツール毎に異なるスクリプト言語の習得を余儀なくされた。この非効率さを嘆いたオースターハウトは、状況を打開するために UNIX アプリケーションにおける標準となる拡張スクリプト言語をデザインしようと考えた。こうして作られたのがTclの始まりである。そのためTclはアプリケーションへの組み込みが容易であることを重視してデザインされた。具体的には処理系をライブラリとして提供することでC言語で書かれたアプリケーションに容易に組み込めることや、言語構造が簡素であり、かつ高い拡張性を持つこと、インタプリタ言語であることが挙げられる。

TkはTcl用に開発された、非常に簡単なコードでGUIを作成できるツールキットである。1990年代初頭にTclにバンドルされる形で公開された。AppleHyperCardに触発されて開発されたと言う。当初Tclの有用な活用事例の一つとして紹介されたTkだが、その取り扱いやすさからTcl言語と共に一躍人気に火がつく。Tcl は当初の設計意図と異なり、アプリケーションの組み込み言語として使われるよりも、Tkと合わせた「Tcl/Tk」の形のGUIスクリプティング環境として人気を博した。

オースターハウトがサン・マイクロシステムズに勤めていた1994-1998年はTcl/Tkは同社で開発が進められた。このころのサンはWWW クライアント環境の制覇に向け邁進していた時期であり、Tcl/Tkもその流れの上、その対象領域をウェブに広げていく。ウェブブラウザ上でTk GUIを動作させるプラグイン「Tcletティクレット」や、ブラウザのスクリプト言語としてのTclの組み込み、国際化対応(内部処理のUnicode化)、インタプリタからバイトコンパイラへの変更による実行速度の大幅な向上など、Tcl/Tkはこの時期機能的にもっとも大きな進歩を果たした。しかし他のリッチインターネットアプリケーション技術との競合やブラウザでのサポートの薄さなどもあり、WWWを「第二のTk」として人気を拡大することは出来なかった。

オースターハウトの退職に伴い、Tcl/Tkの開発はサンの手を離れた。2000年からはTcl/Tkの開発はオープンソースにその場を移し、精力的に開発が続けられている。

2005年現在Tcl言語は「Tcl/Tk」の知名度とは裏腹に利用者数は少なく、PerlやPython、Rubyに比べ劣勢と言わざるを得ない。特に日本国内での利用者数は少ない。ただし、EDAツールにおいては標準的なスクリプト言語として広く利用されているほか、電子国土Webシステムにおいても一部でTcl言語が使われている。一方Tkは、後発のGTK+Qtと並び、軽量プログラミング言語における事実上の標準GUIツールキットの一つとなっており、広く利用されている。他の言語のTkを利用するバインディングとしてPerl/Tk、Tkinter(Python)Ruby/Tkなどがある。

なお、Tcl言語の名前は「ツールコマンド言語」を意味する英語「tool command language」に由来し、Tkの名前は「ツールキット」を意味する英語「toolkit」に由来する。

特徴

[編集]
→Tkについては、Tk (ツールキット)を参照

ここではTcl言語の特徴を記す。

Tcl言語の特徴は一言でいえば「全て文字列」である。Tclは、

  • コマンド行の順次実行のみのシンプルな文法
  • 首尾一貫したリスト構造

の二つを組み合わせ、非常に小さいルールで広範囲の領域をカバーする点にある。コマンド行はひとつのリストであり、先頭の要素がコマンド、それ以降の要素がコマンドへの引数として扱われる。LISPと同様の一貫性を持つと言えるが、リストの要素分離記号がブランクやタブであるため、これらの文字が無視される一般のプログラム言語に慣れた人には「とまどい」を与えるかもしれない。

リスト構造

[編集]

リストは文字列である。リストを構成する要素はブランクかタブで区切られる。ブランクやタブを要素に含めたい場合には、その要素をブレース{})で囲む。以下の文字列は三つの要素から成るリストである。

Thisis{apen}

リストの抽出

[編集]

Tclパーサーがソースコードからコマンド行としてリストを取り出すとき、ひとつのリストの終端は改行コードかセミコロン(;)で判断する。しかし改行コードやセミコロンがブレースの内側にあれば、それをリストの終端記号とは見なさない。したがって、以下の3行の文字列は3つの要素から成るひとつのリストである。改行コードは3個あるが、最後の改行コードだけがリストの終端記号の役割を果たす。

if{$a<0}{seta0}

上記リストは Tcl のif コマンドでありC言語のif 文に似ている。しかしブレースがリスト記述記号として採用されているので似ているだけである。上記のif コマンドを以下のように改行の位置を変えると、ifコマンドの引数エラーになる。

if{$a<0}{seta0}

Tclパーサーは上記4行の文字列を2つのリストとして認識する。したがって先頭行のifコマンド行は引数がひとつだけしか与えられていないことになり、引数エラーが発生する。

ここで重要なのは、エラーを返したのはif コマンドであり、Tclパーサーが検出した「文法エラー」ではないということである。Tclには制御文などの「文」はない。分岐や繰り返しなどの実行制御も単にコマンドによって実現されているだけである。「Tclには細かなルールが無い」のであり、そこにはリスト構造と、以下に解説する「特殊記号」しかない。

ブラケット記号(コマンド置換)

[編集]

Tclパーサーはリストの先頭要素を常にコマンド名として認識する。それ以外の要素はコマンドに渡すべき引数として認識する。しかし、その引数要素がブラケット([])で囲まれていると、その中身をコマンド行と認識し、それを実行してから本来の引数値を求めてくれる。これが「コマンド置換」である。

seta[expr100*2]

$記号(変数置換)

[編集]

Tclパーサーは変数機能の提供によりコマンド間でのデータの受け渡しも扱ってくれる。変数はsetコマンドにより生成される。そして、$記号が先頭に付いた要素を変数名とみなし、その値に要素全体を置換する。これが「変数置換」である。

seta{Thisisapen.}puts$a

変数置換はコマンド行の先頭要素に対しても機能する。つまりコマンド名を変数で与えることも可能である。

変数置換は1回しか実行されない。変数置換結果に文字「$」が含まれていても再置換が試みられることはない。

また、変数置換で得られた文字列にブランクが含まれていても、2つの引数としてではなく、ブランクが含まれた1つの引数として渡される。引数の分離(リストの認識)は変数置換前に行われるからである。もし置換結果から改めて引数分離を行わせたいならeval コマンドを利用する。

ダブルクォーテーション記号

[編集]

リスト構造で解説したように、ブレース({})は、改行コードなどの特殊文字の機能を無効化する。この法則はコマンド置換子であるブラケットや、変数置換子である$記号に対しても貫かれる。そのため、下記のコードではコマンド置換も変数置換も行われない。

puts{[expr100*$num]}

そのため、実行結果として出力される文字列は以下のものになる。

[expr 100*$num] 円

コマンド置換も変数置換も機能させ、かつブランクを含む文字列をひとつの引数としてputs コマンドに渡すには、ブレースの代わりにダブルクォーテーション(")で囲む。

puts"[expr 100*$num] 円"

num変数に3がセットされていればコマンドの実行結果として

300 円

が出力される。

このように、ダブルクォーテーションの機能はブレース機能とほぼ同等であるが、コマンド置換と変数置換をTclパーサーに許すところが異なる。ダブルクォーテーションの中でのコマンド置換、変数置換は Tcl の特長的な機能である。なお、ダブルクォーテーションとブレースの機能は、それらがリスト要素の先頭と末尾に記述された場合にのみ有効である。下の例では" は文字として扱われる。

settext石を投げたら"ゴツン"と音がした

セミコロン記号(マルチコマンド)

[編集]

複数のコマンドを1行に記述したい場合はコマンド行をセミコロン(;)で区切れば良い。

seta100;setb200;puts[expr$a*$b]

#記号(注釈行)

[編集]

コマンドの位置に「#」を記述すると行末までコメントと見なされる。「コマンドの位置に」ということが重要であり、以下の最後のコードは「#」がコマンドの位置にないので誤りである。

#初期値セット# set a 0setb0;# 初期値setc0#初期値

バックスラッシュ記号

[編集]

ひとつのコマンドを複数行で記述したい場合は、行の末尾にバックスラッシュを付ける。

command$arg1$arg2\$arg3$arg4

$文字の前に\を置くと$置換子の機能を抑制して単なる文字として扱わせることができる。

puts"金額=\$100"

コマンド置換子([])、ブレース({})の前に置いた場合も同様である。基本的にバックスラッシュにはC言語とほぼ同じ機能がある。例えば改行コードは「\n」と書ける。

補足:引数をブレースで挟むことのもうひとつの意味

[編集]

Tclパーサーから渡された引数がコマンド内部で評価されるか否かは重要である。ここでの評価とはコマンド置換や変数置換のことである。例えば算術演算を行うexpr コマンドは引数を内部で評価する。従って以下の2つのコマンドは同じ結果を返す。

expr$a*$bexpr{$a*$b}

Tclパーサーが行ってくれずとも、exprコマンドが内部で変数置換しているので同じ結果が得られるのである。ifコマンドやwhileコマンドは第1引数として与えられた文字列を、コマンド内部でexpr によって評価し、その結果を使う。

setvaltrueif$val{処理}while$val{処理}

Tclパーサーは第一引数の「$val」を評価して「true」(文字列)に書き換える。同様に第二引数「{〜処理〜}」を評価して、文字列リテラルなのでそのまま同じ値「{〜処理〜}」に書き換え、以下のようにする。

iftrue{処理}whiletrue{処理}

Tclパーサーはその後、true{〜処理〜}ifwhile コマンドに渡し、ifwhile コマンド側は受け取った文字列をexpr コマンドに「[expr true]」として渡し、その結果をもって処理を続ける。気をつけなければならないのはwhile コマンドは{〜処理〜} をひと通り評価し終えた後、再度、第一引数をexpr で評価するが、上記のコードでは与えられている値がtrue なので{〜処理〜} の中に「set val false」のような文があったとしても、第一引数で与えられた値そのものは変化しないので、ループを無限に繰り返すことになる({〜処理〜} の中にbreak コマンドが含まれていない場合)。

setvaltrueif{$val}{処理}while{$val}{処理}

この場合、Tclパーサーは引数を順に評価し、第一引数として「{$val}」という4文字の文字列リテラルを、第二引数として「{〜処理〜}」を、ifwhile コマンドに渡す。ifwhile コマンド側は受け取った文字列をexpr コマンドに「[expr {$val}]」として渡し、その結果をもって処理を続ける。while コマンドは{〜処理〜} をひと通り評価し終えた後、再度、第一引数をexpr で評価するが、上記のコードでは与えられている値が{$val} なので{〜処理〜} の中で「set val false」のような文があれば、その評価値もfalse になって、その時点で繰り返し処理を終了する。

一方、switchコマンドの第1引数は内部では評価されない。したがって下記の2番目の記述は期待する結果が得られない。

switch$val{...}switch{$val}{...}

もし引数をコマンド内部で評価してくれるならば、その引数はブレース({})で挟んで渡した方が効率が良い。Tclパーサーとコマンドの両方による多重評価処理を回避できるからである。このような理由により、ifコマンドやwhileコマンドやforコマンドの引数は常にブレースで挟むべきである。特にループの条件式は必ずブレースで囲む必要がある。ブレースを用いないと変数置換されてからループコマンドに渡されてしまうので、定数を並べた条件式になってしまい、無限ループをもたらす。

コマンドの拡張

[編集]

コマンドには、Tclパーサーにあらかじめ実装されているビルトインコマンドと、ユーザーにより作成された拡張コマンドがある。ユーザーによる拡張コマンドの実装は簡単である。まず、C言語などで「コマンド本体関数」と「登録用関数」を記述し、ダイナミックリンクライブラリファイルに格納する。そして組み込みコマンドのload を用いて拡張コマンドを登録する。

loadhello.dllhello

load コマンドの引数には、「ライブラリファイル名」と「登録用コマンド名」を与える。load コマンドは、登録用コマンド名から「登録用関数」名を求め、これを実行してくれる。例えばhello コマンドであればHello_Init を実行してくれる。この「登録用関数」の中に本当の登録処理を記述しておく。従って、例えばload コマンドに渡したコマンド名にライブラリ名の意味を持たせ、「登録用関数」の中で複数のコマンドを登録してしまうことも可能である。

「本当の」コマンド登録はC言語のインターフェース用の関数のTcl_CreateCommandTcl_CreateObjCommand を用いて行う。拡張コマンドの実体となる「コマンド本体関数」を、決められた型と引数に従って先に定義しておき、そのコマンド関数アドレスと公開コマンド名を引数に与えて実行すれば登録される。

コマンド登録関数

intHello_Init(Tcl_Interp*interp){Tcl_CreateObjCommand(interp,"hello",Tcl_HelloCmd,NULL,NULL);returnTCL_OK;}

コマンド本体関数

intTcl_HelloCmd(ClientDatadummy,Tcl_Interp*interp,intobjc,Tcl_Obj*CONSTobjv[]){...returnTCL_OK;}

コマンド本体関数は、Tclパーサーからの引数を、引数の数(argc)と引数文字列配列(argv)で受け取る。ただし、引数を文字列で受け取るこの方式はTcl_CreateCommand 関数で登録する場合であり、引数をTclオブジェクトで受け取りたい場合には、上例のようにTcl_CreateObjCommand 関数で登録する。

内部構造(Tclオブジェクト)

[編集]

Tclパーサーは、スクリプト文字列を受け取り、常に処理結果を文字列で返すように見える。これでは文字列の解析や文字コードとバイナリ値への変換が頻繁に行われていることになり、いかにも効率が悪く思える。しかし決してそのような単純なものではなく、内部では可能な限りバイナリ値を維持している。Tclスクリプトでバイナリ値も扱えるのはこのおかげである。

例えば下記のように変数に数値を与え、その計算結果を変数に格納するとき、おのおの変数の型は文字列とdouble の二つの型を持つ。文字列として参照されるときは文字列型として振る舞い、double として参照されるとき(計算式の中などで)はdouble 型として振る舞うことができる。

seta100.0setb200.0setc[expr$a*$b]

Tclパーサー内部において変数はTclオブジェクトとして存在している。Tclオブジェクトはchar *intdouble の共用体を持った構造体であり、文字列への変換が要求されない限り、int 値はint値のまま、double 値はdouble 値のまま維持される。スクリプトでは変数の型宣言を行えないが、変数への値セットで型が仮定され、Tclオブジェクト間のデータ移動で無駄な型変換が行われないように配慮されている、ということである。

リストも同様に内部ではリスト型のTclオブジェクトとして存在している。このような仕組みになっているので、リストを作成する時はlist コマンドを用いるべきであることが分かる。また、巨大なリストを文字列として全体参照するのも効率を悪くするので慎重にすべきである。下記の例では変数listA には文字列として格納されるが、listB にはint 値のリストとして格納される。この後、これらの変数にリスト処理コマンドでアクセスすると、listA に対しては要素分解処理が行われるが、listB に対しては不要となる。

seta100setb200setlistA"$a $b"setlistB[list$a$b]

前項で解説したコマンド登録関数の後者(Tcl_CreateObjCommand)は、Tclパーサーからの引数を、無駄に文字列変換することなく、Tclオブジェクトのままで受け取るコマンド関数を登録するためのものである。

返値もTclオブジェクトを通じて返すことができる。下記はlong 値をそのまま返す例である。

/* long値を返す例 */Tcl_SetLongObj(Tcl_GetObjResult(interp),longVal);

返値のみならず、Tclオブジェクトは不要になったとき自動的に削除される仕組みがある。それはTclオブジェクトが持つ「参照カウンタ」による制御である。アプリケーションでTclオブジェクトを作成し、それを存続したければ参照カウンタを増加させておき(Tcl_IncrRefCount)、必要なしとなったところで減ずる(Tcl_DecrRefCount)ようにする。参照カウンタが減らされて0 になったときにのみ、そのオブジェクトは削除される。つまり、存続と廃棄の要求を自分だけの都合で出しておくだけで、削除のタイミングがコントロールされるという仕組みである(スマートポインタ)。

変数のみならず、スクリプト自体もTclオブジェクトとして存在している。

脚注

[編集]
[脚注の使い方]

注釈

[編集]

出典

[編集]
  1. ^Tcl/Tk Core Development”. Tcl Developer Xchange. 2025年2月7日閲覧。
  2. ^Tcl/Tk 9.0”. 2025年2月7日閲覧。
  3. ^Windows PowerShell : PowerShell and WPF: WTF
  4. ^Tcl/Tk Licensing Terms”. Tcl Developer Xchange. 2025年2月7日閲覧。

外部リンク

[編集]
コンピュータ・プログラミング言語
低水準言語
高水準言語
1950年代
1960年代
1970年代
1980年代
1990年代
2000年代
2010年代
2020年代
架空の言語
全般
解説
開発
OS英語版
ソフトウェア
ホスティング
団体
標準化団体
開発支援団体
ライセンス
指針
主要例
その他
課題
特性上の課題
対立と論争
その他
一覧
スタブアイコン

この項目は、FOSSに関連した書きかけの項目です。この項目を加筆・訂正などしてくださる協力者を求めていますPortal:FOSS)。

国立図書館
その他
https://ja.wikipedia.org/w/index.php?title=Tcl&oldid=107472946」から取得
カテゴリ:
隠しカテゴリ:

[8]ページ先頭

©2009-2026 Movatter.jp