Fortran-クイックガイド

Formula Translating Systemから派生したFortranは、汎用の命令型プログラミング言語です。数値計算および科学計算に使用されます。

Fortranは、もともと1950年代に科学および工学アプリケーションのためにIBMによって開発されました。Fortranはこのプログラミング領域を長い間支配し、高性能コンピューティングで非常に人気がありました。

−をサポートします

  • 数値解析と科学計算
  • 構造化プログラミング
  • 配列プログラミング
  • モジュラープログラミング
  • ジェネリックプログラミング
  • スーパーコンピューターでのハイパフォーマンスコンピューティング
  • オブジェクト指向プログラミング
  • 並行プログラミング
  • コンピュータシステム間の適度な移植性

Fortranについての事実

  • Fortranは、1957年にIBMのJohnBackusが率いるチームによって作成されました。

  • 当初、名前はすべて大文字で書かれていましたが、現在の標準と実装では、最初の文字を大文字にする必要があります。

  • Fortranは、FORmulaTRANslatorの略です。

  • もともとは科学計算用に開発されましたが、汎用プログラミングに必要な文字列やその他の構造のサポートは非​​常に限られていました。

  • その後の拡張と開発により、移植性の高い高水準プログラミング言語になりました。

  • 元のバージョンであるFortranI、II、およびIIIは、現在は廃止されていると見なされています。

  • まだ使用されている最も古いバージョンは、FortranIVとFortran66です。

  • 今日最も一般的に使用されているバージョンは、Fortran 77、Fortran 90、およびFortran95です。

  • Fortran 77は、別個のタイプとして文字列を追加しました。

  • Fortran 90は、さまざまな種類のスレッド化と直接配列処理を追加しました。

WindowsでのFortranのセットアップ

G95は、WindowsでFortranをセットアップするために使用されるGNUFortranマルチアーキテクチャコンパイラです。Windowsバージョンは、WindowsでMingWを使用してUNIX環境をエミュレートします。インストーラーがこれを処理し、g95をWindowsのPATH変数に自動的に追加します。

ここからG95の安定バージョンを入手できます

G95の使い方

インストール中、 g95「推奨」オプションを選択すると、PATH変数に自動的に追加されます。これは、新しいコマンドプロンプトウィンドウを開き、「g95」と入力するだけでコンパイラを起動できることを意味します。開始するには、以下のいくつかの基本的なコマンドを見つけてください。

シニア番号 コマンドと説明
1

g95 –c hello.f90

hello.f90をhello.oという名前のオブジェクトファイルにコンパイルします

2

g95 hello.f90

hello.f90をコンパイルし、それをリンクして実行可能ファイルa.outを生成します

3

g95 -c h1.f90 h2.f90 h3.f90

複数のソースファイルをコンパイルします。すべてがうまくいけば、オブジェクトファイルh1.o、h2.o、h3.oが作成されます

4

g95 -o hello h1.f90 h2.f90 h3.f90

複数のソースファイルをコンパイルし、それらを「hello」という名前の実行可能ファイルにリンクします

G95のコマンドラインオプション

-c Compile only, do not run the linker.
-o Specify the name of the output file, either an object file or the executable.

複数のソースファイルとオブジェクトファイルを一度に指定できます。Fortranファイルは、「。f」、「。F」、「。for」、「。FOR」、「。f90」、「。F90」、「。f95」、「。F95」、「」で終わる名前で示されます。 f03 "および" .F03 "。複数のソースファイルを指定できます。オブジェクトファイルも指定でき、リンクされて実行可能ファイルを形成します。

Fortranプログラムは、メインプログラム、モジュール、外部サブプログラムまたはプロシージャなどのプログラムユニットのコレクションで構成されています。

各プログラムには1つのメインプログラムが含まれ、他のプログラムユニットが含まれる場合と含まれない場合があります。メインプログラムの構文は次のとおりです-

program program_name
implicit none      

! type declaration statements      
! executable statements  

end program program_name

Fortranの単純なプログラム

2つの数値を加算して結果を出力するプログラムを書いてみましょう-

program addNumbers

! This simple program adds two numbers
   implicit none

! Type declarations
   real :: a, b, result

! Executable statements
   a = 12.0
   b = 15.0
   result = a + b
   print *, 'The total is ', result

end program addNumbers

上記のプログラムをコンパイルして実行すると、次の結果が得られます-

The total is 27.0000000

注意してください-

  • すべてのFortranプログラムはキーワードで始まります program そしてキーワードで終わります end program, プログラムの名前が続きます。

  • ザ・ implicit noneステートメントを使用すると、コンパイラはすべての変数タイプが正しく宣言されていることを確認できます。常に使用する必要がありますimplicit none すべてのプログラムの開始時に。

  • この後のすべての文字(文字ストリングを除く)はコンパイラーによって無視されるため、Fortranのコメントは感嘆符(!)で始まります。

  • ザ・ print * コマンドは画面にデータを表示します。

  • コード行のインデントは、プログラムを読みやすくするための良い方法です。

  • Fortranでは、大文字と小文字の両方を使用できます。文字列リテラルを除いて、Fortranでは大文字と小文字は区別されません。

基本

ザ・ basic character set Fortranの-が含まれています

  • 文字A ... Zおよびa ... z
  • 数字0 ... 9
  • アンダースコア(_)文字
  • 特殊文字=:+空白-* /()[] 、。$ '!"%&; <>?

Tokens基本文字セットの文字で構成されています。トークンには、キーワード、識別子、定数、文字列リテラル、または記号を使用できます。

プログラムステートメントはトークンで構成されています。

識別子

識別子は、変数、プロシージャ、またはその他のユーザー定義アイテムを識別するために使用される名前です。Fortranでの名前は、次の規則に従う必要があります-

  • 31文字を超えることはできません。

  • 英数字(アルファベットのすべての文字、および0から9の数字)とアンダースコア(_)で構成されている必要があります。

  • 名前の最初の文字は文字でなければなりません。

  • 名前は大文字と小文字を区別しません

キーワード

キーワードは特別な単語であり、その言語のために予約されています。これらの予約語は、識別子または名前として使用できません。

次の表に、Fortranキーワード-を示します。

非I / Oキーワード
割り当て可能 割り当てる 割当 割り当て ブロックデータ
コール 場合 キャラクター 一般 繁雑
含まれています 継続する サイクル データ 割り当て解除
デフォルト 行う 倍精度 そうしないと それ以外の場合
他の場所 エンドブロックデータ 終わります 終了機能 終了する場合
エンドインターフェース エンドモジュール プログラム終了 選択を終了 サブルーチンの終了
エンドタイプ ここで終了 エントリ 等価 出口
外部 関数 に行く もし 暗黙
inout 整数 意図 インターフェース
内在的 種類 len 論理的 モジュール
ネームリスト 無効化 のみ オペレーター オプション
アウト パラメータ 一時停止 ポインター 民間
プログラム 公衆 リアル 再帰的 結果
戻る 保存する ケースを選択 やめる サブルーチン
目標 その後 タイプ タイプ() 使用する
どこ 一方
I / O関連のキーワード
バックスペース 閉じる エンドファイル フォーマット お問い合わせ
開いた 印刷 読んだ 巻き戻し 書く

Fortranは5つの固有のデータ型を提供しますが、独自のデータ型を導出することもできます。5つの固有のタイプは次のとおりです。

  • 整数型
  • 実数型
  • 複合型
  • 論理型
  • 文字タイプ

整数型

整数型は整数値のみを保持できます。次の例では、通常の4バイト整数で保持できる最大値を抽出します-

program testingInt
implicit none

   integer :: largeval
   print *, huge(largeval)
   
end program testingInt

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

2147483647

注意してください huge()関数は、特定の整数データ型が保持できる最大数を示します。を使用してバイト数を指定することもできますkind指定子。次の例はこれを示しています-

program testingInt
implicit none

   !two byte integer
   integer(kind = 2) :: shortval
   
   !four byte integer
   integer(kind = 4) :: longval
   
   !eight byte integer
   integer(kind = 8) :: verylongval
   
   !sixteen byte integer
   integer(kind = 16) :: veryverylongval
   
   !default integer 
   integer :: defval
        
   print *, huge(shortval)
   print *, huge(longval)
   print *, huge(verylongval)
   print *, huge(veryverylongval)
   print *, huge(defval)
   
end program testingInt

上記のプログラムをコンパイルして実行すると、次の結果が得られます-

32767
2147483647
9223372036854775807
170141183460469231731687303715884105727
2147483647

実数型

2.0、3.1415、-100.876などの浮動小数点数を格納します。

従来、2つの異なる実数型があり、デフォルトです。 real タイプと double precision タイプ。

ただし、Fortran 90/95では、実数および整数のデータ型の精度を kind 指定子。これについては、数値の章で学習します。

次の例は、実データ型の使用を示しています-

program division   
implicit none  

   ! Define real variables   
   real :: p, q, realRes 
   
   ! Define integer variables  
   integer :: i, j, intRes  
   
   ! Assigning  values   
   p = 2.0 
   q = 3.0    
   i = 2 
   j = 3  
   
   ! floating point division
   realRes = p/q  
   intRes = i/j
   
   print *, realRes
   print *, intRes
   
end program division

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

0.666666687    
0

複合型

これは、複素数を格納するために使用されます。複素数には、実数部と虚数部の2つの部分があります。2つの連続する数値ストレージユニットは、これら2つの部分を格納します。

たとえば、複素数(3.0、-5.0)は3.0 –5.0iに等しくなります

複素数のタイプについては、数値の章で詳しく説明します。

論理型

論理値は2つだけです。 .true. そして .false.

文字タイプ

文字タイプには、文字と文字列が格納されます。文字列の長さは、len指定子で指定できます。長さが指定されていない場合は1です。

For example,

character (len = 40) :: name  
name = “Zara Ali”

表現、 name(1:4) 部分文字列「Zara」を与えます。

暗黙の入力

古いバージョンのFortranでは、暗黙的な型指定と呼ばれる機能が許可されていました。つまり、使用する前に変数を宣言する必要はありません。変数が宣言されていない場合、その名前の最初の文字がその型を決定します。

i、j、k、l、m、またはnで始まる変数名は整数変数と見なされ、その他は実変数です。ただし、これは優れたプログラミング手法であるため、すべての変数を宣言する必要があります。そのためには、次のステートメントでプログラムを開始します-

implicit none

このステートメントは、暗黙の入力をオフにします。

変数は、プログラムが操作できるストレージ領域に付けられた名前に他なりません。各変数には、変数のメモリのサイズとレイアウトを決定する特定のタイプが必要です。そのメモリ内に保存できる値の範囲。変数に適用できる一連の操作。

変数の名前は、文字、数字、および下線文字で構成できます。Fortranでの名前は、次の規則に従う必要があります-

  • 31文字を超えることはできません。

  • 英数字(アルファベットのすべての文字、および0から9の数字)とアンダースコア(_)で構成されている必要があります。

  • 名前の最初の文字は文字でなければなりません。

  • 名前では大文字と小文字は区別されません。

前の章で説明した基本タイプに基づいて、以下は変数タイプです。

シニア番号 タイプと説明
1

Integer

整数値のみを保持できます。

2

Real

浮動小数点数を格納します。

3

Complex

複素数を格納するために使用されます。

4

Logical

論理ブール値を格納します。

5

Character

文字または文字列を格納します。

変数宣言

変数は、型宣言ステートメントのプログラム(またはサブプログラム)の先頭で宣言されます。

変数宣言の構文は次のとおりです-

type-specifier :: variable_name

例えば

integer :: total  	
real :: average 
complex :: cx  
logical :: done 
character(len = 80) :: message ! a string of 80 characters

後で、次のようにこれらの変数に値を割り当てることができます。

total = 20000  
average = 1666.67   
done = .true.   
message = “A big Hello from Tutorials Point” 
cx = (3.0, 5.0) ! cx = 3.0 + 5.0i

組み込み関数を使用することもできます cmplx, 複素変数に値を割り当てる-

cx = cmplx (1.0/2.0, -7.0) ! cx = 0.5 – 7.0i 
cx = cmplx (x, y) ! cx = x + yi

次の例は、変数の宣言、割り当て、および画面への表示を示しています。

program variableTesting
implicit none

   ! declaring variables
   integer :: total      
   real :: average 
   complex :: cx  
   logical :: done 
   character(len=80) :: message ! a string of 80 characters
   
   !assigning values
   total = 20000  
   average = 1666.67   
   done = .true.   
   message = "A big Hello from Tutorials Point" 
   cx = (3.0, 5.0) ! cx = 3.0 + 5.0i

   Print *, total
   Print *, average
   Print *, cx
   Print *, done
   Print *, message
   
end program variableTesting

上記のコードをコンパイルして実行すると、次の結果が得られます。

20000
1666.67004    
(3.00000000, 5.00000000 )
T
A big Hello from Tutorials Point

定数は、プログラムの実行中に変更できない固定値を参照します。これらの固定値は、literals

定数は、整数定数、浮動定数、文字定数、複素定数、文字列リテラルなど、基本的なデータ型のいずれでもかまいません。論理定数は2つだけです。.true. そして .false.

定数は、定義後に値を変更できないことを除いて、通常の変数と同じように扱われます。

名前付き定数とリテラル

定数には2つのタイプがあります-

  • リテラル定数
  • 名前付き定数

リテラル定数には値がありますが、名前はありません。

たとえば、以下はリテラル定数-です。

タイプ
整数定数 0 1 -1 300 123456789
実定数 0.0 1.0 -1.0 123.456 7.1E + 10 -52.715E-30
複素定数 (0.0、0.0)(-123.456E + 30、987.654E-29)
論理定数 .true。.false。
文字定数

"PQR" "a" "123'abc $%#@!"

" 見積もり "" "

'PQR''a''123 "abc $%#@!'

'アポストロフィ' ''

名前付き定数には、名前だけでなく値もあります。

名前付き定数は、変数の型宣言と同じように、プログラムまたはプロシージャの先頭で宣言して、その名前と型を示す必要があります。名前付き定数は、パラメータ属性で宣言されます。例えば、

real, parameter :: pi = 3.1415927

次のプログラムは、重力下での垂直運動による変位を計算します。

program gravitationalDisp

! this program calculates vertical motion under gravity 
implicit none  

   ! gravitational acceleration
   real, parameter :: g = 9.81   
   
   ! variable declaration
   real :: s ! displacement   
   real :: t ! time  
   real :: u ! initial speed  
   
   ! assigning values 
   t = 5.0   
   u = 50  
   
   ! displacement   
   s = u * t - g * (t**2) / 2  
   
   ! output 
   print *, "Time = ", t
   print *, 'Displacement = ',s  
   
end program gravitationalDisp

上記のコードをコンパイルして実行すると、次の結果が得られます。

Time = 5.00000000    
Displacement = 127.374992

演算子は、特定の数学的または論理的操作を実行するようにコンパイラーに指示する記号です。Fortranは、次のタイプの演算子を提供します-

  • 算術演算子
  • 関係演算子
  • 論理演算子

これらすべてのタイプの演算子を1つずつ見ていきましょう。

算術演算子

次の表は、Fortranでサポートされているすべての算術演算子を示しています。変数を想定A 5と変数を保持します B 3を保持し、その後-

例を表示

オペレーター 説明
+ 加算演算子。2つのオペランドを加算します。 A + Bは8を与えます
- 減算演算子。最初のオペランドから2番目のオペランドを減算します。 A-Bは2を与えます
* 乗算演算子。両方のオペランドを乗算します。 A * Bは15を与えます
/ 除算演算子。分子を分子で除算します。 A / Bは1を与えます
**** べき乗演算子。一方のオペランドをもう一方の累乗にします。 A ** Bは125を与えます

関係演算子

次の表は、Fortranでサポートされているすべての関係演算子を示しています。変数を想定A 10と変数を保持します B 20を保持し、次に−

例を表示

オペレーター 同等 説明
== .eq。 2つのオペランドの値が等しいかどうかをチェックし、等しい場合は条件が真になります。 (A == B)は真ではありません。
/ = .ne。 2つのオペランドの値が等しいかどうかをチェックし、値が等しくない場合は条件が真になります。 (A!= B)は真です。
>> .gt。 左のオペランドの値が右のオペランドの値より大きいかどうかを確認します。大きい場合は、条件が真になります。 (A> B)は正しくありません。
< .lt。 左のオペランドの値が右のオペランドの値よりも小さいかどうかを確認します。小さい場合は、条件が真になります。 (A <B)は真です。
> = .ge。 左のオペランドの値が右のオペランドの値以上であるかどうかをチェックします。はいの場合、条件は真になります。 (A> = B)は正しくありません。
<= .le。 左のオペランドの値が右のオペランドの値以下であるかどうかをチェックします。はいの場合、条件は真になります。 (A <= B)は真です。

論理演算子

Fortranの論理演算子は、論理値.trueでのみ機能します。および.false。

次の表は、Fortranでサポートされているすべての論理演算子を示しています。変数Aが.trueを保持すると仮定します。変数Bは.falseを保持します。、次に−

例を表示

オペレーター 説明
。そして。 論理AND演算子と呼ばれます。両方のオペランドがゼロ以外の場合、条件は真になります。 (A.and。B)は偽です。
。または。 論理OR演算子と呼ばれます。2つのオペランドのいずれかがゼロ以外の場合、条件は真になります。 (A.or。B)は真です。
.not。 論理NOT演算子と呼ばれます。オペランドの論理状態を逆にするために使用します。条件がtrueの場合、LogicalNOT演算子はfalseになります。 !(A.and。B)は真です。
.eqv。 論理等価演算子と呼ばれます。2つの論理値の同等性をチェックするために使用されます。 (A.eqv。B)は偽です。
.neqv。 論理非等価演算子と呼ばれます。2つの論理値の非等価性をチェックするために使用されます。 (A.neqv。B)は真です。

Fortranでの演算子の優先順位

演算子の優先順位は、式内の用語のグループ化を決定します。これは、式の評価方法に影響します。特定の演算子は他の演算子よりも優先されます。たとえば、乗算演算子は加算演算子よりも優先されます。

たとえば、x = 7 + 3 * 2; ここでは、演算子*の優先順位が+よりも高いため、xには20ではなく13が割り当てられます。したがって、最初に3 * 2が乗算され、次に7に加算されます。

ここでは、優先順位が最も高い演算子がテーブルの上部に表示され、優先順位が最も低い演算子が下部に表示されます。式内では、優先順位の高い演算子が最初に評価されます。

例を表示

カテゴリー オペレーター 結合性
論理NOTおよび否定記号 .not。(-) 左から右へ
べき乗 **** 左から右へ
乗法 * / 左から右へ
添加剤 +- 左から右へ
関連した << = >> = 左から右へ
平等 == / = 左から右へ
論理積 。そして。 左から右へ
論理OR 。または。 左から右へ
割り当て = 右から左へ

意思決定構造では、プログラマーは、プログラムによって評価またはテストされる1つ以上の条件と、条件が真であると判断された場合に実行される1つまたは複数のステートメント、およびオプションで、条件はfalseと判断されます。

以下は、ほとんどのプログラミング言語に見られる典型的な意思決定構造の一般的な形式です。

Fortranは、次のタイプの意思決定構造を提供します。

シニア番号 ステートメントと説明
1 もし…なら構築する

アン if… then… end if ステートメントは、論理式とそれに続く1つ以上のステートメントで構成されます。

2 If…then ... else構文

アン if… then ステートメントの後にオプションを続けることができます else statement, これは、論理式がfalseの場合に実行されます。

3 if ... else if ... elseステートメント

アン if ステートメント構成には、1つ以上のオプションを含めることができます else-if構成。いつif 条件が失敗し、直後に続く else-if実行されます。いつelse-if また失敗します、その後継者 else-if ステートメント(存在する場合)が実行されます。

4 ネストされたif構文

あなたは1つを使用することができます if または else if 別の内部のステートメント if または else if ステートメント。

5 ケース構成を選択

select case ステートメントを使用すると、値のリストに対して変数が等しいかどうかをテストできます。

6 ネストされた選択ケース構成

あなたは1つを使用することができます select case 別の内部のステートメント select case ステートメント。

コードのブロックを数回実行する必要がある場合があります。一般に、ステートメントは順番に実行されます。関数の最初のステートメントが最初に実行され、次に2番目のステートメントが実行されます。

プログラミング言語は、より複雑な実行パスを可能にするさまざまな制御構造を提供します。

ループステートメントを使用すると、ステートメントまたはステートメントのグループを複数回実行できます。以下は、ほとんどのプログラミング言語でのループステートメントの一般的な形式です。

Fortranは、ループ要件を処理するために次のタイプのループ構造を提供します。詳細を確認するには、次のリンクをクリックしてください。

シニア番号 ループの種類と説明
1 do loop

この構成により、特定の条件が真である間、ステートメントまたは一連のステートメントを繰り返し実行できます。

2 whileループを実行します

指定された条件が真である間、ステートメントまたはステートメントのグループを繰り返します。ループ本体を実行する前に条件をテストします。

3 ネストされたループ

他のループ構造内で1つ以上のループ構造を使用できます。

ループ制御ステートメント

ループ制御ステートメントは、実行を通常のシーケンスから変更します。実行がスコープを離れると、そのスコープで作成されたすべての自動オブジェクトが破棄されます。

Fortranは、以下の制御ステートメントをサポートします。詳細を確認するには、次のリンクをクリックしてください。

シニア番号 制御ステートメントと説明
1 出口

exitステートメントが実行されると、ループが終了し、プログラムの実行はenddoステートメントの後の最初の実行可能ステートメントから続行されます。

2 サイクル

サイクルステートメントが実行されると、プログラムは次の反復の開始時に続行されます。

3 やめる

プログラムの実行を停止したい場合は、stopステートメントを挿入できます。

Fortranの数値は、3つの固有のデータ型で表されます-

  • 整数型
  • 実数型
  • 複合型

整数型

整数型は整数値のみを保持できます。次の例では、通常の4バイト整数で保持できる最大値を抽出します。

program testingInt
implicit none

   integer :: largeval
   print *, huge(largeval)
   
end program testingInt

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

2147483647

注意してください huge()関数は、特定の整数データ型が保持できる最大数を示します。を使用してバイト数を指定することもできますkind指定子。次の例はこれを示しています-

program testingInt
implicit none

   !two byte integer
   integer(kind = 2) :: shortval
   
   !four byte integer
   integer(kind = 4) :: longval
   
   !eight byte integer
   integer(kind = 8) :: verylongval
   
   !sixteen byte integer
   integer(kind = 16) :: veryverylongval
   
   !default integer 
   integer :: defval
        
   print *, huge(shortval)
   print *, huge(longval)
   print *, huge(verylongval)
   print *, huge(veryverylongval)
   print *, huge(defval)
   
end program testingInt

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

32767
2147483647
9223372036854775807
170141183460469231731687303715884105727
2147483647

実数型

2.0、3.1415、-100.876などの浮動小数点数を格納します。

伝統的に2つの異なるものがありました real タイプ:デフォルトの実際のタイプと double precision タイプ。

ただし、Fortran 90/95では、実数および整数のデータ型の精度を kind 指定子。これについては後ほど説明します。

次の例は、実データ型の使用を示しています-

program division   
implicit none

   ! Define real variables   
   real :: p, q, realRes 
   
   ! Define integer variables  
   integer :: i, j, intRes  
   
   ! Assigning  values   
   p = 2.0 
   q = 3.0    
   i = 2 
   j = 3  
   
   ! floating point division
   realRes = p/q  
   intRes = i/j
   
   print *, realRes
   print *, intRes
   
end program division

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

0.666666687    
0

複合型

これは、複素数を格納するために使用されます。複素数には、実数部と虚数部の2つの部分があります。2つの連続する数値ストレージユニットは、これら2つの部分を格納します。

たとえば、複素数(3.0、-5.0)は3.0 –5.0iに等しくなります

汎用関数 cmplx()複素数を作成します。入力引数のタイプに関係なく、実数部と虚数部が単精度であるという結果が生成されます。

program createComplex
implicit none

   integer :: i = 10
   real :: x = 5.17
   print *, cmplx(i, x)
   
end program createComplex

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

(10.0000000, 5.17000008)

次のプログラムは、複素数の算術演算を示しています。

program ComplexArithmatic
implicit none

   complex, parameter :: i = (0, 1)   ! sqrt(-1)   
   complex :: x, y, z 
   
   x = (7, 8); 
   y = (5, -7)   
   write(*,*) i * x * y
   
   z = x + y
   print *, "z = x + y = ", z
   
   z = x - y
   print *, "z = x - y = ", z 
   
   z = x * y
   print *, "z = x * y = ", z 
   
   z = x / y
   print *, "z = x / y = ", z 
   
end program ComplexArithmatic

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

(9.00000000, 91.0000000)
z = x + y = (12.0000000, 1.00000000)
z = x - y = (2.00000000, 15.0000000)
z = x * y = (91.0000000, -9.00000000)
z = x / y = (-0.283783793, 1.20270276)

数値の範囲、精度、サイズ

整数の範囲、浮動小数点数の精度とサイズは、特定のデータ型に割り当てられたビット数によって異なります。

次の表は、整数のビット数と範囲を示しています-

ビット数 最大値 理由
64 9,223,372,036,854,774,807 (2 ** 63)–1
32 2,147,483,647 (2 ** 31)–1

次の表に、ビット数、最小値と最大値、および実数の精度を示します。

ビット数 最大値 最小値 精度
64 0.8E + 308 0.5E–308 15〜18
32 1.7E + 38 0.3E–38 6-9

次の例はこれを示しています-

program rangePrecision
implicit none

   real:: x, y, z
   x = 1.5e+40
   y = 3.73e+40
   z = x * y 
   print *, z
   
end program rangePrecision

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

x = 1.5e+40
          1
Error : Real constant overflows its kind at (1)
main.f95:5.12:

y = 3.73e+40
           1
Error : Real constant overflows its kind at (1)

ここで、より小さな数を使用しましょう-

program rangePrecision
implicit none

   real:: x, y, z
   x = 1.5e+20
   y = 3.73e+20
   z = x * y 
   print *, z
   
   z = x/y
   print *, z
   
end program rangePrecision

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

Infinity
0.402144760

それでは、アンダーフローを見てみましょう-

program rangePrecision
implicit none

   real:: x, y, z
   x = 1.5e-30
   y = 3.73e-60
   z = x * y 
   print *, z
   
   z = x/y
   print *, z

end program rangePrecision

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

y = 3.73e-60
           1
Warning : Real constant underflows its kind at (1)

Executing the program....
$demo 

0.00000000E+00
Infinity

種類指定子

科学プログラミングでは、作業が行われているハードウェアプラットフォームのデータの範囲と精度を知る必要があることがよくあります。

組み込み関数 kind() プログラムを実行する前に、ハードウェアのデータ表現の詳細を照会できます。

program kindCheck
implicit none
   
   integer :: i 
   real :: r 
   complex :: cp 
   print *,' Integer ', kind(i) 
   print *,' Real ', kind(r) 
   print *,' Complex ', kind(cp) 
   
end program kindCheck

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

Integer 4
Real 4
Complex 4

すべてのデータ型の種類を確認することもできます-

program checkKind
implicit none

   integer :: i 
   real :: r 
   character :: c 
   logical :: lg 
   complex :: cp 
   
   print *,' Integer ', kind(i) 
   print *,' Real ', kind(r) 
   print *,' Complex ', kind(cp)
   print *,' Character ', kind(c) 
   print *,' Logical ', kind(lg)
   
end program checkKind

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

Integer 4
Real 4
Complex 4
Character 1
Logical 4

Fortran言語は、文字を単一の文字または連続した文字列として扱うことができます。

文字は、基本的な文字セット、つまり文字、10進数、アンダースコア、および21個の特殊文字から取得した任意の記号にすることができます。

文字定数は、固定値の文字列です。

固有のデータ型 character文字と文字列を格納します。文字列の長さは、次の式で指定できます。len指定子。長さが指定されていない場合は1です。位置で参照する文字列内の個々の文字を参照できます。左端の文字は位置1にあります。

文字宣言

文字型データの宣言は他の変数と同じです-

type-specifier :: variable_name

例えば、

character :: reply, sex

次のような値を割り当てることができます。

reply = ‘N’ 
sex = ‘F’

次の例は、文字データ型の宣言と使用法を示しています。

program hello
implicit none

   character(len = 15) :: surname, firstname 
   character(len = 6) :: title 
   character(len = 25)::greetings
   
   title = 'Mr. ' 
   firstname = 'Rowan ' 
   surname = 'Atkinson'
   greetings = 'A big hello from Mr. Bean'
   
   print *, 'Here is ', title, firstname, surname
   print *, greetings
   
end program hello

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

Here is Mr. Rowan Atkinson       
A big hello from Mr. Bean

文字の連結

連結演算子//は、文字を連結します。

次の例はこれを示しています-

program hello
implicit none

   character(len = 15) :: surname, firstname 
   character(len = 6) :: title 
   character(len = 40):: name
   character(len = 25)::greetings
   
   title = 'Mr. ' 
   firstname = 'Rowan ' 
   surname = 'Atkinson'
   
   name = title//firstname//surname
   greetings = 'A big hello from Mr. Bean'
   
   print *, 'Here is ', name
   print *, greetings
   
end program hello

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

Here is Mr.Rowan Atkinson       
A big hello from Mr.Bean

いくつかの文字関数

次の表に、一般的に使用されるいくつかの文字関数とその説明を示します。

シニア番号 機能と説明
1

len(string)

文字列の長さを返します

2

index(string,sustring)

別の文字列内の部分文字列の場所を見つけ、見つからない場合は0を返します。

3

achar(int)

整数を文字に変換します

4

iachar(c)

文字を整数に変換します

5

trim(string)

末尾の空白が削除された文字列を返します。

6

scan(string, chars)

「文字列」を左から右に検索し(back = .true。を除く)、「chars」に含まれる文字が最初に出現するかどうかを検索します。その文字の位置を示す整数を返します。「chars」内の文字が見つからない場合はゼロを返します。

7

verify(string, chars)

「chars」に含まれていない文字が最初に出現するかどうか、「string」を左から右にスキャンします(back = .true。を除く)。その文字の位置を示す整数を返します。「chars」内の文字のみが見つかった場合はゼロを返します。

8

adjustl(string)

「文字列」に含まれる文字を左揃えにします

9

adjustr(string)

「文字列」に含まれる文字を正しく正当化します

10

len_trim(string)

「string」(len(string))の長さから末尾のブランクの数を引いたものに等しい整数を返します。

11

repeat(string,ncopy)

「ncopy」に「string」の長さを掛けたものに等しい長さで、「string」の「ncopy」連結コピーを含む文字列を返します。

例1

この例は、 index 関数-

program testingChars
implicit none

   character (80) :: text 
   integer :: i 
   
   text = 'The intrinsic data type character stores characters and   strings.'
   i=index(text,'character') 
   
   if (i /= 0) then
      print *, ' The word character found at position ',i 
      print *, ' in text: ', text 
   end if
   
end program testingChars

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

The word character found at position 25
in text : The intrinsic data type character stores characters and strings.

例2

この例は、 trim 関数-

program hello
implicit none

   character(len = 15) :: surname, firstname 
   character(len = 6) :: title 
   character(len = 25)::greetings
   
   title = 'Mr.' 
   firstname = 'Rowan' 
   surname = 'Atkinson'
   
   print *, 'Here is', title, firstname, surname
   print *, 'Here is', trim(title),' ',trim(firstname),' ', trim(surname)
   
end program hello

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

Here isMr.   Rowan          Atkinson       
 Here isMr. Rowan Atkinson

例3

この例は、 achar 関数-

program testingChars
implicit none

   character:: ch
   integer:: i
   
   do i = 65, 90
      ch = achar(i)
      print*, i, ' ', ch
   end do
   
end program testingChars

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

65  A
66  B
67  C
68  D
69  E
70  F
71  G
72  H
73  I
74  J
75  K
76  L
77  M
78  N
79  O
80  P
81  Q
82  R
83  S
84  T
85  U
86  V
87  W
88  X
89  Y
90  Z

文字の辞書式順序の確認

次の関数は、文字の字句シーケンスを決定します-

シニア番号 機能と説明
1

lle(char, char)

最初の文字が字句的に2番目の文字以下であるかどうかを比較します

2

lge(char, char)

最初の文字が字句的に2番目の文字以上であるかどうかを比較します

3

lgt(char, char)

最初の文字が2番目の文字よりも字句的に大きいかどうかを比較します

4

llt(char, char)

最初の文字が語彙的に2番目の文字よりも小さいかどうかを比較します

Example 4

次の関数は、使用法を示しています-

program testingChars
implicit none

   character:: a, b, c
   a = 'A'
   b = 'a'
   c = 'B'
   
   if(lgt(a,b)) then
      print *, 'A is lexically greater than a'
   else
      print *, 'a is lexically greater than A'
   end if
   
   if(lgt(a,c)) then
      print *, 'A is lexically greater than B'
   else
      print *, 'B is lexically greater than A'
   end if  
   
   if(llt(a,b)) then
      print *, 'A is lexically less than a'
   end if
   
   if(llt(a,c)) then
      print *, 'A is lexically less than B'
   end if
   
end program testingChars

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

a is lexically greater than A
B is lexically greater than A
A is lexically less than a
A is lexically less than B

Fortran言語は、文字を単一の文字または連続した文字列として扱うことができます。

文字列の長さは1文字のみの場合もあれば、長さがゼロの場合もあります。Fortranでは、文字定数は二重引用符または一重引用符のペアの間に指定されます。

固有のデータ型 character文字と文字列を格納します。文字列の長さは、次の式で指定できます。len specifier。長さが指定されていない場合は1です。位置で参照する文字列内の個々の文字を参照できます。左端の文字は位置1にあります。

文字列宣言

文字列の宣言は他の変数と同じです-

type-specifier :: variable_name

例えば、

Character(len = 20) :: firstname, surname

次のような値を割り当てることができます。

character (len = 40) :: name  
name = “Zara Ali”

次の例は、文字データ型の宣言と使用法を示しています。

program hello
implicit none

   character(len = 15) :: surname, firstname 
   character(len = 6) :: title 
   character(len = 25)::greetings
   
   title = 'Mr.' 
   firstname = 'Rowan' 
   surname = 'Atkinson'
   greetings = 'A big hello from Mr. Beans'
   
   print *, 'Here is', title, firstname, surname
   print *, greetings
   
end program hello

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

Here isMr.   Rowan          Atkinson       
A big hello from Mr. Bean

文字列の連結

連結演算子//は、文字列を連結します。

次の例はこれを示しています-

program hello
implicit none

   character(len = 15) :: surname, firstname 
   character(len = 6) :: title 
   character(len = 40):: name
   character(len = 25)::greetings
   
   title = 'Mr.' 
   firstname = 'Rowan' 
   surname = 'Atkinson'
   
   name = title//firstname//surname
   greetings = 'A big hello from Mr. Beans'
   
   print *, 'Here is', name
   print *, greetings
   
end program hello

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

Here is Mr. Rowan Atkinson       
A big hello from Mr. Bean

部分文字列の抽出

Fortranでは、文字列にインデックスを付け、部分文字列の開始インデックスと終了インデックスを角かっこで囲んで、文字列から部分文字列を抽出できます。これはエクステント指定子と呼ばれます。

次の例は、文字列「helloworld」から部分文字列「world」を抽出する方法を示しています。

program subString

   character(len = 11)::hello
   hello = "Hello World"
   print*, hello(7:11)
   
end program subString

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

World

次の例では、 date_and_time日付と時刻の文字列を指定する関数。エクステント指定子を使用して、年、日付、月、時、分、秒の情報を個別に抽出します。

program  datetime
implicit none

   character(len = 8) :: dateinfo ! ccyymmdd
   character(len = 4) :: year, month*2, day*2

   character(len = 10) :: timeinfo ! hhmmss.sss
   character(len = 2)  :: hour, minute, second*6

   call  date_and_time(dateinfo, timeinfo)

   !  let’s break dateinfo into year, month and day.
   !  dateinfo has a form of ccyymmdd, where cc = century, yy = year
   !  mm = month and dd = day

   year  = dateinfo(1:4)
   month = dateinfo(5:6)
   day   = dateinfo(7:8)

   print*, 'Date String:', dateinfo
   print*, 'Year:', year
   print *,'Month:', month
   print *,'Day:', day

   !  let’s break timeinfo into hour, minute and second.
   !  timeinfo has a form of hhmmss.sss, where h = hour, m = minute
   !  and s = second

   hour   = timeinfo(1:2)
   minute = timeinfo(3:4)
   second = timeinfo(5:10)

   print*, 'Time String:', timeinfo
   print*, 'Hour:', hour
   print*, 'Minute:', minute
   print*, 'Second:', second   
   
end program  datetime

上記のプログラムをコンパイルして実行すると、詳細な日付と時刻の情報が表示されます-

Date String: 20140803
Year: 2014
Month: 08
Day: 03
Time String: 075835.466
Hour: 07
Minute: 58
Second: 35.466

文字列のトリミング

ザ・ trim 関数は文字列を受け取り、末尾の空白をすべて削除した後、入力文字列を返します。

program trimString
implicit none

   character (len = *), parameter :: fname="Susanne", sname="Rizwan"
   character (len = 20) :: fullname 
   
   fullname = fname//" "//sname !concatenating the strings
   
   print*,fullname,", the beautiful dancer from the east!"
   print*,trim(fullname),", the beautiful dancer from the east!"
   
end program trimString

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

Susanne Rizwan      , the beautiful dancer from the east!
 Susanne Rizwan, the beautiful dancer from the east!

文字列の左右の調整

関数 adjustl 文字列を受け取り、先頭の空白を削除して末尾の空白として追加することにより、文字列を返します。

関数 adjustr 文字列を受け取り、末尾の空白を削除して先頭の空白として追加することで文字列を返します。

program hello
implicit none

   character(len = 15) :: surname, firstname 
   character(len = 6) :: title 
   character(len = 40):: name
   character(len = 25):: greetings
   
   title = 'Mr. ' 
   firstname = 'Rowan' 
   surname = 'Atkinson'
   greetings = 'A big hello from Mr. Beans'
   
   name = adjustl(title)//adjustl(firstname)//adjustl(surname)
   print *, 'Here is', name
   print *, greetings
   
   name = adjustr(title)//adjustr(firstname)//adjustr(surname)
   print *, 'Here is', name
   print *, greetings
   
   name = trim(title)//trim(firstname)//trim(surname)
   print *, 'Here is', name
   print *, greetings
   
end program hello

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

Here is Mr. Rowan  Atkinson           
A big hello from Mr. Bean
Here is Mr. Rowan Atkinson    
A big hello from Mr. Bean
Here is Mr.RowanAtkinson                        
A big hello from Mr. Bean

文字列内の部分文字列の検索

インデックス関数は2つの文字列を受け取り、2番目の文字列が最初の文字列の部分文字列であるかどうかを確認します。2番目の引数が最初の引数の部分文字列である場合は、最初の文字列の2番目の文字列の開始インデックスである整数を返します。それ以外の場合はゼロを返します。

program hello
implicit none

   character(len=30) :: myString
   character(len=10) :: testString
   
   myString = 'This is a test'
   testString = 'test'
   
   if(index(myString, testString) == 0)then
      print *, 'test is not found'
   else
      print *, 'test is found at index: ', index(myString, testString)
   end if
   
end program hello

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

test is found at index: 11

配列は、同じタイプの要素の固定サイズの順次コレクションを格納できます。配列はデータのコレクションを格納するために使用されますが、配列を同じタイプの変数のコレクションと考える方が便利な場合がよくあります。

すべてのアレイは、連続したメモリ位置で構成されています。最小アドレスは最初の要素に対応し、最大アドレスは最後の要素に対応します。

民数記(1) 民数記(2) 数字(3) ナンバーズ(4)

配列は1次元(ベクトルのように)、2次元(行列のように)にすることができ、Fortranでは最大7次元の配列を作成できます。

配列の宣言

配列はで宣言されます dimension 属性。

たとえば、5つの要素を含む実数のnumberという名前の1次元配列を宣言するには、次のように記述します。

real, dimension(5) :: numbers

配列の個々の要素は、添え字を指定することによって参照されます。配列の最初の要素には、添え字が1つあります。配列numbersには、numbers(1)、numbers(2)、numbers(3)、numbers(4)、numbers(5)の5つの実変数が含まれています。

行列という名前の整数の5x 5 2次元配列を作成するには、次のように記述します。

integer, dimension (5,5) :: matrix

明示的な下限を使用して配列を宣言することもできます。たとえば、-

real, dimension(2:6) :: numbers
integer, dimension (-3:2,0:4) :: matrix

値の割り当て

次のように、個々のメンバーに値を割り当てることができます。

numbers(1) = 2.0

または、ループを使用できます。

do i  =1,5
   numbers(i) = i * 2.0
end do

1次元配列要素には、次のような配列コンストラクターと呼ばれる省略記号を使用して、値を直接割り当てることができます。

numbers = (/1.5, 3.2,4.5,0.9,7.2 /)

please note that there are no spaces allowed between the brackets ‘( ‘and the back slash ‘/’

次の例は、上記の概念を示しています。

program arrayProg

   real :: numbers(5) !one dimensional integer array
   integer :: matrix(3,3), i , j !two dimensional real array
   
   !assigning some values to the array numbers
   do i=1,5
      numbers(i) = i * 2.0
   end do
   
   !display the values
   do i = 1, 5
      Print *, numbers(i)
   end do
   
   !assigning some values to the array matrix
   do i=1,3
      do j = 1, 3
         matrix(i, j) = i+j
      end do
   end do
   
   !display the values
   do i=1,3
      do j = 1, 3
         Print *, matrix(i,j)
      end do
   end do
   
   !short hand assignment
   numbers = (/1.5, 3.2,4.5,0.9,7.2 /)
   
   !display the values
   do i = 1, 5
      Print *, numbers(i)
   end do
   
end program arrayProg

上記のコードをコンパイルして実行すると、次の結果が得られます。

2.00000000    
 4.00000000    
 6.00000000    
 8.00000000    
 10.0000000    
         2
         3
         4
         3
         4
         5
         4
         5
         6
 1.50000000    
 3.20000005    
 4.50000000    
0.899999976    
 7.19999981

いくつかのアレイ関連用語

次の表に、配列に関連するいくつかの用語を示します。

期間 意味
ランク これは、配列の次元数です。たとえば、matrixという名前の配列の場合、rankは2であり、numbersという名前の配列の場合、rankは1です。
エクステント 次元に沿った要素の数です。たとえば、配列番号の範囲は5で、matrixという名前の配列の範囲は両方の次元で3です。
形状 配列の形状は1次元の整数配列であり、各次元の要素の数(範囲)が含まれています。たとえば、配列行列の場合、形状は(3、3)であり、配列番号は(5)です。
サイズ 配列に含まれる要素の数です。配列行列の場合は9、配列番号の場合は5です。

配列をプロシージャに渡す

配列を引数としてプロシージャに渡すことができます。次の例は、概念を示しています-

program arrayToProcedure      
implicit none      

   integer, dimension (5) :: myArray  
   integer :: i
   
   call fillArray (myArray)      
   call printArray(myArray)
   
end program arrayToProcedure


subroutine fillArray (a)      
implicit none      

   integer, dimension (5), intent (out) :: a
   
   ! local variables     
   integer :: i     
   do i = 1, 5         
      a(i) = i      
   end do  
   
end subroutine fillArray 


subroutine printArray(a)

   integer, dimension (5) :: a  
   integer::i
   
   do i = 1, 5
      Print *, a(i)
   end do
   
end subroutine printArray

上記のコードをコンパイルして実行すると、次の結果が得られます。

1
2
3
4
5

上記の例では、サブルーチンfillArrayとprintArrayは、次元5の配列でのみ呼び出すことができます。ただし、任意のサイズの配列に使用できるサブルーチンを作成するには、次の手法を使用して書き換えることができます。

program arrayToProcedure      
implicit  none    

   integer, dimension (10) :: myArray  
   integer :: i
   
   interface 
      subroutine fillArray (a)
         integer, dimension(:), intent (out) :: a 
         integer :: i         
      end subroutine fillArray      

      subroutine printArray (a)
         integer, dimension(:) :: a 
         integer :: i         
      end subroutine printArray   
   end interface 
   
   call fillArray (myArray)      
   call printArray(myArray)
   
end program arrayToProcedure


subroutine fillArray (a)      
implicit none      
   integer,dimension (:), intent (out) :: a      
   
   ! local variables     
   integer :: i, arraySize  
   arraySize = size(a)
   
   do i = 1, arraySize         
      a(i) = i      
   end do  
   
end subroutine fillArray 


subroutine printArray(a)
implicit none

   integer,dimension (:) :: a  
   integer::i, arraySize
   arraySize = size(a)
   
   do i = 1, arraySize
     Print *, a(i)
   end do
   
end subroutine printArray

プログラムはを使用していることに注意してください size 配列のサイズを取得する関数。

上記のコードをコンパイルして実行すると、次の結果が得られます。

1
2
3
4
5
6
7
8
9
10

配列セクション

これまで配列全体を参照してきましたが、Fortranは、単一のステートメントを使用して、複数の要素または配列のセクションを参照する簡単な方法を提供します。

配列セクションにアクセスするには、セクションの下限と上限、およびすべての次元のストライド(増分)を指定する必要があります。この表記は、subscript triplet:

array ([lower]:[upper][:stride], ...)

下限と上限が指定されていない場合、デフォルトで宣言した範囲になり、ストライド値はデフォルトで1になります。

次の例は、概念を示しています-

program arraySubsection

   real, dimension(10) :: a, b
   integer:: i, asize, bsize
   
   a(1:7) = 5.0 ! a(1) to a(7) assigned 5.0
   a(8:) = 0.0  ! rest are 0.0 
   b(2:10:2) = 3.9
   b(1:9:2) = 2.5
   
   !display
   asize = size(a)
   bsize = size(b)
   
   do i = 1, asize
      Print *, a(i)
   end do
   
   do i = 1, bsize
      Print *, b(i)
   end do
   
end program arraySubsection

上記のコードをコンパイルして実行すると、次の結果が得られます。

5.00000000    
5.00000000    
5.00000000    
5.00000000    
5.00000000    
5.00000000    
5.00000000    
0.00000000E+00
0.00000000E+00
0.00000000E+00
2.50000000    
3.90000010    
2.50000000    
3.90000010    
2.50000000    
3.90000010    
2.50000000    
3.90000010    
2.50000000    
3.90000010

配列組み込み関数

Fortran 90/95は、いくつかの固有の手順を提供します。それらは7つのカテゴリーに分けることができます。

  • ベクトルと行列の乗算

  • Reduction

  • Inquiry

  • Construction

  • Reshape

  • Manipulation

  • Location

A dynamic array は配列であり、そのサイズはコンパイル時にはわかりませんが、実行時にはわかります。

動的配列は属性で宣言されます allocatable

例えば、

real, dimension (:,:), allocatable :: darray

配列のランク、つまり次元を指定する必要がありますが、そのような配列にメモリを割り当てるには、 allocate 関数。

allocate ( darray(s1,s2) )

配列が使用された後、プログラムで、作成されたメモリはを使用して解放する必要があります deallocate 関数

deallocate (darray)

次の例は、上記の概念を示しています。

program dynamic_array 
implicit none 

   !rank is 2, but size not known   
   real, dimension (:,:), allocatable :: darray    
   integer :: s1, s2     
   integer :: i, j     
   
   print*, "Enter the size of the array:"     
   read*, s1, s2      
   
   ! allocate memory      
   allocate ( darray(s1,s2) )      
   
   do i = 1, s1           
      do j = 1, s2                
         darray(i,j) = i*j               
         print*, "darray(",i,",",j,") = ", darray(i,j)           
      end do      
   end do      
   
   deallocate (darray)  
end program dynamic_array

上記のコードをコンパイルして実行すると、次の結果が得られます。

Enter the size of the array: 3,4
darray( 1 , 1 ) = 1.00000000    
darray( 1 , 2 ) = 2.00000000    
darray( 1 , 3 ) = 3.00000000    
darray( 1 , 4 ) = 4.00000000    
darray( 2 , 1 ) = 2.00000000    
darray( 2 , 2 ) = 4.00000000    
darray( 2 , 3 ) = 6.00000000    
darray( 2 , 4 ) = 8.00000000    
darray( 3 , 1 ) = 3.00000000    
darray( 3 , 2 ) = 6.00000000    
darray( 3 , 3 ) = 9.00000000    
darray( 3 , 4 ) = 12.0000000

データステートメントの使用

ザ・ data ステートメントは、複数の配列の初期化、または配列セクションの初期化に使用できます。

データステートメントの構文は次のとおりです。

data variable / list / ...

次の例は、概念を示しています-

program dataStatement
implicit none

   integer :: a(5), b(3,3), c(10),i, j
   data a /7,8,9,10,11/ 
   
   data b(1,:) /1,1,1/ 
   data b(2,:)/2,2,2/ 
   data b(3,:)/3,3,3/ 
   data (c(i),i = 1,10,2) /4,5,6,7,8/ 
   data (c(i),i = 2,10,2)/5*2/
   
   Print *, 'The A array:'
   do j = 1, 5                
      print*, a(j)           
   end do 
   
   Print *, 'The B array:'
   do i = lbound(b,1), ubound(b,1)
      write(*,*) (b(i,j), j = lbound(b,2), ubound(b,2))
   end do

   Print *, 'The C array:' 
   do j = 1, 10                
      print*, c(j)           
   end do      
   
end program dataStatement

上記のコードをコンパイルして実行すると、次の結果が得られます。

The A array:
           7
           8
           9
          10
          11
 The B array:
           1           1           1
           2           2           2
           3           3           3
 The C array:
           4
           2
           5
           2
           6
           2
           7
           2
           8
           2

Whereステートメントの使用

ザ・ whereステートメントを使用すると、論理条件の結果に応じて、式で配列の一部の要素を使用できます。指定された条件が真の場合、要素に対して式を実行できます。

次の例は、概念を示しています-

program whereStatement
implicit none

   integer :: a(3,5), i , j
   
   do i = 1,3
      do j = 1, 5                
         a(i,j) = j-i          
      end do 
   end do
   
   Print *, 'The A array:'
   
   do i = lbound(a,1), ubound(a,1)
      write(*,*) (a(i,j), j = lbound(a,2), ubound(a,2))
   end do
   
   where( a<0 ) 
      a = 1 
   elsewhere
      a = 5
   end where
  
   Print *, 'The A array:'
   do i = lbound(a,1), ubound(a,1)
      write(*,*) (a(i,j), j = lbound(a,2), ubound(a,2))
   end do   
   
end program whereStatement

上記のコードをコンパイルして実行すると、次の結果が得られます。

The A array:
           0           1           2           3           4
          -1           0           1           2           3
          -2          -1           0           1           2
 The A array:
           5           5           5           5           5
           1           5           5           5           5
           1           1           5           5           5

Fortranでは、派生データ型を定義できます。派生データ型は構造体とも呼ばれ、さまざまな型のデータオブジェクトで構成できます。

派生データ型は、レコードを表すために使用されます。たとえば、図書館で本を追跡したい場合は、各本に関する次の属性を追跡したい場合があります。

  • Title
  • Author
  • Subject
  • ブックID

派生データ型の定義

派生データを定義するには type、タイプと end typeステートメントが使用されます。。typeステートメントは、プログラムに複数のメンバーを含む新しいデータ型を定義します。typeステートメントの形式は次のとおりです-

type type_name      
   declarations
end type

これがあなたが本の構造を宣言する方法です-

type Books
   character(len = 50) :: title
   character(len = 50) :: author
   character(len = 150) :: subject
   integer :: book_id
end type Books

構造体メンバーへのアクセス

派生データ型のオブジェクトは構造体と呼ばれます。

Books型の構造体は、次のような型宣言ステートメントで作成できます。

type(Books) :: book1

構造体のコンポーネントには、コンポーネントセレクター文字(%)を使用してアクセスできます。

book1%title = "C Programming"
book1%author = "Nuha Ali"
book1%subject = "C Programming Tutorial"
book1%book_id = 6495407

Note that there are no spaces before and after the % symbol.

次のプログラムは、上記の概念を示しています-

program deriveDataType

   !type declaration
   type Books
      character(len = 50) :: title
      character(len = 50) :: author
      character(len = 150) :: subject
      integer :: book_id
   end type Books
   
   !declaring type variables
   type(Books) :: book1 
   type(Books) :: book2 
   
   !accessing the components of the structure
   
   book1%title = "C Programming"
   book1%author = "Nuha Ali"
   book1%subject = "C Programming Tutorial"
   book1%book_id = 6495407 
   
   book2%title = "Telecom Billing"
   book2%author = "Zara Ali"
   book2%subject = "Telecom Billing Tutorial"
   book2%book_id = 6495700
  
   !display book info
   
   Print *, book1%title 
   Print *, book1%author 
   Print *, book1%subject 
   Print *, book1%book_id  
   
   Print *, book2%title 
   Print *, book2%author 
   Print *, book2%subject 
   Print *, book2%book_id  

end program deriveDataType

上記のコードをコンパイルして実行すると、次の結果が得られます。

C Programming                                     
 Nuha Ali                                          
 C Programming Tutorial            
   6495407
 Telecom Billing                                   
 Zara Ali                                          
 Telecom Billing Tutorial            
   6495700

構造体の配列

派生型の配列を作成することもできます-

type(Books), dimension(2) :: list

配列の個々の要素には、次のようにアクセスできます。

list(1)%title = "C Programming"
list(1)%author = "Nuha Ali"
list(1)%subject = "C Programming Tutorial"
list(1)%book_id = 6495407

次のプログラムは、概念を示しています-

program deriveDataType

   !type declaration
   type Books
      character(len = 50) :: title
      character(len = 50) :: author
      character(len = 150) :: subject
      integer :: book_id
   end type Books
   
   !declaring array of books
   type(Books), dimension(2) :: list 
    
   !accessing the components of the structure
   
   list(1)%title = "C Programming"
   list(1)%author = "Nuha Ali"
   list(1)%subject = "C Programming Tutorial"
   list(1)%book_id = 6495407 
   
   list(2)%title = "Telecom Billing"
   list(2)%author = "Zara Ali"
   list(2)%subject = "Telecom Billing Tutorial"
   list(2)%book_id = 6495700
  
   !display book info
   
   Print *, list(1)%title 
   Print *, list(1)%author 
   Print *, list(1)%subject 
   Print *, list(1)%book_id  
   
   Print *, list(1)%title 
   Print *, list(2)%author 
   Print *, list(2)%subject 
   Print *, list(2)%book_id  

end program deriveDataType

上記のコードをコンパイルして実行すると、次の結果が得られます。

C Programming                                     
Nuha Ali                                          
C Programming Tutorial               
   6495407
C Programming                                     
Zara Ali                                          
Telecom Billing Tutorial                                      
   6495700

ほとんどのプログラミング言語では、ポインタ変数はオブジェクトのメモリアドレスを格納します。ただし、Fortranでは、ポインタは単にメモリアドレスを格納するだけではない機能を備えたデータオブジェクトです。タイプ、ランク、エクステント、メモリアドレスなど、特定のオブジェクトに関する詳細情報が含まれています。

ポインターは、割り当てまたはポインター割り当てによってターゲットに関連付けられます。

ポインタ変数の宣言

ポインター変数は、ポインター属性で宣言されます。

次の例は、ポインタ変数の宣言を示しています。

integer, pointer :: p1 ! pointer to integer  
real, pointer, dimension (:) :: pra ! pointer to 1-dim real array  
real, pointer, dimension (:,:) :: pra2 ! pointer to 2-dim real array

ポインタは-を指すことができます

  • 動的に割り当てられたメモリの領域。

  • ポインタと同じタイプのデータオブジェクトで、 target 属性。

ポインタ用のスペースの割り当て

ザ・ allocateステートメントを使用すると、ポインタオブジェクトにスペースを割り当てることができます。例-

program pointerExample
implicit none

   integer, pointer :: p1
   allocate(p1)
   
   p1 = 1
   Print *, p1
   
   p1 = p1 + 4
   Print *, p1
   
end program pointerExample

上記のコードをコンパイルして実行すると、次の結果が得られます。

1
5

割り当てられたストレージスペースを空にする必要があります deallocate 不要になったときのステートメント。未使用および使用不可のメモリスペースの蓄積を回避します。

ターゲットと関連付け

ターゲットは別の正規変数であり、スペースが確保されています。ターゲット変数は、で宣言する必要がありますtarget 属性。

関連付け演算子(=>)を使用して、ポインター変数をターゲット変数に関連付けます。

前の例を書き直して、概念を示しましょう-

program pointerExample
implicit none

   integer, pointer :: p1
   integer, target :: t1 
   
   p1=>t1
   p1 = 1
   
   Print *, p1
   Print *, t1
   
   p1 = p1 + 4
   
   Print *, p1
   Print *, t1
   
   t1 = 8
   
   Print *, p1
   Print *, t1
   
end program pointerExample

上記のコードをコンパイルして実行すると、次の結果が得られます。

1
1
5
5
8
8

ポインタは次のようになります-

  • Undefined
  • Associated
  • Disassociated

上記のプログラムでは、 associated=>演算子を使用して、ターゲットt1を持つポインターp1。関連付けられた関数は、ポインターの関連付けステータスをテストします。

ザ・ nullify ステートメントは、ポインターとターゲットの関連付けを解除します。

同じターゲットを指すポインターが複数存在する可能性があるため、Nullifyはターゲットを空にしません。ただし、ポインタを空にすると、無効化も意味します。

例1

次の例は、概念を示しています-

program pointerExample
implicit none

   integer, pointer :: p1
   integer, target :: t1 
   integer, target :: t2
   
   p1=>t1
   p1 = 1
   
   Print *, p1
   Print *, t1
   
   p1 = p1 + 4
   Print *, p1
   Print *, t1
   
   t1 = 8
   Print *, p1
   Print *, t1
   
   nullify(p1)
   Print *, t1
   
   p1=>t2
   Print *, associated(p1)
   Print*, associated(p1, t1)
   Print*, associated(p1, t2)
   
   !what is the value of p1 at present
   Print *, p1
   Print *, t2
   
   p1 = 10
   Print *, p1
   Print *, t2
   
end program pointerExample

上記のコードをコンパイルして実行すると、次の結果が得られます。

1
1
5
5
8
8
8
T
F
T
952754640
952754640
10
10

コードを実行するたびに、メモリアドレスが異なることに注意してください。

例2

program pointerExample
implicit none

   integer, pointer :: a, b
   integer, target :: t
   integer :: n
   
   t = 1
   a => t
   t = 2
   b => t
   n = a + b
   
   Print *, a, b, t, n 
   
end program pointerExample

上記のコードをコンパイルして実行すると、次の結果が得られます。

2  2  2  4

これまで、キーボードからデータを読み取ることができることを確認しました。 read * ステートメント、およびを使用して画面に出力を表示します print*それぞれステートメント。この形式の入出力はfree format I / O、それは呼ばれます list-directed 入出力。

フリーフォーマットのシンプルI / Oの形式は-です。

read(*,*) item1, item2, item3...
print *, item1, item2, item3
write(*,*) item1, item2, item3...

ただし、フォーマットされたI / Oを使用すると、データ転送の柔軟性が高まります。

フォーマットされた入出力

フォーマットされた入出力の構文は次のとおりです。

read fmt, variable_list 
print fmt, variable_list 
write fmt, variable_list

どこ、

  • fmtはフォーマット仕様です

  • variable-listは、キーボードから読み取られる、または画面に書き込まれる変数のリストです。

フォーマット仕様は、フォーマットされたデータが表示される方法を定義します。これは、次のリストを含む文字列で構成されます。edit descriptors 括弧内。

アン edit descriptor 文字や数字が表示される正確な形式(幅、小数点以下の桁数など)を指定します。

例えば

Print "(f6.3)", pi

次の表で、記述子について説明します-

ディスクリプタ 説明

これは整数出力に使用されます。これは「rIw.m」の形式を取ります。ここで、r、w、およびmの意味は次の表に示されています。整数値は、そのフィールドで右寄せされます。フィールド幅が整数を収容するのに十分な大きさでない場合、フィールドはアスタリスクで埋められます。

印刷 "(3i5)"、i、j、k
F

実数出力に使用します。これは「rFw.d」の形式を取ります。ここで、r、w、およびdの意味は次の表に示されています。実数値は、その分野で正しく正当化されます。フィールド幅が実数を収容するのに十分な大きさでない場合、フィールドはアスタリスクで埋められます。

印刷 "(f​​12.3)"、pi
E

これは、指数表記の実際の出力に使用されます。'E'記述子ステートメントは 'rEw.d'の形式を取ります。ここで、r、w、およびdの意味は次の表に示されています。実数値は、その分野で正しく正当化されます。フィールド幅が実数を収容するのに十分な大きさでない場合、フィールドはアスタリスクで埋められます。

小数点以下3桁の実数を印刷するには、少なくとも10のフィールド幅が必要であることに注意してください。1つは仮数の符号、2つはゼロ、4つは仮数、2つは指数自体です。一般に、w≥d+ 7。

print "(e10.3)"、123456.0は「0.123e +06」を返します
ES

これは実際の出力(科学的記数法)に使用されます。これは「rESw.d」の形式を取ります。ここで、r、w、およびdの意味は次の表に示されています。上記の「E」記述子は、従来のよく知られている「科学的記数法」とは少し異なります。科学的記数法では、仮数が0.1から1.0の範囲にあるE記述子とは異なり、仮数は1.0〜10.0の範囲にあります。実数値は、その分野で正しく正当化されます。フィールド幅が実数を収容するのに十分な大きさでない場合、フィールドはアスタリスクで埋められます。ここでも、幅フィールドは式w≥d+ 7を満たす必要があります

print "(es10.3)"、123456.0は「1.235e +05」を返します
A

文字出力に使用します。これは「rAw」の形式を取り、rとwの意味は次の表に示されています。文字タイプは、そのフィールドで正しく正当化されます。フィールド幅が文字列を収容するのに十分な大きさでない場合、フィールドは文字列の最初の「w」文字で埋められます。

印刷 "(a10)"、str
バツ

これはスペース出力に使用されます。これは「nX」の形式を取ります。ここで、「n」は必要なスペースの数です。

印刷 "(5x、a10)"、str
/

スラッシュ記述子–空白行を挿入するために使用されます。これは「/」の形式を取り、次のデータ出力を強制的に改行します。

印刷 "(/、5x、a10)"、str

次の記号は、フォーマット記述子で使用されます-

シニア番号 記号と説明
1

c

列番号

2

d

実際の入力または出力の小数点以下の桁数

3

m

表示する最小桁数

4

n

スキップするスペースの数

5

r

繰り返し回数–記述子または記述子のグループを使用する回数

6

w

フィールド幅–入力または出力に使用する文字数

例1

program printPi

   pi = 3.141592653589793238 
   
   Print "(f6.3)", pi 
   Print "(f10.7)", pi
   Print "(f20.15)", pi 
   Print "(e16.4)", pi/100 
   
end program printPi

上記のコードをコンパイルして実行すると、次の結果が得られます。

3.142
3.1415927
3.141592741012573
0.3142E-01

例2

program printName
implicit none

   character (len = 15) :: first_name
   print *,' Enter your first name.' 
   print *,' Up to 20 characters, please'
   
   read *,first_name 
   print "(1x,a)",first_name
   
end program printName

上記のコードをコンパイルして実行すると、次の結果が生成されます:(ユーザーがZaraという名前を入力したと仮定します)

Enter your first name.
Up to 20 characters, please
Zara

例3

program formattedPrint
implicit none

   real :: c = 1.2786456e-9, d = 0.1234567e3 
   integer :: n = 300789, k = 45, i = 2
   character (len=15) :: str="Tutorials Point"
   
   print "(i6)", k 
   print "(i6.3)", k 
   print "(3i10)", n, k, i 
   print "(i10,i3,i5)", n, k, i 
   print "(a15)",str 
   print "(f12.3)", d
   print "(e12.4)", c 
   print '(/,3x,"n = ",i6, 3x, "d = ",f7.4)', n, d
   
end program formattedPrint

上記のコードをコンパイルして実行すると、次の結果が得られます。

45
045
300789 45  2
300789 45  2
Tutorials Point
123.457
0.1279E-08

n = 300789 d = *******

フォーマットステートメント

formatステートメントを使用すると、1つのステートメントで文字、整数、および実数の出力を組み合わせることができます。次の例はこれを示しています-

program productDetails 
implicit none 

   character (len = 15) :: name
   integer :: id 
   real :: weight
   name = 'Ardupilot'
   id = 1
   weight = 0.08
   
   print *,' The product details are' 
   
   print 100
   100 format (7x,'Name:', 7x, 'Id:', 1x, 'Weight:')
   
   print 200, name, id, weight 
   200 format(1x, a, 2x, i3, 2x, f5.2) 
   
end program productDetails

上記のコードをコンパイルして実行すると、次の結果が得られます。

The product details are
Name:       Id:    Weight:
Ardupilot   1       0.08

Fortranを使用すると、ファイルからデータを読み取ったり、ファイルにデータを書き込んだりできます。

前の章では、端末からデータを読み取り、端末にデータを書き込む方法を見てきました。この章では、Fortranが提供するファイルの入出力機能について学習します。

1つ以上のファイルの読み取りと書き込みを行うことができます。OPEN、WRITE、READ、およびCLOSEステートメントを使用すると、これを実現できます。

ファイルを開いたり閉じたりする

ファイルを使用する前に、ファイルを開く必要があります。ザ・openコマンドは、読み取りまたは書き込み用にファイルを開くために使用されます。コマンドの最も単純な形式は次のとおりです。

open (unit = number, file = "name").

ただし、オープンステートメントは一般的な形式である可能性があります-

open (list-of-specifiers)

次の表に、最も一般的に使用される指定子を示します。

シニア番号 指定子と説明
1

[UNIT=] u

ユニット番号uは、9〜99の範囲の任意の番号で、ファイルを示します。任意の番号を選択できますが、プログラムで開いているすべてのファイルには一意の番号が必要です。

2

IOSTAT= ios

これはI / Oステータス識別子であり、整数変数である必要があります。openステートメントが成功した場合、返されるios値はゼロです。それ以外の場合はゼロ以外の値です。

3

ERR = err

これは、エラーが発生した場合にコントロールがジャンプするラベルです。

4

FILE = fname

ファイル名、文字列。

5

STATUS = sta

ファイルの以前のステータスが表示されます。文字列であり、NEW、OLD、またはSCRATCHの3つの値のいずれかを持つことができます。スクラッチファイルは、閉じるかプログラムが終了すると作成および削除されます。

6

ACCESS = acc

ファイルアクセスモードです。SEQUENTIALまたはDIRECTの2つの値のいずれかを持つことができます。デフォルトはSEQUENTIALです。

7

FORM = frm

ファイルのフォーマットステータスを示します。FORMATTEDまたはUNFORMATTEDの2つの値のいずれかを持つことができます。デフォルトはUNFORMATTEDです

8

RECL = rl

直接アクセスファイルの各レコードの長さを指定します。

ファイルが開かれた後、読み取りおよび書き込みステートメントによってアクセスされます。完了したら、を使用して閉じる必要がありますclose ステートメント。

closeステートメントの構文は次のとおりです-

close ([UNIT = ]u[,IOSTAT = ios,ERR = err,STATUS = sta])

括弧内のパラメーターはオプションであることに注意してください。

Example

この例は、ファイルにデータを書き込むために新しいファイルを開く方法を示しています。

program outputdata   
implicit none

   real, dimension(100) :: x, y  
   real, dimension(100) :: p, q
   integer :: i  
   
   ! data  
   do i=1,100  
      x(i) = i * 0.1 
      y(i) = sin(x(i)) * (1-cos(x(i)/3.0))  
   end do  
   
   ! output data into a file 
   open(1, file = 'data1.dat', status = 'new')  
   do i=1,100  
      write(1,*) x(i), y(i)   
   end do  
   
   close(1) 
   
end program outputdata

上記のコードをコンパイルして実行すると、ファイルdata1.datが作成され、x配列とy配列の値が書き込まれます。そして、ファイルを閉じます。

ファイルからの読み取りとファイルへの書き込み

readステートメントとwriteステートメントは、それぞれファイルからの読み取りとファイルへの書き込みに使用されます。

それらは次の構文を持っています-

read ([UNIT = ]u, [FMT = ]fmt, IOSTAT = ios, ERR = err, END = s)
write([UNIT = ]u, [FMT = ]fmt, IOSTAT = ios, ERR = err, END = s)

ほとんどの指定子については、上記の表ですでに説明しています。

END = s指定子は、プログラムがファイルの終わりに達したときにジャンプするステートメントラベルです。

Example

この例は、ファイルからの読み取りとファイルへの書き込みを示しています。

このプログラムでは、最後の例で作成したファイルdata1.datから読み取り、画面に表示します。

program outputdata   
implicit none   

   real, dimension(100) :: x, y  
   real, dimension(100) :: p, q
   integer :: i  
   
   ! data  
   do i = 1,100  
      x(i) = i * 0.1 
      y(i) = sin(x(i)) * (1-cos(x(i)/3.0))  
   end do  
   
   ! output data into a file 
   open(1, file = 'data1.dat', status='new')  
   do i = 1,100  
      write(1,*) x(i), y(i)   
   end do  
   close(1) 

   ! opening the file for reading
   open (2, file = 'data1.dat', status = 'old')

   do i = 1,100  
      read(2,*) p(i), q(i)
   end do 
   
   close(2)
   
   do i = 1,100  
      write(*,*) p(i), q(i)
   end do 
   
end program outputdata

上記のコードをコンパイルして実行すると、次の結果が得られます。

0.100000001  5.54589933E-05
0.200000003  4.41325130E-04
0.300000012  1.47636665E-03
0.400000006  3.45637114E-03
0.500000000  6.64328877E-03
0.600000024  1.12552457E-02
0.699999988  1.74576249E-02
0.800000012  2.53552198E-02
0.900000036  3.49861123E-02
1.00000000   4.63171229E-02
1.10000002   5.92407547E-02
1.20000005   7.35742599E-02
1.30000007   8.90605897E-02
1.39999998   0.105371222    
1.50000000   0.122110792    
1.60000002   0.138823599    
1.70000005   0.155002072    
1.80000007   0.170096487    
1.89999998   0.183526158    
2.00000000   0.194692180    
2.10000014   0.202990443    
2.20000005   0.207826138    
2.29999995   0.208628103    
2.40000010   0.204863414    
2.50000000   0.196052119    
2.60000014   0.181780845    
2.70000005   0.161716297    
2.79999995   0.135617107    
2.90000010   0.103344671    
3.00000000   6.48725405E-02
3.10000014   2.02930309E-02
3.20000005  -3.01767997E-02
3.29999995  -8.61928314E-02
3.40000010  -0.147283033    
3.50000000  -0.212848678    
3.60000014  -0.282169819    
3.70000005  -0.354410470    
3.79999995  -0.428629100    
3.90000010  -0.503789663    
4.00000000  -0.578774154    
4.09999990  -0.652400017    
4.20000029  -0.723436713    
4.30000019  -0.790623367    
4.40000010  -0.852691114    
4.50000000  -0.908382416    
4.59999990  -0.956472993    
4.70000029  -0.995793998    
4.80000019  -1.02525222    
4.90000010  -1.04385209    
5.00000000  -1.05071592    
5.09999990  -1.04510069    
5.20000029  -1.02641726    
5.30000019  -0.994243503    
5.40000010  -0.948338211    
5.50000000  -0.888650239    
5.59999990  -0.815326691    
5.70000029  -0.728716135    
5.80000019  -0.629372001    
5.90000010  -0.518047631    
6.00000000  -0.395693362    
6.09999990  -0.263447165    
6.20000029  -0.122622721    
6.30000019   2.53026206E-02
6.40000010   0.178709000    
6.50000000   0.335851669    
6.59999990   0.494883657    
6.70000029   0.653881252    
6.80000019   0.810866773    
6.90000010   0.963840425    
7.00000000   1.11080539    
7.09999990   1.24979746    
7.20000029   1.37891412    
7.30000019   1.49633956    
7.40000010   1.60037732    
7.50000000   1.68947268    
7.59999990   1.76223695    
7.70000029   1.81747139    
7.80000019   1.85418403    
7.90000010   1.87160957    
8.00000000   1.86922085    
8.10000038   1.84674001    
8.19999981   1.80414569    
8.30000019   1.74167395    
8.40000057   1.65982044    
8.50000000   1.55933595    
8.60000038   1.44121361    
8.69999981   1.30668485    
8.80000019   1.15719533    
8.90000057   0.994394958    
9.00000000   0.820112705    
9.10000038   0.636327863    
9.19999981   0.445154816    
9.30000019   0.248800844    
9.40000057   4.95488606E-02
9.50000000  -0.150278628    
9.60000038  -0.348357052    
9.69999981  -0.542378068    
9.80000019  -0.730095863    
9.90000057  -0.909344316    
10.0000000  -1.07807255

A procedureは、明確に定義されたタスクを実行し、プログラムから呼び出すことができるステートメントのグループです。情報(またはデータ)は、呼び出し元のプログラムに、引数としてプロシージャーに渡されます。

手順には2つのタイプがあります-

  • Functions
  • Subroutines

関数

関数は、単一の数量を返すプロシージャです。関数は引数を変更してはなりません。

返される数量は、 function value、および関数名で示されます。

Syntax

関数の構文は次のとおりです-

function name(arg1, arg2, ....)  
   [declarations, including those for the arguments]   
   [executable statements] 
end function [name]

次の例は、area_of_circleという名前の関数を示しています。半径rの円の面積を計算します。

program calling_func

   real :: a
   a = area_of_circle(2.0) 
   
   Print *, "The area of a circle with radius 2.0 is"
   Print *, a
   
end program calling_func


! this function computes the area of a circle with radius r  
function area_of_circle (r)  

! function result     
implicit none      

   ! dummy arguments        
   real :: area_of_circle   
   
   ! local variables 
   real :: r     
   real :: pi
   
   pi = 4 * atan (1.0)     
   area_of_circle = pi * r**2  
   
end function area_of_circle

上記のプログラムをコンパイルして実行すると、次の結果が得られます-

The area of a circle with radius 2.0 is
   12.5663710

注意してください-

  • 指定する必要があります implicit none メインプログラムと手順の両方で。

  • 呼び出された関数の引数rが呼び出されます dummy argument

結果オプション

戻り値を関数名以外の名前で保存したい場合は、 result オプション。

戻り変数名を-として指定できます

function name(arg1, arg2, ....) result (return_var_name)  
   [declarations, including those for the arguments]   
   [executable statements] 
end function [name]

サブルーチン

サブルーチンは値を返しませんが、引数を変更することはできます。

Syntax

subroutine name(arg1, arg2, ....)    
   [declarations, including those for the arguments]    
   [executable statements]  
end subroutine [name]

サブルーチンの呼び出し

を使用してサブルーチンを呼び出す必要があります call ステートメント。

次の例は、引数の値を変更するサブルーチンスワップの定義と使用法を示しています。

program calling_func
implicit none

   real :: a, b
   a = 2.0
   b = 3.0
   
   Print *, "Before calling swap"
   Print *, "a = ", a
   Print *, "b = ", b
   
   call swap(a, b)
   
   Print *, "After calling swap"
   Print *, "a = ", a
   Print *, "b = ", b
   
end program calling_func


subroutine swap(x, y) 
implicit none

   real :: x, y, temp   
   
   temp = x  
   x = y 
   y = temp  
   
end subroutine swap

上記のプログラムをコンパイルして実行すると、次の結果が得られます-

Before calling swap
a = 2.00000000    
b = 3.00000000    
After calling swap
a = 3.00000000    
b = 2.00000000

引数の意図を指定する

intent属性を使用すると、プロシージャで引数を使用する意図を指定できます。次の表に、インテント属性の値を示します。

使用されます 説明
インテント(in) 入力値として使用され、関数では変更されません
アウト インテント(アウト) 出力値として使用され、上書きされます
inout インテント(inout) 引数は使用され、上書きされます

次の例は、概念を示しています-

program calling_func
implicit none

   real :: x, y, z, disc
   
   x = 1.0
   y = 5.0
   z = 2.0
   
   call intent_example(x, y, z, disc)
   
   Print *, "The value of the discriminant is"
   Print *, disc
   
end program calling_func


subroutine intent_example (a, b, c, d)     
implicit none     

   ! dummy arguments      
   real, intent (in) :: a     
   real, intent (in) :: b      
   real, intent (in) :: c    
   real, intent (out) :: d   
   
   d = b * b - 4.0 * a * c 
   
end subroutine intent_example

上記のプログラムをコンパイルして実行すると、次の結果が得られます-

The value of the discriminant is
   17.0000000

再帰的手順

再帰は、プログラミング言語で同じ関数内の関数を呼び出すことができる場合に発生します。これは、関数の再帰呼び出しと呼ばれます。

プロシージャがそれ自体を直接的または間接的に呼び出す場合、再帰的プロシージャと呼ばれます。このタイプの手順は、単語の前に宣言する必要がありますrecursive その宣言の前に。

関数を再帰的に使用すると、 result オプションを使用する必要があります。

以下は、再帰的手順を使用して特定の数の階乗を計算する例です。

program calling_func
implicit none

   integer :: i, f
   i = 15
   
   Print *, "The value of factorial 15 is"
   f = myfactorial(15)
   Print *, f
   
end program calling_func

! computes the factorial of n (n!)      
recursive function myfactorial (n) result (fac)  
! function result     
implicit none     

   ! dummy arguments     
   integer :: fac     
   integer, intent (in) :: n     
   
   select case (n)         
      case (0:1)         
         fac = 1         
      case default    
         fac = n * myfactorial (n-1)  
   end select 
   
end function myfactorial

内部手順

プロシージャがプログラム内に含まれている場合、それはプログラムの内部プロシージャと呼ばれます。内部プロシージャを含めるための構文は次のとおりです。

program program_name     
   implicit none         
   ! type declaration statements         
   ! executable statements    
   . . .     
   contains         
   ! internal procedures      
   . . .  
end program program_name

次の例は、概念を示しています-

program mainprog  
implicit none 

   real :: a, b 
   a = 2.0
   b = 3.0
   
   Print *, "Before calling swap"
   Print *, "a = ", a
   Print *, "b = ", b
   
   call swap(a, b)
   
   Print *, "After calling swap"
   Print *, "a = ", a
   Print *, "b = ", b
 
contains   
   subroutine swap(x, y)     
      real :: x, y, temp      
      temp = x 
      x = y  
      y = temp   
   end subroutine swap 
   
end program mainprog

上記のプログラムをコンパイルして実行すると、次の結果が得られます-

Before calling swap
a = 2.00000000    
b = 3.00000000    
After calling swap
a = 3.00000000    
b = 2.00000000

モジュールは、非常に大きなプログラムを作成している場合や、関数やサブルーチンを複数のプログラムで使用できる場合に備えて、関数やサブルーチンを保持できるパッケージのようなものです。

モジュールは、プログラムを複数のファイルに分割する方法を提供します。

モジュールは次の目的で使用されます-

  • サブプログラム、データ、およびインターフェイスブロックのパッケージ化。

  • 複数のルーチンで使用できるグローバルデータを定義します。

  • 選択した任意のルーチン内で使用可能にすることができる変数を宣言します。

  • モジュールを完全にインポートして、使用するために、別のプログラムまたはサブルーチンにインポートします。

モジュールの構文

モジュールは2つの部分で構成されています-

  • ステートメント宣言の仕様部分
  • aには、サブルーチンと関数の定義の一部が含まれています

モジュールの一般的な形式は次のとおりです。

module name     
   [statement declarations]  
   [contains [subroutine and function definitions] ] 
end module [name]

プログラムへのモジュールの使用

useステートメントによってモジュールをプログラムまたはサブルーチンに組み込むことができます-

use name

その点に注意してください

  • 必要な数のモジュールを追加できます。各モジュールは別々のファイルにあり、別々にコンパイルされます。

  • モジュールは、さまざまなプログラムで使用できます。

  • モジュールは同じプログラムで何度も使用できます。

  • モジュール仕様部分で宣言された変数は、モジュールに対してグローバルです。

  • モジュールで宣言された変数は、モジュールが使用されるすべてのプログラムまたはルーチンでグローバル変数になります。

  • useステートメントは、メインプログラム、または特定のモジュールで宣言されたルーチンまたは変数を使用するその他のサブルーチンまたはモジュールに表示できます。

次の例は、概念を示しています-

module constants  
implicit none 

   real, parameter :: pi = 3.1415926536  
   real, parameter :: e = 2.7182818285 
   
contains      
   subroutine show_consts()          
      print*, "Pi = ", pi          
      print*,  "e = ", e     
   end subroutine show_consts 
   
end module constants 


program module_example     
use constants      
implicit none     

   real :: x, ePowerx, area, radius 
   x = 2.0
   radius = 7.0
   ePowerx = e ** x
   area = pi * radius**2     
   
   call show_consts() 
   
   print*, "e raised to the power of 2.0 = ", ePowerx
   print*, "Area of a circle with radius 7.0 = ", area  
   
end program module_example

上記のプログラムをコンパイルして実行すると、次の結果が得られます-

Pi = 3.14159274    
e =  2.71828175    
e raised to the power of 2.0 = 7.38905573    
Area of a circle with radius 7.0 = 153.938049

モジュール内の変数とサブルーチンのアクセシビリティ

デフォルトでは、モジュール内のすべての変数とサブルーチンは、モジュールコードを使用しているプログラムで利用可能になります。 use ステートメント。

ただし、を使用してモジュールコードのアクセシビリティを制御できます。 private そして public属性。変数またはサブルーチンをプライベートとして宣言すると、モジュールの外部では使用できなくなります。

次の例は、概念を示しています-

前の例では、2つのモジュール変数がありました。 e そして pi. それらをプライベートにして、出力を観察しましょう-

module constants  
implicit none 

   real, parameter,private :: pi = 3.1415926536  
   real, parameter, private :: e = 2.7182818285 
   
contains      
   subroutine show_consts()          
      print*, "Pi = ", pi          
      print*, "e = ", e     
   end subroutine show_consts 
   
end module constants 


program module_example     
use constants      
implicit none     

   real :: x, ePowerx, area, radius 
   x = 2.0
   radius = 7.0
   ePowerx = e ** x
   area = pi * radius**2     
   
   call show_consts() 
   
   print*, "e raised to the power of 2.0 = ", ePowerx
   print*, "Area of a circle with radius 7.0 = ", area  
   
end program module_example

上記のプログラムをコンパイルして実行すると、次のエラーメッセージが表示されます-

ePowerx = e ** x
   1
Error: Symbol 'e' at (1) has no IMPLICIT type
main.f95:19.13:

   area = pi * radius**2     
   1
Error: Symbol 'pi' at (1) has no IMPLICIT type

以来 e そして pi, 両方ともプライベートとして宣言され、プログラムmodule_exampleはこれらの変数にアクセスできなくなります。

ただし、他のモジュールサブルーチンはそれらにアクセスできます-

module constants  
implicit none 

   real, parameter,private :: pi = 3.1415926536  
   real, parameter, private :: e = 2.7182818285 
   
contains      
   subroutine show_consts()          
      print*, "Pi = ", pi          
      print*, "e = ", e     
   end subroutine show_consts 
   
   function ePowerx(x)result(ePx) 
   implicit none
      real::x
      real::ePx
      ePx = e ** x
   end function ePowerx
    
   function areaCircle(r)result(a)  
   implicit none
      real::r
      real::a
      a = pi * r**2  
   end function areaCircle
    
end module constants 


program module_example     
use constants      
implicit none     

   call show_consts() 
   
   Print*, "e raised to the power of 2.0 = ", ePowerx(2.0)
   print*, "Area of a circle with radius 7.0 = ", areaCircle(7.0)  
   
end program module_example

上記のプログラムをコンパイルして実行すると、次の結果が得られます-

Pi = 3.14159274    
e = 2.71828175    
e raised to the power of 2.0 = 7.38905573    
Area of a circle with radius 7.0 = 153.938049

組み込み関数は、Fortran言語の一部として提供されるいくつかの一般的で重要な関数です。これらの関数のいくつかについては、配列、文​​字、文字列の章ですでに説明しました。

組み込み関数は次のように分類できます-

  • 数値関数
  • 数学関数
  • 数値照会機能
  • 浮動小数点操作関数
  • ビット操作機能
  • 文字機能
  • 種類の機能
  • 論理関数
  • 配列関数。

配列の章で配列関数について説明しました。次のセクションでは、他のカテゴリのこれらすべての機能について簡単に説明します。

関数名の列で、

  • Aは任意のタイプの数値変数を表します
  • Rは実数または整数の変数を表します
  • XとYは実変数を表します
  • Zは複素変数を表します
  • Wは実変数または複素変数を表します

数値関数

シニア番号 機能と説明
1

ABS (A)

Aの絶対値を返します

2

AIMAG (Z)

複素数Zの虚数部を返します

3

AINT (A [, KIND])

Aの小数部分をゼロに向かって切り捨て、実数の整数を返します。

4

ANINT (A [, KIND])

実数値、最も近い整数または整数を返します。

5

CEILING (A [, KIND])

数値A以上の最小の整数を返します。

6

CMPLX (X [, Y, KIND])

実変数XとYを複素数X + iYに変換します。Yがない場合は、0が使用されます。

7

CONJG (Z)

複素数Zの複素共役を返します。

8

DBLE (A)

Aを倍精度の実数に変換します。

9

DIM (X, Y)

XとYの正の差を返します。

10

DPROD (X, Y)

XとYの倍精度の実積を返します。

11

FLOOR (A [, KIND])

数値A以下の最大の整数を提供します。

12

INT (A [, KIND])

数値(実数または整数)を整数に変換し、実数部をゼロに向かって切り捨てます。

13

MAX (A1, A2 [, A3,...])

引数から最大値を返し、すべて同じタイプです。

14

MIN (A1, A2 [, A3,...])

引数から最小値を返します。すべて同じタイプです。

15

MOD (A, P)

Pで除算すると、Aの余りが返されます。両方の引数は同じ型です(A-INT(A / P)* P)

16

MODULO (A, P)

Pを法としてAを返します:(A-FLOOR(A / P)* P)

17

NINT (A [, KIND])

数値Aの最も近い整数を返します

18

REAL (A [, KIND])

実数型に変換します

19

SIGN (A, B)

Aの絶対値にPの符号を掛けた値を返します。基本的にはBの符号をAに転送します。

program numericFunctions
implicit none  

   ! define constants  
   ! define variables
   real :: a, b 
   complex :: z
   
   ! values for a, b 
   a = 15.2345
   b = -20.7689
    
   write(*,*) 'abs(a): ',abs(a),' abs(b): ',abs(b)   
   write(*,*) 'aint(a): ',aint(a),' aint(b): ',aint(b) 
   write(*,*) 'ceiling(a): ',ceiling(a),' ceiling(b): ',ceiling(b)   
   write(*,*) 'floor(a): ',floor(a),' floor(b): ',floor(b)  
    
   z = cmplx(a, b)
   write(*,*) 'z: ',z   
   
end program numericFunctions

上記のプログラムをコンパイルして実行すると、次の結果が得られます-

abs(a): 15.2344999   abs(b): 20.7688999    
aint(a): 15.0000000  aint(b): -20.0000000    
ceiling(a): 16  ceiling(b): -20
floor(a): 15  floor(b): -21
z: (15.2344999, -20.7688999)

数学関数

シニア番号 機能と説明
1

ACOS (X)

(0、π)の範囲の逆コサインをラジアンで返します。

2

ASIN (X)

(-π/ 2、π/ 2)の範囲の逆正弦をラジアンで返します。

3

ATAN (X)

範囲(-π/ 2、π/ 2)の逆タンジェントをラジアンで返します。

4

ATAN2 (Y, X)

範囲(-π、π)の逆タンジェントをラジアンで返します。

5

COS (X)

引数のコサインをラジアンで返します。

6

COSH (X)

引数の双曲線余弦をラジアンで返します。

7

EXP (X)

Xの指数値を返します。

8

LOG (X)

Xの自然対数値を返します。

9

LOG10 (X)

Xの常用対数(基数10)値を返します。

10

SIN (X)

引数のサインをラジアンで返します。

11

SINH (X)

引数の双曲線正弦をラジアンで返します。

12

SQRT (X)

Xの平方根を返します。

13

TAN (X)

引数の接線をラジアンで返します。

14

TANH (X)

引数の双曲線タンジェントをラジアンで返します。

次のプログラムは、時間t −後の発射体の水平位置と垂直位置xとyをそれぞれ計算します。

ここで、x = ut cosaおよびy = ut sin a --g t2 / 2

program projectileMotion  
implicit none  

   ! define constants  
   real, parameter :: g = 9.8  
   real, parameter :: pi = 3.1415927  
   
   !define variables
   real :: a, t, u, x, y   
   
   !values for a, t, and u 
   a = 45.0
   t = 20.0
   u = 10.0
   
   ! convert angle to radians  
   a = a * pi / 180.0  
   x = u * cos(a) * t   
   y = u * sin(a) * t - 0.5 * g * t * t  
   
   write(*,*) 'x: ',x,'  y: ',y   
   
end program projectileMotion

上記のプログラムをコンパイルして実行すると、次の結果が得られます-

x: 141.421356  y: -1818.57861

数値照会機能

これらの関数は、整数および浮動小数点演算の特定のモデルで機能します。関数は、変数Xと同じ種類の数値のプロパティを返します。これは実数であり、場合によっては整数です。

シニア番号 機能と説明
1

DIGITS (X)

モデルの有効桁数を返します。

2

EPSILON (X)

1と比較してほとんど無視できる数を返します。つまり、REAL(1.0、KIND(X))+ EPSILON(X)がREAL(1.0、KIND(X))と等しくないような最小値を返します。

3

HUGE (X)

モデルの最大数を返します

4

MAXEXPONENT (X)

モデルの最大指数を返します

5

MINEXPONENT (X)

モデルの最小指数を返します

6

PRECISION (X)

10進精度を返します

7

RADIX (X)

モデルのベースを返します

8

RANGE (X)

10進数の指数範囲を返します

9

TINY (X)

モデルの最小の正の数を返します

浮動小数点操作関数

シニア番号 機能と説明
1

EXPONENT (X)

モデル番号の指数部分を返します

2

FRACTION (X)

数値の小数部分を返します

3

NEAREST (X, S)

指定された方向に最も近い異なるプロセッサ番号を返します

4

RRSPACING (X)

指定された数値に近いモデル番号の相対間隔の逆数を返します

5

SCALE (X, I)

実数にその底を掛けて整数乗します

6

SET_EXPONENT (X, I)

数値の指数部分を返します

7

SPACING (X)

指定された数値に近いモデル番号の絶対間隔を返します

ビット操作機能

シニア番号 機能と説明
1

BIT_SIZE (I)

モデルのビット数を返します

2

BTEST (I, POS)

ビットテスト

3

IAND (I, J)

論理積

4

IBCLR (I, POS)

クリアビット

5

IBITS (I, POS, LEN)

ビット抽出

6

IBSET (I, POS)

セットビット

7

IEOR (I, J)

排他的論理和

8

IOR (I, J)

包括的または

9

ISHFT (I, SHIFT)

論理シフト

10

ISHFTC (I, SHIFT [, SIZE])

循環シフト

11

NOT (I)

論理的補数

文字機能

シニア番号 機能と説明
1

ACHAR (I)

ASCII照合シーケンスのI番目の文字を返します。

2

ADJUSTL (STRING)

先頭の空白を削除し、末尾の空白を挿入することで、左の文字列を調整します

3

ADJUSTR (STRING)

末尾の空白を削除し、先頭の空白を挿入することで、文字列を正しく調整します。

4

CHAR (I [, KIND])

マシン固有の照合シーケンスのI番目の文字を返します

5

IACHAR (C)

ASCII照合シーケンス内の文字の位置を返します。

6

ICHAR (C)

マシン(プロセッサ)固有の照合シーケンス内の文字の位置を返します。

7

INDEX (STRING, SUBSTRING [, BACK])

STRING内のSUBSTRINGの左端(BACKが.TRUEの場合は右端)の開始位置を返します。

8

LEN (STRING)

文字列の長さを返します。

9

LEN_TRIM (STRING)

末尾の空白文字なしで文字列の長さを返します。

10

LGE (STRING_A, STRING_B)

字句的に以上

11

LGT (STRING_A, STRING_B)

字句的により大きい

12

LLE (STRING_A, STRING_B)

字句的に以下

13

LLT (STRING_A, STRING_B)

字句的に

14

REPEAT (STRING, NCOPIES)

繰り返される連結

15

SCAN (STRING, SET [, BACK])

SETに属するSTRINGの左端(BACKが.TRUEの場合は右端)の文字のインデックスを返します。何も属していない場合は0を返します。

16

TRIM (STRING)

末尾の空白文字を削除します

17

VERIFY (STRING, SET [, BACK])

文字列内の文字のセットを検証します

種類の機能

シニア番号 機能と説明
1

KIND (X)

種類タイプのパラメータ値を返します。

2

SELECTED_INT_KIND (R)

指定された指数範囲のタイプパラメータの種類を返します。

3

SELECTED_REAL_KIND ([P, R])

精度と範囲が指定された、実際の種類のタイプのパラメーター値

論理関数

シニア番号 機能と説明
1

LOGICAL (L [, KIND])

異なる種類の型パラメーターを持つ論理型のオブジェクト間で変換する

以前のバージョンのFortranには2つあったことはすでに説明しました real タイプ:デフォルトの実際のタイプと double precision タイプ。

ただし、Fortran 90/95では、実数および整数のデータ型の精度を kind 具体的に。

種類属性

コンピュータ内には、さまざまな種類の番号がさまざまに格納されます。ザ・kind属性を使用すると、数値を内部的に保存する方法を指定できます。例えば、

real, kind = 2 :: a, b, c
real, kind = 4 :: e, f, g
integer, kind = 2 :: i, j, k
integer, kind = 3 :: l, m, n

上記の宣言では、実変数e、f、およびgは、実変数a、b、およびcよりも精度が高くなっています。整数変数l、m、およびnは、整数変数i、j、およびkよりも大きな値を格納でき、格納する桁数も多くなります。これはマシンに依存しますが。

program kindSpecifier
implicit none

   real(kind = 4) :: a, b, c
   real(kind = 8) :: e, f, g
   integer(kind = 2) :: i, j, k
   integer(kind = 4) :: l, m, n
   integer :: kind_a, kind_i, kind_e, kind_l
   
   kind_a = kind(a)
   kind_i = kind(i)
   kind_e = kind(e)
   kind_l = kind(l)
   
   print *,'default kind for real is', kind_a
   print *,'default kind for int is', kind_i
   print *,'extended kind for real is', kind_e
   print *,'default kind for int is', kind_l
   
end program kindSpecifier

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

default kind for real is 4
default kind for int is 2
extended kind for real is 8
default kind for int is 4

変数のサイズの問い合わせ

数値のサイズを調べることができる組み込み関数がいくつかあります。

たとえば、 bit_size(i)組み込み関数は、ストレージに使用されるビット数を指定します。実数の場合、precision(x) 組み込み関数は、精度の10進数の桁数を返しますが、 range(x) 組み込み関数は、指数の10進範囲を返します。

program getSize
implicit none

   real (kind = 4) :: a
   real (kind = 8) :: b
   integer (kind = 2) :: i
   integer (kind = 4) :: j

   print *,'precision of real(4) =', precision(a)
   print *,'precision of real(8) =', precision(b)
   
   print *,'range of real(4) =', range(a)
   print *,'range of real(8) =', range(b)
   

   print *,'maximum exponent of real(4) =' , maxexponent(a)
   print *,'maximum exponent of real(8) =' , maxexponent(b)
  
   print *,'minimum exponent of real(4) =' , minexponent(a)
   print *,'minimum exponent of real(8) =' , minexponent(b)
   
   print *,'bits in integer(2) =' , bit_size(i)
   print *,'bits in integer(4) =' , bit_size(j)
   
end program getSize

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

precision of real(4) = 6
precision of real(8) = 15
range of real(4) = 37
range of real(8) = 307
maximum exponent of real(4) = 128
maximum exponent of real(8) = 1024
minimum exponent of real(4) = -125
minimum exponent of real(8) = -1021
bits in integer(2) = 16
bits in integer(4) = 32

種類の値を取得する

Fortranは、整数と実数の必要な精度に必要な種類の値を取得するために、さらに2つの組み込み関数を提供します。

  • selected_int_kind(r)
  • selected_real_kind([p、r])

selected_real_kind関数は、指定された10進精度pおよび10進指数範囲rに必要な種類タイプのパラメーター値である整数を返します。10進精度は有効桁数であり、10進指数範囲は表現可能な最小数と最大数を指定します。したがって、範囲は10-rから10 + rです。

たとえば、selected_real_kind(p = 10、r = 99)は、小数点以下10桁の精度、および少なくとも10-99から10 +99の範囲に必要な種類の値を返します。

program getKind
implicit none

   integer:: i
   i = selected_real_kind (p = 10, r = 99) 
   print *,'selected_real_kind (p = 10, r = 99)', i
   
end program getKind

上記のプログラムをコンパイルして実行すると、次の結果が生成されます-

selected_real_kind (p = 10, r = 99) 8

さまざまなFortranツールとライブラリがあります。無料のものもあれば、有料サービスのものもあります。

以下はいくつかの無料のライブラリです-

  • RANDLIB、乱数および統計分布ジェネレーター
  • BLAS
  • EISPACK
  • 利用可能な数学ソフトウェアのGAMS-NISTガイド
  • NISTからのいくつかの統計およびその他のルーチン
  • LAPACK
  • LINPACK
  • MINPACK
  • MUDPACK
  • NCAR数学ライブラリ
  • 数学ソフトウェア、論文、データベースのNetlibコレクション。
  • ODEPACK
  • ODERPACK、ランク付けと順序付けのためのルーチンのセット。
  • 行列指数を計算するためのExpokit
  • SLATEC
  • SPECFUN
  • STARPAC
  • StatLib統計ライブラリ
  • TOMS
  • 文字列の並べ替えとマージ

以下のライブラリは無料ではありません-

  • NAGFortran数値ライブラリ
  • Visual NumericsIMSLライブラリ
  • 数値レシピ

プログラミングスタイルとは、プログラムを開発する際にいくつかのルールに従うことです。これらのグッドプラクティスは、読みやすさや明確さなどの価値をプログラムに与えます。

優れたプログラムには、次の特性が必要です。

  • Readability
  • 適切な論理構造
  • 自明のメモとコメント

たとえば、次のようなコメントをすると、あまり役に立ちません。

! loop from 1 to 10 
do i = 1,10

ただし、二項係数を計算していて、nCrに対してこのループが必要な場合は、このようなコメントが役立ちます。

! loop to calculate nCr 
do i = 1,10
  • さまざまなレベルのコードを明確にするためのインデントされたコードブロック。

  • ゼロ除算、負の実数の平方根、負の実数の対数などの数値エラーがないことを確認するためのセルフチェックコード。

  • 変数が不正な値や範囲外の値をとらないようにするコードを含める、つまり入力検証。

  • 不要な場所にチェックを入れないと、実行が遅くなります。例-

real :: x 
x = sin(y) + 1.0

if (x >= 0.0) then
   z = sqrt(x)
end if
  • 適切なアルゴリズムを使用して明確に記述されたコード。
  • 継続マーカー「&」を使用して長い式を分割します。
  • 意味のある変数名を作成します。

デバッガツールは、プログラムのエラーを検索するために使用されます。

デバッガープログラムはコードをステップスルーし、プログラムの実行中に変数やその他のデータオブジェクトの値を調べることができます。

ソースコードが読み込まれ、デバッガー内でプログラムを実行することになっています。デバッガーは次の方法でプログラムをデバッグします-

  • ブレークポイントの設定、
  • ソースコードをステップスルーし、
  • ウォッチポイントの設定。

ブレークポイントは、特に重要なコード行の後で、プログラムを停止する場所を指定します。変数がブレークポイントでチェックされた後のプログラムの実行。

デバッガプログラムは、ソースコードも1行ずつチェックします。

ウォッチポイントは、特に読み取りまたは書き込み操作の後で、いくつかの変数の値をチェックする必要があるポイントです。

gdbデバッガー

gdbデバッガーであるGNUデバッガーには、Linuxオペレーティングシステムが付属しています。X Windowsシステムの場合、gdbにはグラフィカルインターフェイスが付属しており、プログラムの名前はxxgdbです。

次の表に、gdbのいくつかのコマンドを示します。

コマンド 目的
ブレーク ブレークポイントの設定
実行 実行を開始します
続き 実行を継続
関数呼び出しにステップインせずに、ソースコードの次の行のみを実行します
ステップ 関数呼び出しの場合に関数にステップインして、ソースコードの次の行を実行します。

dbxデバッガー

Linux用の別のデバッガーであるdbxデバッガーがあります。

次の表に、dbx −のいくつかのコマンドを示します。

コマンド 目的
stop [var] 変数varの値が変更されたときにブレークポイントを設定します。
[proc]で停止 プロシージャプロシージャが入力されると実行を停止します
[行]で停止 指定された行にブレークポイントを設定します。
実行 実行を開始します。
続き 実行を継続します。
関数呼び出しにステップインせずに、ソースコードの次の行のみを実行します。
ステップ 関数呼び出しの場合に関数にステップインして、ソースコードの次の行を実行します。