C ++クイックガイド
C ++は、静的に型付けされ、コンパイルされた、汎用の、大文字と小文字を区別する自由形式のプログラミング言語であり、手続き型、オブジェクト指向、およびジェネリックプログラミングをサポートします。
C ++は middle-level 言語は、高水準言語機能と低水準言語機能の両方の組み合わせで構成されているためです。
C ++は、1979年にニュージャージー州マレーヒルのベル研究所でBjarne Stroustrupによって開発され、C言語の拡張機能として、元々はC with Classesという名前でしたが、1983年にC ++に名前が変更されました。
C ++はCのスーパーセットであり、事実上すべての正当なCプログラムは正当なC ++プログラムです。
Note −プログラミング言語は、実行時ではなくコンパイル時に型チェックが実行されるときに静的型付けを使用すると言われています。
オブジェクト指向プログラミング
C ++は、オブジェクト指向開発の4つの柱を含む、オブジェクト指向プログラミングを完全にサポートしています。
- Encapsulation
- データの隠蔽
- Inheritance
- Polymorphism
標準ライブラリ
標準C ++は3つの重要な部分で構成されています-
変数、データ型、リテラルなどを含むすべての構成要素を提供するコア言語。
ファイルや文字列などを操作する豊富な関数セットを提供するC ++標準ライブラリ。
データ構造などを操作する豊富なメソッドセットを提供する標準テンプレートライブラリ(STL)。
ANSI規格
ANSI規格は、C ++の移植性を確保するための試みです。Microsoftのコンパイラ用に記述したコードは、Mac、UNIX、Windowsボックス、またはAlphaのコンパイラを使用して、エラーなしでコンパイルされます。
ANSI規格はしばらくの間安定しており、すべての主要なC ++コンパイラメーカーがANSI規格をサポートしています。
C ++の学習
C ++を学ぶ上で最も重要なことは、概念に焦点を当てることです。
プログラミング言語を学ぶ目的は、より優れたプログラマーになることです。つまり、新しいシステムの設計と実装、および古いシステムの保守をより効果的に行うことです。
C ++は、さまざまなプログラミングスタイルをサポートしています。Fortran、C、Smalltalkなどのスタイルで任意の言語で書くことができます。各スタイルは、実行時とスペースの効率を維持しながら、その目的を効果的に達成できます。
C ++の使用
C ++は、本質的にすべてのアプリケーションドメインで数十万人のプログラマーによって使用されています。
C ++は、リアルタイムの制約の下でハードウェアの直接操作に依存するデバイスドライバーやその他のソフトウェアを作成するために非常に使用されています。
C ++は、基本的な概念の教育を成功させるのに十分クリーンであるため、教育と研究に広く使用されています。
AppleMacintoshまたはWindowsを実行しているPCのいずれかを使用したことがある人は、これらのシステムの主要なユーザーインターフェイスがC ++で記述されているため、間接的にC ++を使用しています。
ローカル環境のセットアップ
それでもC ++の環境をセットアップする場合は、コンピューターに次の2つのソフトウェアが必要です。
テキストエディタ
これは、プログラムの入力に使用されます。いくつかのエディターの例には、Windowsメモ帳、OS Editコマンド、Brief、Epsilon、EMACS、vimまたはviが含まれます。
テキストエディタの名前とバージョンは、オペレーティングシステムによって異なる場合があります。たとえば、メモ帳はWindowsで使用され、vimまたはviはWindowsだけでなく、LinuxまたはUNIXでも使用できます。
エディターで作成するファイルはソースファイルと呼ばれ、C ++の場合、通常、拡張子.cpp、.cp、または.cで名前が付けられます。
C ++プログラミングを開始するには、テキストエディタを配置する必要があります。
C ++コンパイラ
これは実際のC ++コンパイラであり、ソースコードを最終的な実行可能プログラムにコンパイルするために使用されます。
ほとんどのC ++コンパイラは、ソースコードにどの拡張子を付けるかを気にしませんが、特に指定しない場合、多くのコンパイラはデフォルトで.cppを使用します。
最も頻繁に使用され、無料で利用できるコンパイラはGNU C / C ++コンパイラです。それ以外の場合は、それぞれのオペレーティングシステムがあれば、HPまたはSolarisのコンパイラを使用できます。
GNU C / C ++コンパイラのインストール
UNIX / Linuxのインストール
使用している場合 Linux or UNIX 次に、コマンドラインから次のコマンドを入力して、GCCがシステムにインストールされているかどうかを確認します-
$ g++ -v
GCCをインストールした場合は、次のようなメッセージが出力されます。
Using built-in specs.
Target: i386-redhat-linux
Configured with: ../configure --prefix=/usr .......
Thread model: posix
gcc version 4.1.2 20080704 (Red Hat 4.1.2-46)
GCCがインストールされていない場合は、で入手可能な詳細な手順を使用して自分でインストールする必要があります。 https://gcc.gnu.org/install/
Mac OSXのインストール
Mac OS Xを使用している場合、GCCを入手する最も簡単な方法は、AppleのWebサイトからXcode開発環境をダウンロードし、簡単なインストール手順に従うことです。
Xcodeは現在developer.apple.com/technologies/tools/で入手できます。
Windowsのインストール
WindowsにGCCをインストールするには、MinGWをインストールする必要があります。MinGWをインストールするには、MinGWホームページwww.mingw.orgにアクセスし、MinGWダウンロードページへのリンクをたどります。MinGW- <version> .exeという名前のMinGWインストールプログラムの最新バージョンをダウンロードします。
MinGWをインストールする際には、少なくともgcc-core、gcc-g ++、binutils、およびMinGWランタイムをインストールする必要がありますが、さらにインストールすることもできます。
MinGWインストールのbinサブディレクトリをに追加します PATH これらのツールをコマンドラインで単純な名前で指定できるようにするための環境変数。
インストールが完了すると、Windowsコマンドラインからgcc、g ++、ar、ranlib、dlltool、およびその他のいくつかのGNUツールを実行できるようになります。
C ++プログラムを考えるとき、それは相互のメソッドを呼び出すことによって通信するオブジェクトのコレクションとして定義できます。ここで、クラス、オブジェクト、メソッド、およびインスタンス変数の意味を簡単に見てみましょう。
Object−オブジェクトには状態と動作があります。例:犬には、色、名前、品種、行動などの状態があります。振る、吠える、食べる。オブジェクトはクラスのインスタンスです。
Class −クラスは、そのタイプのオブジェクトがサポートする動作/状態を説明するテンプレート/ブループリントとして定義できます。
Methods−メソッドは基本的に動作です。クラスには多くのメソッドを含めることができます。これは、ロジックが書き込まれ、データが操作され、すべてのアクションが実行されるメソッド内にあります。
Instance Variables−各オブジェクトには、固有のインスタンス変数のセットがあります。オブジェクトの状態は、これらのインスタンス変数に割り当てられた値によって作成されます。
C ++プログラム構造
私たちは言葉印刷し、簡単なコードを見てみましょうのHello Worldを。
#include <iostream>
using namespace std;
// main() is where program execution begins.
int main() {
cout << "Hello World"; // prints Hello World
return 0;
}
上記のプログラムのさまざまな部分を見てみましょう-
C ++言語は、プログラムに必要または有用な情報を含むいくつかのヘッダーを定義します。このプログラムの場合、ヘッダー<iostream> が必要です。
この線 using namespace std;std名前空間を使用するようにコンパイラーに指示します。名前空間は、C ++に比較的最近追加されたものです。
次の行 '// main() is where program execution begins.'は、C ++で使用可能な1行のコメントです。単一行コメントは//で始まり、行の終わりで終わります。
この線 int main() プログラムの実行を開始する主な機能です。
次の行 cout << "Hello World"; 「HelloWorld」というメッセージが画面に表示されます。
次の行 return 0; main()関数を終了し、呼び出し元のプロセスに値0を返します。
C ++プログラムのコンパイルと実行
ファイルを保存し、プログラムをコンパイルして実行する方法を見てみましょう。以下の手順に従ってください-
テキストエディタを開き、上記のようにコードを追加します。
ファイルを次のように保存します:hello.cpp
コマンドプロンプトを開き、ファイルを保存したディレクトリに移動します。
'g ++ hello.cpp'と入力し、Enterキーを押してコードをコンパイルします。コードにエラーがない場合、コマンドプロンプトで次の行に移動し、.out実行可能ファイルを生成します。
ここで、「a.out」と入力してプログラムを実行します。
ウィンドウに「HelloWorld」が印刷されているのがわかります。
$ g++ hello.cpp
$ ./a.out
Hello World
g ++がパスにあり、ファイルhello.cppを含むディレクトリで実行していることを確認してください。
makefileを使用してC / C ++プログラムをコンパイルできます。詳細については、「Makefileチュートリアル」を確認してください。
C ++のセミコロンとブロック
C ++では、セミコロンはステートメントターミネータです。つまり、個々のステートメントはセミコロンで終了する必要があります。1つの論理エンティティの終わりを示します。
たとえば、以下は3つの異なるステートメントです-
x = y;
y = y + 1;
add(x, y);
ブロックは、中括弧を開いたり閉じたりすることで囲まれた、論理的に接続されたステートメントのセットです。例-
{
cout << "Hello World"; // prints Hello World
return 0;
}
C ++は、行の終わりをターミネータとして認識しません。このため、ステートメントをどこに配置してもかまいません。例-
x = y;
y = y + 1;
add(x, y);
と同じです
x = y; y = y + 1; add(x, y);
C ++識別子
C ++識別子は、変数、関数、クラス、モジュール、またはその他のユーザー定義アイテムを識別するために使用される名前です。識別子は、文字AからZまたはaからzまたはアンダースコア(_)で始まり、その後に0個以上の文字、アンダースコア、および数字(0から9)が続きます。
C ++では、識別子内に@、$、%などの句読文字を使用できません。C ++は、大文字と小文字を区別するプログラミング言語です。したがって、Manpower そして manpower C ++の2つの異なる識別子です。
受け入れ可能な識別子の例を次に示します-
mohd zara abc move_name a_123
myname50 _temp j a23b9 retVal
C ++キーワード
次のリストは、C ++で予約されている単語を示しています。これらの予約語は、定数や変数、またはその他の識別子名として使用することはできません。
asm | そうしないと | 新着 | この |
自動 | 列挙型 | オペレーター | スロー |
ブール | 明示的 | 民間 | true |
ブレーク | 書き出す | 保護 | 試してみてください |
場合 | extern | 公衆 | typedef |
キャッチ | false | 登録 | typeid |
char | 浮く | reinterpret_cast | typename |
クラス | にとって | 戻る | 連合 |
const | 友達 | ショート | 署名なし |
const_cast | 後藤 | 署名 | を使用して |
継続する | もし | のサイズ | バーチャル |
デフォルト | 列をなして | 静的 | ボイド |
削除 | int | static_cast | 揮発性 |
行う | 長いです | 構造体 | wchar_t |
ダブル | 可変 | スイッチ | 一方 |
dynamic_cast | 名前空間 | テンプレート |
三重音字
いくつかの文字には、三重音字シーケンスと呼ばれる代替表現があります。トリグラフは、1文字を表す3文字のシーケンスであり、シーケンスは常に2つの疑問符で始まります。
トリグラフは、文字列リテラルや文字リテラル内、コメント内、プリプロセッサディレクティブ内など、表示される場所であればどこでも展開されます。
以下は、最も頻繁に使用されるトリグラフシーケンスです-
トリグラフ | 置換 |
---|---|
?? = | # |
?? / | \ |
?? ' | ^ |
??( | [ |
??) | ] |
??! | | |
?? < | {{ |
??> | } |
??- | 〜 |
すべてのコンパイラがトリグラフをサポートしているわけではなく、混乱を招く性質があるため、使用することはお勧めしません。
C ++の空白
空白のみを含む行(おそらくコメント付き)は空白行と呼ばれ、C ++コンパイラはそれを完全に無視します。
空白は、C ++で空白、タブ、改行文字、およびコメントを説明するために使用される用語です。空白は、ステートメントの一部を別の部分から分離し、コンパイラーが、intなどのステートメント内の1つの要素が終了し、次の要素が開始する場所を識別できるようにします。
ステートメント1
int age;
上記のステートメントでは、コンパイラがそれらを区別できるようにするには、intとageの間に少なくとも1つの空白文字(通常はスペース)が必要です。
ステートメント2
fruit = apples + oranges; // Get the total fruit
上記のステートメント2では、fruitと=の間、または=とapplesの間に空白文字は必要ありませんが、読みやすくするために空白文字を自由に含めることができます。
プログラムコメントは、C ++コードに含めることができる説明ステートメントです。これらのコメントは、誰でもソースコードを読むのに役立ちます。すべてのプログラミング言語では、何らかの形式のコメントが許可されています。
C ++は、単一行および複数行のコメントをサポートします。コメント内で使用可能なすべての文字は、C ++コンパイラによって無視されます。
C ++コメントは/ *で始まり、* /で終わります。例-
/* This is a comment */
/* C++ comments can also
* span multiple lines
*/
コメントは//で始まり、行末まで延長することもできます。例-
#include <iostream>
using namespace std;
main() {
cout << "Hello World"; // prints Hello World
return 0;
}
上記のコードをコンパイルすると、無視されます // prints Hello World 最終的な実行可能ファイルは次の結果を生成します-
Hello World
/ *および* /コメント内では、//文字に特別な意味はありません。//コメント内では、/ *と* /に特別な意味はありません。したがって、ある種類のコメントを別の種類の中に「ネスト」することができます。例-
/* Comment out printing of Hello World:
cout << "Hello World"; // prints Hello World
*/
任意の言語でプログラムを作成する場合、さまざまな情報を格納するためにさまざまな変数を使用する必要があります。変数は、値を格納するために予約されたメモリ位置に他なりません。これは、変数を作成するときに、メモリにいくらかのスペースを予約することを意味します。
文字、ワイド文字、整数、浮動小数点、倍精度浮動小数点、ブール値など、さまざまなデータ型の情報を格納することができます。変数のデータ型に基づいて、オペレーティングシステムはメモリを割り当て、格納できるものを決定します。予約済みメモリ。
プリミティブビルトインタイプ
C ++は、プログラマーに組み込みデータ型とユーザー定義データ型の豊富な品揃えを提供します。次の表に、7つの基本的なC ++データ型を示します。
タイプ | キーワード |
---|---|
ブール値 | ブール |
キャラクター | char |
整数 | int |
浮動小数点 | 浮く |
倍精度浮動小数点 | ダブル |
価値のない | ボイド |
ワイド文字 | wchar_t |
基本的な型のいくつかは、これらの型修飾子の1つ以上を使用して変更できます-
- signed
- unsigned
- short
- long
次の表は、変数のタイプ、メモリに値を格納するために必要なメモリの量、およびそのようなタイプの変数に格納できる最大値と最小値を示しています。
タイプ | 典型的なビット幅 | 典型的な範囲 |
---|---|---|
char | 1バイト | -127〜127または0〜255 |
unsigned char | 1バイト | 0から255 |
符号付き文字 | 1バイト | -127〜127 |
int | 4バイト | -2147483648から2147483647 |
unsigned int | 4バイト | 0から4294967295 |
符号付き整数 | 4バイト | -2147483648から2147483647 |
短い整数 | 2バイト | -32768〜32767 |
unsigned short int | 2バイト | 0〜65,535 |
署名されたshortint | 2バイト | -32768〜32767 |
long int | 8バイト | -2,147,483,648から2,147,483,647 |
署名されたlongint | 8バイト | longintと同じ |
unsigned long int | 8バイト | 0から4,294,967,295 |
long long int | 8バイト | -(2 ^ 63)から(2 ^ 63)-1 |
unsigned long long int | 8バイト | 0から18,446,744,073,709,551,615 |
浮く | 4バイト | |
ダブル | 8バイト | |
ロングダブル | 12バイト | |
wchar_t | 2または4バイト | 1つのワイド文字 |
変数のサイズは、使用しているコンパイラーとコンピューターによっては、上記の表に示されているものとは異なる場合があります。
以下は、コンピューター上でさまざまなデータ型の正しいサイズを生成する例です。
#include <iostream>
using namespace std;
int main() {
cout << "Size of char : " << sizeof(char) << endl;
cout << "Size of int : " << sizeof(int) << endl;
cout << "Size of short int : " << sizeof(short int) << endl;
cout << "Size of long int : " << sizeof(long int) << endl;
cout << "Size of float : " << sizeof(float) << endl;
cout << "Size of double : " << sizeof(double) << endl;
cout << "Size of wchar_t : " << sizeof(wchar_t) << endl;
return 0;
}
この例では endl、すべての行の後に改行文字を挿入し、<<演算子を使用して複数の値を画面に渡します。私たちも使用していますsizeof() さまざまなデータ型のサイズを取得する演算子。
上記のコードをコンパイルして実行すると、マシンごとに異なる可能性のある次の結果が生成されます。
Size of char : 1
Size of int : 4
Size of short int : 2
Size of long int : 4
Size of float : 4
Size of double : 8
Size of wchar_t : 4
typedef宣言
を使用して、既存のタイプの新しい名前を作成できます typedef。以下は、typedef −を使用して新しい型を定義するための簡単な構文です。
typedef type newname;
たとえば、次のように、フィートはint −の別名であることをコンパイラに通知します。
typedef int feet;
さて、次の宣言は完全に合法であり、distance −と呼ばれる整数変数を作成します。
feet distance;
列挙型
列挙型は、オプションの型名と、型の値として使用できる0個以上の識別子のセットを宣言します。各列挙子は、型が列挙型である定数です。
列挙型を作成するには、キーワードを使用する必要があります enum。列挙型の一般的な形式は次のとおりです。
enum enum-name { list of names } var-list;
ここで、enum-nameは列挙型のタイプ名です。名前のリストはコンマで区切られています。
たとえば、次のコードは、colorsと呼ばれる色の列挙とcolor型の変数cを定義します。最後に、cには値「blue」が割り当てられます。
enum color { red, green, blue } c;
c = blue;
デフォルトでは、名の値は0、2番目の名前の値は1、3番目の名前の値は2というように続きます。ただし、初期化子を追加することで、名前、特定の値を指定できます。たとえば、次の列挙では、green 値は5になります。
enum color { red, green = 5, blue };
ここに、 blue 各名前はその前の名前より1大きいため、値は6になります。
変数は、プログラムが操作できる名前付きストレージを提供します。C ++の各変数には特定のタイプがあり、変数のメモリのサイズとレイアウトを決定します。そのメモリ内に保存できる値の範囲。変数に適用できる一連の操作。
変数の名前は、文字、数字、および下線文字で構成できます。文字またはアンダースコアで始まる必要があります。C ++では大文字と小文字が区別されるため、大文字と小文字は区別されます-
前の章で説明したように、C ++には次の基本的なタイプの変数があります-
シニア番号 | タイプと説明 |
---|---|
1 | bool 値trueまたはfalseのいずれかを格納します。 |
2 | char 通常、1オクテット(1バイト)。これは整数型です。 |
3 | int マシンの整数の最も自然なサイズ。 |
4 | float 単精度浮動小数点値。 |
5 | double 倍精度浮動小数点値。 |
6 | void タイプがないことを表します。 |
7 | wchar_t ワイド文字タイプ。 |
C ++では、他のさまざまなタイプの変数を定義することもできます。これについては、次のような後続の章で説明します。 Enumeration, Pointer, Array, Reference, Data structures, そして Classes。
次のセクションでは、さまざまなタイプの変数を定義、宣言、および使用する方法について説明します。
C ++での変数定義
変数定義は、変数用に作成するストレージの場所と量をコンパイラーに指示します。変数定義はデータ型を指定し、次のようにその型の1つ以上の変数のリストを含みます-
type variable_list;
ここに、 type char、w_char、int、float、double、bool、または任意のユーザー定義オブジェクトなどを含む有効なC ++データ型である必要があります。 variable_listコンマで区切られた1つ以上の識別子名で構成されている場合があります。いくつかの有効な宣言をここに示します-
int i, j, k;
char c, ch;
float f, salary;
double d;
この線 int i, j, k;変数i、j、kの両方を宣言および定義します。これは、int型のi、j、およびkという名前の変数を作成するようにコンパイラーに指示します。
変数は、宣言で初期化(初期値を割り当てる)できます。初期化子は、等号とそれに続く次のような定数式で構成されます。
type variable_name = value;
いくつかの例は-
extern int d = 3, f = 5; // declaration of d and f.
int d = 3, f = 5; // definition and initializing d and f.
byte z = 22; // definition and initializes z.
char x = 'x'; // the variable x has the value 'x'.
初期化子なしの定義の場合:静的ストレージ期間の変数は暗黙的にNULLで初期化されます(すべてのバイトの値は0です)。他のすべての変数の初期値は未定義です。
C ++での変数宣言
変数宣言は、指定されたタイプと名前の変数が1つ存在することをコンパイラーに保証するため、コンパイラーは変数に関する完全な詳細を必要とせずにさらにコンパイルを進めることができます。変数宣言はコンパイル時にのみ意味があり、コンパイラはプログラムのリンク時に実際の変数定義を必要とします。
変数宣言は、複数のファイルを使用していて、プログラムのリンク時に使用できるファイルの1つで変数を定義する場合に役立ちます。使用しますextern任意の場所で変数を宣言するキーワード。C ++プログラムで変数を複数回宣言できますが、ファイル、関数、またはコードブロックで定義できるのは1回だけです。
例
変数が上部で宣言されているが、メイン関数内で定義されている次の例を試してください-
#include <iostream>
using namespace std;
// Variable declaration:
extern int a, b;
extern int c;
extern float f;
int main () {
// Variable definition:
int a, b;
int c;
float f;
// actual initialization
a = 10;
b = 20;
c = a + b;
cout << c << endl ;
f = 70.0/3.0;
cout << f << endl ;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
30
23.3333
同じ概念が関数宣言にも当てはまり、宣言時に関数名を指定すると、実際の定義は他の場所で指定できます。例-
// function declaration
int func();
int main() {
// function call
int i = func();
}
// function definition
int func() {
return 0;
}
左辺値と右辺値
C ++には2種類の式があります-
lvalue−メモリ位置を参照する式は「左辺値」式と呼ばれます。左辺値は、割り当ての左側または右側のいずれかとして表示される場合があります。
rvalue−右辺値という用語は、メモリ内のあるアドレスに格納されているデータ値を指します。右辺値は、値を割り当てることができない式です。つまり、右辺値は割り当ての右側には表示されますが、左側には表示されません。
変数は左辺値であるため、割り当ての左側に表示される場合があります。数値リテラルは右辺値であるため、割り当てられない場合があり、左側に表示することはできません。以下は有効なステートメントです-
int g = 20;
しかし、以下は有効なステートメントではなく、コンパイル時エラーを生成します-
10 = 20;
スコープはプログラムの領域であり、大まかに言えば、変数を宣言できる場所は3つあります。
ローカル変数と呼ばれる関数またはブロックの内部では、
仮パラメータと呼ばれる関数パラメータの定義。
グローバル変数と呼ばれるすべての関数の外。
関数とは何か、そしてそのパラメーターは次の章で学びます。ここでは、ローカル変数とグローバル変数について説明します。
ローカル変数
関数またはブロック内で宣言される変数はローカル変数です。それらは、その関数またはコードのブロック内にあるステートメントによってのみ使用できます。ローカル変数は、それ自体の外部の関数には知られていません。以下はローカル変数を使用した例です-
#include <iostream>
using namespace std;
int main () {
// Local variable declaration:
int a, b;
int c;
// actual initialization
a = 10;
b = 20;
c = a + b;
cout << c;
return 0;
}
グローバル変数
グローバル変数は、すべての関数の外部、通常はプログラムの上で定義されます。グローバル変数は、プログラムの存続期間を通じてその値を保持します。
グローバル変数には、任意の関数からアクセスできます。つまり、グローバル変数は、宣言後、プログラム全体で使用できます。以下は、グローバル変数とローカル変数を使用した例です-
#include <iostream>
using namespace std;
// Global variable declaration:
int g;
int main () {
// Local variable declaration:
int a, b;
// actual initialization
a = 10;
b = 20;
g = a + b;
cout << g;
return 0;
}
プログラムはローカル変数とグローバル変数に同じ名前を付けることができますが、関数内のローカル変数の値が優先されます。例-
#include <iostream>
using namespace std;
// Global variable declaration:
int g = 20;
int main () {
// Local variable declaration:
int g = 10;
cout << g;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
10
ローカル変数とグローバル変数の初期化
ローカル変数が定義されている場合、システムによって初期化されないため、自分で初期化する必要があります。グローバル変数は、次のように定義すると、システムによって自動的に初期化されます。
データ・タイプ | イニシャライザー |
---|---|
int | 0 |
char | '\ 0' |
浮く | 0 |
ダブル | 0 |
ポインター | ヌル |
変数を適切に初期化することは良いプログラミング手法です。そうしないと、プログラムが予期しない結果を生成することがあります。
定数は、プログラムが変更できない固定値を参照し、呼び出されます literals。
定数は、基本的なデータ型のいずれかであり、整数、浮動小数点数、文字、文字列、およびブール値に分割できます。
繰り返しますが、定数は、定義後に値を変更できないことを除いて、通常の変数と同じように扱われます。
整数リテラル
整数リテラルは、10進数、8進数、または16進数の定数にすることができます。接頭辞は、基数または基数を指定します。16進数の場合は0xまたは0X、8進数の場合は0、10進数の場合は何も指定しません。
整数リテラルには、unsignedとlongのそれぞれについて、UとLの組み合わせである接尾辞を付けることもできます。接尾辞は大文字または小文字にすることができ、任意の順序にすることができます。
整数リテラルの例を次に示します-
212 // Legal
215u // Legal
0xFeeL // Legal
078 // Illegal: 8 is not an octal digit
032UU // Illegal: cannot repeat a suffix
以下は、さまざまなタイプの整数リテラルの他の例です。
85 // decimal
0213 // octal
0x4b // hexadecimal
30 // int
30u // unsigned int
30l // long
30ul // unsigned long
浮動小数点リテラル
浮動小数点リテラルには、整数部分、小数点、小数部分、および指数部分があります。浮動小数点リテラルは、10進形式または指数形式のいずれかで表すことができます。
10進形式を使用して表す場合は、小数点、指数、またはその両方を含める必要があります。指数形式を使用して表す場合は、整数部分、小数部分、またはその両方を含める必要があります。符号付き指数は、eまたはEによって導入されます。
浮動小数点リテラルの例を次に示します-
3.14159 // Legal
314159E-5L // Legal
510E // Illegal: incomplete exponent
210f // Illegal: no decimal or exponent
.e55 // Illegal: missing integer or fraction
ブールリテラル
2つのブールリテラルがあり、それらは標準のC ++キーワードの一部です-
の値 true 真を表す。
の値 false falseを表します。
trueの値が1に等しく、falseの値が0に等しいと見なすべきではありません。
文字リテラル
文字リテラルは一重引用符で囲まれています。リテラルがLで始まる場合(大文字のみ)、それはワイド文字リテラル(L'x 'など)であり、に格納する必要があります。wchar_t変数のタイプ。それ以外の場合は、狭い文字リテラル(たとえば、「x」)であり、次の単純な変数に格納できます。char タイプ。
文字リテラルは、プレーン文字(たとえば、「x」)、エスケープシーケンス(たとえば、「\ t」)、またはユニバーサル文字(たとえば、「\ u02C0」)にすることができます。
C ++には、バックスラッシュが前に付いている特定の文字があり、それらは特別な意味を持ち、改行(\ n)やタブ(\ t)のように表すために使用されます。ここに、そのようなエスケープシーケンスコードのいくつかのリストがあります-
エスケープシーケンス | 意味 |
---|---|
\\ | \ キャラクター |
\ ' | ' キャラクター |
\ " | " キャラクター |
\? | ?キャラクター |
\ a | アラートまたはベル |
\ b | バックスペース |
\ f | フォームフィード |
\ n | 改行 |
\ r | キャリッジリターン |
\ t | 水平タブ |
\ v | 垂直タブ |
\ ooo | 1〜3桁の8進数 |
\ xhh。。。 | 1桁以上の16進数 |
以下は、いくつかのエスケープシーケンス文字を示す例です-
#include <iostream>
using namespace std;
int main() {
cout << "Hello\tWorld\n\n";
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Hello World
文字列リテラル
文字列リテラルは二重引用符で囲みます。文字列には、文字リテラルに類似した文字(プレーン文字、エスケープシーケンス、ユニバーサル文字)が含まれます。
文字列リテラルを使用して長い行を複数の行に分割し、空白を使用してそれらを区切ることができます。
文字列リテラルの例を次に示します。3つの形式はすべて同一の文字列です。
"hello, dear"
"hello, \
dear"
"hello, " "d" "ear"
定数の定義
C ++には、定数を定義する2つの簡単な方法があります-
使用する #define プリプロセッサ。
使用する const キーワード。
#defineプリプロセッサ
以下は、定数を定義するために#defineプリプロセッサを使用する形式です-
#define identifier value
次の例で詳細に説明します-
#include <iostream>
using namespace std;
#define LENGTH 10
#define WIDTH 5
#define NEWLINE '\n'
int main() {
int area;
area = LENGTH * WIDTH;
cout << area;
cout << NEWLINE;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
50
constキーワード
使用できます const 次のように特定のタイプの定数を宣言する接頭辞-
const type variable = value;
次の例で詳細に説明します-
#include <iostream>
using namespace std;
int main() {
const int LENGTH = 10;
const int WIDTH = 5;
const char NEWLINE = '\n';
int area;
area = LENGTH * WIDTH;
cout << area;
cout << NEWLINE;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
50
CAPITALSで定数を定義することは良いプログラミング手法であることに注意してください。
C ++では char, int, そして doubleデータ型の前に修飾子があります。修飾子は、さまざまな状況のニーズにより正確に適合するように、基本タイプの意味を変更するために使用されます。
データ型修飾子はここにリストされています-
- signed
- unsigned
- long
- short
修飾子 signed, unsigned, long, そして short整数ベースタイプに適用できます。加えて、signed そして unsigned charに適用でき、 long ダブルに適用することができます。
修飾子 signed そして unsigned の接頭辞としても使用できます long または short修飾子。例えば、unsigned long int。
C ++では、宣言の省略表記が可能です unsigned, short, または long整数。あなたは単に単語を使うことができますunsigned, short, または long, なし int。それは自動的に意味しますint。たとえば、次の2つのステートメントは、どちらも符号なし整数変数を宣言しています。
unsigned x;
unsigned int y;
符号付き整数修飾子と符号なし整数修飾子がC ++によって解釈される方法の違いを理解するには、次の短いプログラムを実行する必要があります。
#include <iostream>
using namespace std;
/* This program shows the difference between
* signed and unsigned integers.
*/
int main() {
short int i; // a signed short integer
short unsigned int j; // an unsigned short integer
j = 50000;
i = j;
cout << i << " " << j;
return 0;
}
このプログラムを実行すると、次のように出力されます。
-15536 50000
上記の結果は、50,000を短い符号なし整数として表すビットパターンが、shortによって-15,536として解釈されるためです。
C ++の型修飾子
型修飾子は、先行する変数に関する追加情報を提供します。
シニア番号 | 修飾子と意味 |
---|---|
1 | const タイプのオブジェクト const 実行中にプログラムで変更することはできません。 |
2 | volatile 修飾子 volatile プログラムで明示的に指定されていない方法で変数の値が変更される可能性があることをコンパイラーに通知します。 |
3 | restrict によって修飾されたポインタ restrict最初は、それが指すオブジェクトにアクセスできる唯一の手段です。C99のみがrestrictと呼ばれる新しい型修飾子を追加します。 |
ストレージクラスは、C ++プログラム内の変数や関数のスコープ(可視性)と存続期間を定義します。これらの指定子は、変更するタイプの前にあります。C ++プログラムで使用できる次のストレージクラスがあります
- auto
- register
- static
- extern
- mutable
自動ストレージクラス
ザ・ auto storage classは、すべてのローカル変数のデフォルトのストレージクラスです。
{
int mount;
auto int month;
}
上記の例では、同じストレージクラスを持つ2つの変数を定義しています。autoは関数内でのみ使用できます。つまり、ローカル変数です。
レジスタストレージクラス
ザ・ registerストレージクラスは、RAMではなくレジスタに格納する必要があるローカル変数を定義するために使用されます。これは、変数の最大サイズがレジスタサイズ(通常は1ワード)に等しく、単項 '&'演算子を適用できないことを意味します(メモリ位置がないため)。
{
register int miles;
}
レジスタは、カウンタなどの迅速なアクセスを必要とする変数にのみ使用する必要があります。'register'を定義しても、変数がレジスタに格納されることを意味するわけではないことにも注意してください。これは、ハードウェアと実装の制限によっては、レジスタに格納される可能性があることを意味します。
静的ストレージクラス
ザ・ staticstorageクラスは、プログラムがスコープに出入りするたびにローカル変数を作成および破棄するのではなく、プログラムの存続期間中、ローカル変数を存在させ続けるようにコンパイラーに指示します。したがって、ローカル変数を静的にすると、関数呼び出し間で値を維持できます。
静的修飾子は、グローバル変数にも適用できます。これが行われると、その変数のスコープは、それが宣言されているファイルに制限されます。
C ++では、静的がクラスデータメンバーで使用されると、そのメンバーの1つのコピーのみがそのクラスのすべてのオブジェクトによって共有されます。
#include <iostream>
// Function declaration
void func(void);
static int count = 10; /* Global variable */
main() {
while(count--) {
func();
}
return 0;
}
// Function definition
void func( void ) {
static int i = 5; // local static variable
i++;
std::cout << "i is " << i ;
std::cout << " and count is " << count << std::endl;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
i is 6 and count is 9
i is 7 and count is 8
i is 8 and count is 7
i is 9 and count is 6
i is 10 and count is 5
i is 11 and count is 4
i is 12 and count is 3
i is 13 and count is 2
i is 14 and count is 1
i is 15 and count is 0
externストレージクラス
ザ・ externストレージクラスは、すべてのプログラムファイルに表示されるグローバル変数の参照を提供するために使用されます。'extern'を使用すると、変数を初期化できません。これは、以前に定義された保存場所に変数名を指定するだけだからです。
複数のファイルがあり、他のファイルでも使用されるグローバル変数または関数を定義すると、externが別のファイルで使用され、定義された変数または関数の参照が提供されます。externを理解するために、別のファイルでグローバル変数または関数を宣言するために使用されます。
extern修飾子は、以下で説明するように、同じグローバル変数または関数を共有する2つ以上のファイルがある場合に最も一般的に使用されます。
最初のファイル:main.cpp
#include <iostream>
int count ;
extern void write_extern();
main() {
count = 5;
write_extern();
}
2番目のファイル:support.cpp
#include <iostream>
extern int count;
void write_extern(void) {
std::cout << "Count is " << count << std::endl;
}
ここでは、externキーワードを使用して、別のファイルでカウントを宣言しています。次に、これら2つのファイルを次のようにコンパイルします-
$g++ main.cpp support.cpp -o write
これにより、 write 実行可能プログラム、実行してみてください write 次のように結果を確認します-
$./write
5
可変ストレージクラス
ザ・ mutable指定子は、このチュートリアルの後半で説明するクラスオブジェクトにのみ適用されます。これにより、オブジェクトのメンバーがconstメンバー関数をオーバーライドできます。つまり、可変メンバーはconstメンバー関数で変更できます。
演算子は、特定の数学的または論理的操作を実行するようにコンパイラーに指示する記号です。C ++は組み込み演算子が豊富で、次のタイプの演算子を提供します-
- 算術演算子
- 関係演算子
- 論理演算子
- ビット演算子
- 代入演算子
- その他の演算子
この章では、算術、関係、論理、ビット単位、代入、およびその他の演算子を1つずつ調べます。
算術演算子
C ++言語でサポートされている次の算術演算子があります-
変数Aが10を保持し、変数Bが20を保持すると仮定すると、-
例を表示
オペレーター | 説明 | 例 |
---|---|---|
+ | 2つのオペランドを追加します | A + Bは30を与えます |
- | 最初のオペランドから2番目のオペランドを減算します | A-Bは-10を与えます |
* | Multiplies both operands | A * B will give 200 |
/ | Divides numerator by de-numerator | B / A will give 2 |
% | Modulus Operator and remainder of after an integer division | B % A will give 0 |
++ | Increment operator, increases integer value by one | A++ will give 11 |
-- | Decrement operator, decreases integer value by one | A-- will give 9 |
Relational Operators
There are following relational operators supported by C++ language
Assume variable A holds 10 and variable B holds 20, then −
Show Examples
Operator | Description | Example |
---|---|---|
== | Checks if the values of two operands are equal or not, if yes then condition becomes true. | (A == B) is not true. |
!= | Checks if the values of two operands are equal or not, if values are not equal then condition becomes true. | (A != B) is true. |
> | Checks if the value of left operand is greater than the value of right operand, if yes then condition becomes true. | (A > B) is not true. |
< | Checks if the value of left operand is less than the value of right operand, if yes then condition becomes true. | (A < B) is true. |
>= | Checks if the value of left operand is greater than or equal to the value of right operand, if yes then condition becomes true. | (A >= B) is not true. |
<= | Checks if the value of left operand is less than or equal to the value of right operand, if yes then condition becomes true. | (A <= B) is true. |
Logical Operators
There are following logical operators supported by C++ language.
Assume variable A holds 1 and variable B holds 0, then −
Show Examples
Operator | Description | Example |
---|---|---|
&& | Called Logical AND operator. If both the operands are non-zero, then condition becomes true. | (A && B) is false. |
|| | Called Logical OR Operator. If any of the two operands is non-zero, then condition becomes true. | (A || B) is true. |
! | Called Logical NOT Operator. Use to reverses the logical state of its operand. If a condition is true, then Logical NOT operator will make false. | !(A && B) is true. |
Bitwise Operators
Bitwise operator works on bits and perform bit-by-bit operation. The truth tables for &, |, and ^ are as follows −
p | q | p & q | p | q | p ^ q |
---|---|---|---|---|
0 | 0 | 0 | 0 | 0 |
0 | 1 | 0 | 1 | 1 |
1 | 1 | 1 | 1 | 0 |
1 | 0 | 0 | 1 | 1 |
Assume if A = 60; and B = 13; now in binary format they will be as follows −
A = 0011 1100
B = 0000 1101
-----------------
A&B = 0000 1100
A|B = 0011 1101
A^B = 0011 0001
~A = 1100 0011
The Bitwise operators supported by C++ language are listed in the following table. Assume variable A holds 60 and variable B holds 13, then −
Show Examples
Operator | Description | Example |
---|---|---|
& | Binary AND Operator copies a bit to the result if it exists in both operands. | (A & B) will give 12 which is 0000 1100 |
| | Binary OR Operator copies a bit if it exists in either operand. | (A | B) will give 61 which is 0011 1101 |
^ | Binary XOR Operator copies the bit if it is set in one operand but not both. | (A ^ B) will give 49 which is 0011 0001 |
~ | Binary Ones Complement Operator is unary and has the effect of 'flipping' bits. | (~A ) will give -61 which is 1100 0011 in 2's complement form due to a signed binary number. |
<< | Binary Left Shift Operator. The left operands value is moved left by the number of bits specified by the right operand. | A << 2 will give 240 which is 1111 0000 |
>> | Binary Right Shift Operator. The left operands value is moved right by the number of bits specified by the right operand. | A >> 2 will give 15 which is 0000 1111 |
代入演算子
C ++言語でサポートされている次の代入演算子があります-
例を表示
オペレーター | 説明 | 例 |
---|---|---|
= | 単純な代入演算子。右側のオペランドから左側のオペランドに値を代入します。 | C = A + Bは、A + Bの値をCに割り当てます |
+ = | AND代入演算子を追加します。これは、右のオペランドを左のオペランドに追加し、結果を左のオペランドに割り当てます。 | C + = AはC = C + Aと同等です |
-= | AND代入演算子を減算します。左オペランドから右オペランドを減算し、その結果を左オペランドに代入します。 | C- = AはC = C-Aと同等です |
* = | 乗算AND代入演算子。右オペランドと左オペランドを乗算し、その結果を左オペランドに代入します。 | C * = AはC = C * Aと同等です |
/ = | AND代入演算子を除算します。左オペランドを右オペランドで除算し、結果を左オペランドに代入します。 | C / = AはC = C / Aと同等です |
%= | モジュラスAND代入演算子。2つのオペランドを使用してモジュラスを取り、その結果を左側のオペランドに割り当てます。 | C%= AはC = C%Aと同等です |
<< = | 左シフトAND代入演算子。 | C << = 2はC = C << 2と同じです |
>> = | 右シフトAND代入演算子。 | C >> = 2はC = C >> 2と同じです |
&= | ビットごとのAND代入演算子。 | C&= 2はC = C&2と同じです |
^ = | ビット単位の排他的論理和と代入演算子。 | C ^ = 2はC = C ^ 2と同じです |
| = | ビット単位の包括的ORおよび代入演算子。 | C | = 2はC = C |と同じです 2 |
その他の演算子
次の表に、C ++がサポートするその他の演算子を示します。
シニア番号 | オペレーターと説明 |
---|---|
1 | sizeof sizeof演算子は、変数のサイズを返します。たとえば、sizeof(a)、ここで「a」は整数であり、4を返します。 |
2 | Condition ? X : Y 条件演算子(?)。Conditionがtrueの場合、Xの値を返し、それ以外の場合はYの値を返します。 |
3 | , コンマ演算子を使用すると、一連の操作が実行されます。コンマ式全体の値は、コンマ区切りリストの最後の式の値です。 |
4 | . (dot) and -> (arrow) メンバー演算子は、クラス、構造体、および共用体の個々のメンバーを参照するために使用されます。 |
5 | Cast キャスト演算子は、あるデータ型を別のデータ型に変換します。たとえば、int(2.2000)は2を返します。 |
6 | & ポインタ演算子&は変数のアドレスを返します。たとえば、&a; 変数の実際のアドレスを示します。 |
7 | * ポインター演算子*は変数へのポインターです。たとえば、* var; 変数varへのポインタになります。 |
C ++での演算子の優先順位
演算子の優先順位は、式内の用語のグループ化を決定します。これは、式の評価方法に影響します。特定の演算子は他の演算子よりも優先されます。たとえば、乗算演算子は加算演算子よりも優先されます-
たとえば、x = 7 + 3 * 2; ここでは、演算子*の優先順位が+よりも高いため、xには20ではなく13が割り当てられます。したがって、最初に3 * 2が乗算され、次に7に加算されます。
ここでは、優先順位が最も高い演算子がテーブルの上部に表示され、優先順位が最も低い演算子が下部に表示されます。式内では、優先順位の高い演算子が最初に評価されます。
例を表示
カテゴリー | オペレーター | 結合性 |
---|---|---|
Postfix | ()[]->。++ ---- | 左から右へ |
単項 | +-!〜++ ---(タイプ)*&sizeof | 右から左へ |
乗法 | * /% | 左から右へ |
添加剤 | +- | 左から右へ |
シフト | << >> | 左から右へ |
関連した | << = >> = | 左から右へ |
平等 | ==!= | 左から右へ |
ビットごとのAND | & | 左から右へ |
ビット単位のXOR | ^ | 左から右へ |
ビットごとのOR | | | 左から右へ |
論理積 | && | 左から右へ |
論理OR | || | 左から右へ |
条件付き | ?: | 右から左へ |
割り当て | = + =-= * = / =%= >> = << =&= ^ = | = | 右から左へ |
コンマ | 、 | 左から右へ |
コードのブロックを数回実行する必要がある場合があります。一般に、ステートメントは順番に実行されます。関数の最初のステートメントが最初に実行され、次に2番目のステートメントが実行されます。
プログラミング言語は、より複雑な実行パスを可能にするさまざまな制御構造を提供します。
ループステートメントを使用すると、ステートメントまたはステートメントのグループを複数回実行できます。以下は、ほとんどのプログラミング言語でのループステートメントの一般的なfromです。
C ++プログラミング言語は、ループ要件を処理するために次のタイプのループを提供します。
シニア番号 | ループの種類と説明 |
---|---|
1 | whileループ 指定された条件が真である間、ステートメントまたはステートメントのグループを繰り返します。ループ本体を実行する前に条件をテストします。 |
2 | forループ 一連のステートメントを複数回実行し、ループ変数を管理するコードを省略します。 |
3 | do ... whileループ 'while'ステートメントと同様ですが、ループ本体の最後で条件をテストする点が異なります。 |
4 | ネストされたループ 'while'、 'for'、または 'do..while'ループ内で1つ以上のループを使用できます。 |
ループ制御ステートメント
ループ制御ステートメントは、実行を通常のシーケンスから変更します。実行がスコープを離れると、そのスコープで作成されたすべての自動オブジェクトが破棄されます。
C ++は、次の制御ステートメントをサポートしています。
シニア番号 | 制御ステートメントと説明 |
---|---|
1 | breakステートメント を終了します loop または switch ステートメントを実行し、ループまたはスイッチの直後のステートメントに実行を転送します。 |
2 | ステートメントを続ける ループに本体の残りの部分をスキップさせ、繰り返す前にすぐにその状態を再テストします。 |
3 | gotoステートメント ラベル付きステートメントに制御を移します。プログラムでgotoステートメントを使用することはお勧めしませんが。 |
無限ループ
条件がfalseにならない場合、ループは無限ループになります。ザ・forループは伝統的にこの目的のために使用されます。'for'ループを形成する3つの式はいずれも必須ではないため、条件式を空のままにして無限ループを作成できます。
#include <iostream>
using namespace std;
int main () {
for( ; ; ) {
printf("This loop will run forever.\n");
}
return 0;
}
条件式がない場合は、trueと見なされます。初期化式とインクリメント式がある場合もありますが、C ++プログラマーは、より一般的には「for(;;)」構文を使用して無限ループを示します。
NOTE − Ctrl + Cキーを押すと、無限ループを終了できます。
意思決定構造では、プログラマーは、プログラムによって評価またはテストされる1つ以上の条件と、条件が真であると判断された場合に実行される1つまたは複数のステートメント、およびオプションで、条件が真である場合に実行される他のステートメントを指定する必要があります。 falseと判断されます。
以下は、ほとんどのプログラミング言語に見られる典型的な意思決定構造の一般的な形式です。
C ++プログラミング言語は、次のタイプの意思決定ステートメントを提供します。
シニア番号 | ステートメントと説明 |
---|---|
1 | ifステートメント 'if'ステートメントは、ブール式とそれに続く1つ以上のステートメントで構成されます。 |
2 | if ... elseステートメント 'if'ステートメントの後にオプションの 'else'ステートメントを続けることができます。これは、ブール式がfalseの場合に実行されます。 |
3 | switchステートメント 'switch'ステートメントを使用すると、値のリストに対して変数が等しいかどうかをテストできます。 |
4 | ネストされたifステートメント 1つの 'if'または 'elseif'ステートメントを別の 'if'または 'elseif'ステートメント内で使用できます。 |
5 | ネストされたswitchステートメント 1つの 'switch'ステートメントを別の 'switch'ステートメント内で使用できます。 |
?:オペレーター
私たちはカバーされている条件演算子を「?:」を置き換えるために使用できる前の章のif...elseステートメント。それは次の一般的な形式を持っています-
Exp1 ? Exp2 : Exp3;
Exp1、Exp2、およびExp3は式です。コロンの使用と配置に注意してください。
'?'の値 式は次のように決定されます。Exp1が評価されます。trueの場合、Exp2が評価され、「?」全体の値になります。式。Exp1がfalseの場合、Exp3が評価され、その値が式の値になります。
関数は、一緒にタスクを実行するステートメントのグループです。すべてのC ++プログラムには、少なくとも1つの関数があります。main()、およびすべての最も些細なプログラムで追加の関数を定義できます。
コードを個別の関数に分割できます。コードをさまざまな関数にどのように分割するかはあなた次第ですが、論理的には、通常、各関数が特定のタスクを実行するように分割されます。
機能 declaration関数の名前、戻り値の型、およびパラメーターについてコンパイラーに通知します。機能definition 関数の実際の本体を提供します。
C ++標準ライブラリは、プログラムが呼び出すことができる多数の組み込み関数を提供します。たとえば、関数strcat() 2つの文字列を連結するには、関数 memcpy() あるメモリ位置を別の位置にコピーし、さらに多くの機能を実行します。
関数は、メソッド、サブルーチン、プロシージャなど、さまざまな名前で知られています。
関数の定義
C ++関数定義の一般的な形式は次のとおりです。
return_type function_name( parameter list ) {
body of the function
}
C ++関数定義は、関数ヘッダーと関数本体で構成されます。これが関数のすべての部分です-
Return Type−関数は値を返す場合があります。ザ・return_type関数が返す値のデータ型です。一部の関数は、値を返さずに目的の操作を実行します。この場合、return_typeはキーワードですvoid。
Function Name−これは関数の実際の名前です。関数名とパラメータリストが一緒になって関数シグネチャを構成します。
Parameters−パラメータはプレースホルダーのようなものです。関数が呼び出されると、パラメーターに値を渡します。この値は、実際のパラメーターまたは引数と呼ばれます。パラメータリストは、関数のパラメータのタイプ、順序、および数を参照します。パラメータはオプションです。つまり、関数にパラメータを含めることはできません。
Function Body −関数本体には、関数の機能を定義するステートメントのコレクションが含まれています。
例
以下は、と呼ばれる関数のソースコードです。 max()。この関数は、2つのパラメーターnum1とnum2を取り、両方の最大値を返します。
// function returning the max between two numbers
int max(int num1, int num2) {
// local variable declaration
int result;
if (num1 > num2)
result = num1;
else
result = num2;
return result;
}
関数宣言
機能 declaration関数名と関数の呼び出し方法についてコンパイラーに指示します。関数の実際の本体は個別に定義できます。
関数宣言には次の部分があります-
return_type function_name( parameter list );
上記で定義された関数max()の場合、以下は関数宣言です。
int max(int num1, int num2);
パラメータ名は関数宣言では重要ではなく、型だけが必要なので、以下も有効な宣言です-
int max(int, int);
あるソースファイルで関数を定義し、その関数を別のファイルで呼び出す場合は、関数宣言が必要です。このような場合、関数を呼び出すファイルの先頭で関数を宣言する必要があります。
関数の呼び出し
C ++関数を作成するときに、関数が実行する必要があることを定義します。関数を使用するには、その関数を呼び出すか呼び出す必要があります。
プログラムが関数を呼び出すと、プログラムの制御は呼び出された関数に移されます。呼び出された関数は、定義されたタスクを実行し、returnステートメントが実行されるか、関数終了の終了中括弧に達すると、プログラム制御をメインプログラムに戻します。
関数を呼び出すには、必要なパラメーターを関数名と一緒に渡す必要があります。関数が値を返す場合は、戻り値を保存できます。例-
#include <iostream>
using namespace std;
// function declaration
int max(int num1, int num2);
int main () {
// local variable declaration:
int a = 100;
int b = 200;
int ret;
// calling a function to get max value.
ret = max(a, b);
cout << "Max value is : " << ret << endl;
return 0;
}
// function returning the max between two numbers
int max(int num1, int num2) {
// local variable declaration
int result;
if (num1 > num2)
result = num1;
else
result = num2;
return result;
}
max()関数をmain()関数と一緒に保持し、ソースコードをコンパイルしました。最終的な実行可能ファイルを実行すると、次の結果が生成されます-
Max value is : 200
関数の引数
関数が引数を使用する場合は、引数の値を受け入れる変数を宣言する必要があります。これらの変数は、formal parameters 関数の。
仮パラメータは、関数内の他のローカル変数と同様に動作し、関数への入力時に作成され、終了時に破棄されます。
関数の呼び出し中に、引数を関数に渡す方法は2つあります-
シニア番号 | 通話の種類と説明 |
---|---|
1 | 値による呼び出し このメソッドは、引数の実際の値を関数の仮パラメーターにコピーします。この場合、関数内のパラメーターに加えられた変更は引数に影響を与えません。 |
2 | ポインタによる呼び出し このメソッドは、引数のアドレスを仮パラメーターにコピーします。関数内では、アドレスは呼び出しで使用される実際の引数にアクセスするために使用されます。これは、パラメータに加えられた変更が引数に影響を与えることを意味します。 |
3 | 参照による呼び出し このメソッドは、引数の参照を仮パラメーターにコピーします。関数内では、参照は、呼び出しで使用される実際の引数にアクセスするために使用されます。これは、パラメータに加えられた変更が引数に影響を与えることを意味します。 |
デフォルトでは、C ++は call by value引数を渡す。一般に、これは、関数内のコードが、関数の呼び出しに使用される引数と、同じメソッドを使用するmax()関数の呼び出し中に上記の例を変更できないことを意味します。
パラメータのデフォルト値
関数を定義するときに、最後の各パラメーターにデフォルト値を指定できます。この値は、関数を呼び出すときに対応する引数が空白のままの場合に使用されます。
これは、代入演算子を使用し、関数定義の引数に値を割り当てることによって行われます。関数が呼び出されたときにそのパラメーターの値が渡されない場合、デフォルトの指定された値が使用されますが、値が指定されている場合、このデフォルト値は無視され、代わりに渡された値が使用されます。次の例を考えてみましょう-
#include <iostream>
using namespace std;
int sum(int a, int b = 20) {
int result;
result = a + b;
return (result);
}
int main () {
// local variable declaration:
int a = 100;
int b = 200;
int result;
// calling a function to add the values.
result = sum(a, b);
cout << "Total value is :" << result << endl;
// calling a function again as follows.
result = sum(a);
cout << "Total value is :" << result << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Total value is :300
Total value is :120
通常、Numbersを使用する場合は、int、short、long、float、doubleなどのプリミティブデータ型を使用します。C++データ型について説明する際に、数値データ型、それらの可能な値、および数値範囲について説明しました。
C ++での数値の定義
前の章で示したさまざまな例で、すでに数値を定義しています。これは、C ++でさまざまなタイプの数値を定義するための別の統合された例です。
#include <iostream>
using namespace std;
int main () {
// number definition:
short s;
int i;
long l;
float f;
double d;
// number assignments;
s = 10;
i = 1000;
l = 1000000;
f = 230.47;
d = 30949.374;
// number printing;
cout << "short s :" << s << endl;
cout << "int i :" << i << endl;
cout << "long l :" << l << endl;
cout << "float f :" << f << endl;
cout << "double d :" << d << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
short s :10
int i :1000
long l :1000000
float f :230.47
double d :30949.4
C ++での数学演算
作成できるさまざまな関数に加えて、C ++には使用できるいくつかの便利な関数も含まれています。これらの関数は、標準のCおよびC ++ライブラリで使用でき、built-in関数。これらは、プログラムに組み込んで使用できる関数です。
C ++には、さまざまな数値で実行できる豊富な数学演算のセットがあります。次の表に、C ++で使用できるいくつかの便利な組み込み数学関数を示します。
これらの関数を利用するには、数学ヘッダーファイルをインクルードする必要があります <cmath>。
シニア番号 | 機能と目的 |
---|---|
1 | double cos(double); この関数は角度を取り(doubleとして)、コサインを返します。 |
2 | double sin(double); この関数は角度を取り(doubleとして)、正弦を返します。 |
3 | double tan(double); この関数は角度を取り(doubleとして)、接線を返します。 |
4 | double log(double); この関数は数値を受け取り、その数値の自然対数を返します。 |
5 | double pow(double, double); 最初はあなたが上げたい数であり、2番目はあなたがそれを上げたい力ですt |
6 | double hypot(double, double); この関数を直角三角形の2辺の長さに渡すと、斜辺の長さが返されます。 |
7 | double sqrt(double); この関数に数値を渡すと、平方根が得られます。 |
8 | int abs(int); この関数は、渡された整数の絶対値を返します。 |
9 | double fabs(double); この関数は、渡された10進数の絶対値を返します。 |
10 | double floor(double); 渡された引数以下の整数を検索します。 |
以下は、数学演算のいくつかを示す簡単な例です-
#include <iostream>
#include <cmath>
using namespace std;
int main () {
// number definition:
short s = 10;
int i = -1000;
long l = 100000;
float f = 230.47;
double d = 200.374;
// mathematical operations;
cout << "sin(d) :" << sin(d) << endl;
cout << "abs(i) :" << abs(i) << endl;
cout << "floor(d) :" << floor(d) << endl;
cout << "sqrt(f) :" << sqrt(f) << endl;
cout << "pow( d, 2) :" << pow(d, 2) << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
sign(d) :-0.634939
abs(i) :1000
floor(d) :200
sqrt(f) :15.1812
pow( d, 2 ) :40149.7
C ++の乱数
乱数を生成したい場合が多くあります。実際には、乱数の生成について知っておく必要のある2つの関数があります。最初はrand()、この関数は疑似乱数のみを返します。これを修正する方法は、最初にを呼び出すことですsrand() 関数。
以下は、いくつかの乱数を生成する簡単な例です。この例では、time() システム時間の秒数を取得し、rand()関数をランダムにシードする関数-
#include <iostream>
#include <ctime>
#include <cstdlib>
using namespace std;
int main () {
int i,j;
// set the seed
srand( (unsigned)time( NULL ) );
/* generate 10 random numbers. */
for( i = 0; i < 10; i++ ) {
// generate actual random number
j = rand();
cout <<" Random Number : " << j << endl;
}
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Random Number : 1748144778
Random Number : 630873888
Random Number : 2134540646
Random Number : 219404170
Random Number : 902129458
Random Number : 920445370
Random Number : 1319072661
Random Number : 257938873
Random Number : 1256201101
Random Number : 580322989
C ++はデータ構造を提供します。 the array、同じタイプの要素の固定サイズの順次コレクションを格納します。配列はデータのコレクションを格納するために使用されますが、配列を同じタイプの変数のコレクションと考える方が便利な場合がよくあります。
number0、number1、...、number99などの個々の変数を宣言する代わりに、numbersなどの1つの配列変数を宣言し、numbers [0]、numbers [1]、...、numbers [99]を使用して表現します。個々の変数。配列内の特定の要素は、インデックスによってアクセスされます。
すべてのアレイは、連続したメモリ位置で構成されています。最小アドレスは最初の要素に対応し、最大アドレスは最後の要素に対応します。
配列の宣言
C ++で配列を宣言するために、プログラマーは次のように配列に必要な要素のタイプと要素の数を指定します。
type arrayName [ arraySize ];
これは、1次元配列と呼ばれます。ザ・arraySize ゼロより大きい整数定数である必要があり、 type任意の有効なC ++データ型にすることができます。たとえば、double型のbalanceという10要素の配列を宣言するには、次のステートメントを使用します。
double balance[10];
配列の初期化
C ++配列要素は、次のように1つずつ、または1つのステートメントを使用して初期化できます。
double balance[5] = {1000.0, 2.0, 3.4, 17.0, 50.0};
中括弧{}の間の値の数は、角括弧[]の間の配列に対して宣言する要素の数より大きくすることはできません。以下は、配列の単一の要素を割り当てる例です。
配列のサイズを省略すると、初期化を保持するのに十分な大きさの配列が作成されます。したがって、あなたが書く場合-
double balance[] = {1000.0, 2.0, 3.4, 17.0, 50.0};
前の例で行ったのとまったく同じ配列を作成します。
balance[4] = 50.0;
上記のステートメントは、配列の5番目の要素番号に50.0の値を割り当てます。4番目のインデックスを持つ配列は5番目、つまり最後の要素になります。これは、すべての配列の最初の要素のインデックスが0であるためです。これは、ベースインデックスとも呼ばれます。以下は、上記で説明したのと同じアレイの図解です。
配列要素へのアクセス
要素には、配列名にインデックスを付けることでアクセスします。これは、配列名の後に角括弧内に要素のインデックスを配置することによって行われます。例-
double salary = balance[9];
上記のステートメントは、配列から10番目の要素を取得し、その値を給与変数に割り当てます。以下は、上記の3つの概念すべてを使用する例です。宣言、割り当て、配列へのアクセス-
#include <iostream>
using namespace std;
#include <iomanip>
using std::setw;
int main () {
int n[ 10 ]; // n is an array of 10 integers
// initialize elements of array n to 0
for ( int i = 0; i < 10; i++ ) {
n[ i ] = i + 100; // set element at location i to i + 100
}
cout << "Element" << setw( 13 ) << "Value" << endl;
// output each array element's value
for ( int j = 0; j < 10; j++ ) {
cout << setw( 7 )<< j << setw( 13 ) << n[ j ] << endl;
}
return 0;
}
このプログラムはを利用します setw()出力をフォーマットする関数。上記のコードをコンパイルして実行すると、次の結果が得られます。
Element Value
0 100
1 101
2 102
3 103
4 104
5 105
6 106
7 107
8 108
9 109
C ++の配列
配列はC ++にとって重要であり、さらに多くの詳細が必要です。以下のいくつかの重要な概念があり、C ++プログラマーには明らかです。
シニア番号 | コンセプトと説明 |
---|---|
1 | 多次元配列 C ++は多次元配列をサポートしています。多次元配列の最も単純な形式は、2次元配列です。 |
2 | 配列へのポインタ インデックスなしで配列名を指定するだけで、配列の最初の要素へのポインタを生成できます。 |
3 | 関数に配列を渡す インデックスなしで配列の名前を指定することにより、配列へのポインタを関数に渡すことができます。 |
4 | 関数から配列を返す C ++では、関数が配列を返すことができます。 |
C ++は、次の2種類の文字列表現を提供します-
- Cスタイルの文字列。
- 標準C ++で導入された文字列クラスタイプ。
Cスタイルの文字列
Cスタイルの文字列はC言語で作成され、C ++内で引き続きサポートされます。この文字列は、実際には1次元の文字配列であり、null文字 '\ 0'。したがって、nullで終了する文字列には、文字列とそれに続く文字列を構成する文字が含まれます。null。
次の宣言と初期化により、「Hello」という単語で構成される文字列が作成されます。配列の最後にヌル文字を保持するために、文字列を含む文字配列のサイズは、単語「Hello」の文字数より1つ大きくなります。
char greeting[6] = {'H', 'e', 'l', 'l', 'o', '\0'};
配列の初期化のルールに従う場合は、上記のステートメントを次のように記述できます。
char greeting[] = "Hello";
以下は、C / C ++で上記で定義された文字列のメモリ表示です。
実際には、文字列定数の最後にヌル文字を配置しません。C ++コンパイラは、配列を初期化するときに、文字列の最後に「\ 0」を自動的に配置します。上記の文字列を印刷してみましょう-
#include <iostream>
using namespace std;
int main () {
char greeting[6] = {'H', 'e', 'l', 'l', 'o', '\0'};
cout << "Greeting message: ";
cout << greeting << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Greeting message: Hello
C ++は、nullで終了する文字列を操作するさまざまな関数をサポートしています-
シニア番号 | 機能と目的 |
---|---|
1 | strcpy(s1, s2); 文字列s2を文字列s1にコピーします。 |
2 | strcat(s1, s2); 文字列s2を文字列s1の末尾に連結します。 |
3 | strlen(s1); 文字列s1の長さを返します。 |
4 | strcmp(s1, s2); s1とs2が同じ場合は0を返します。s1 <s2の場合は0未満。s1> s2の場合は0より大きい。 |
5 | strchr(s1, ch); 文字列s1で最初に出現する文字chへのポインタを返します。 |
6 | strstr(s1, s2); 文字列s1で最初に出現する文字列s2へのポインタを返します。 |
次の例では、上記の機能のいくつかを使用しています-
#include <iostream>
#include <cstring>
using namespace std;
int main () {
char str1[10] = "Hello";
char str2[10] = "World";
char str3[10];
int len ;
// copy str1 into str3
strcpy( str3, str1);
cout << "strcpy( str3, str1) : " << str3 << endl;
// concatenates str1 and str2
strcat( str1, str2);
cout << "strcat( str1, str2): " << str1 << endl;
// total lenghth of str1 after concatenation
len = strlen(str1);
cout << "strlen(str1) : " << len << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次のような結果が生成されます。
strcpy( str3, str1) : Hello
strcat( str1, str2): HelloWorld
strlen(str1) : 10
C ++の文字列クラス
標準C ++ライブラリは string上記のすべての操作に加えて、はるかに多くの機能をサポートするクラスタイプ。次の例を確認してみましょう-
#include <iostream>
#include <string>
using namespace std;
int main () {
string str1 = "Hello";
string str2 = "World";
string str3;
int len ;
// copy str1 into str3
str3 = str1;
cout << "str3 : " << str3 << endl;
// concatenates str1 and str2
str3 = str1 + str2;
cout << "str1 + str2 : " << str3 << endl;
// total length of str3 after concatenation
len = str3.size();
cout << "str3.size() : " << len << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次のような結果が生成されます。
str3 : Hello
str1 + str2 : HelloWorld
str3.size() : 10
C ++ポインターは、簡単に学ぶことができます。一部のC ++タスクはポインターを使用するとより簡単に実行でき、動的メモリ割り当てなどの他のC ++タスクはポインターなしでは実行できません。
ご存知のように、すべての変数はメモリ位置であり、すべてのメモリ位置にはアドレスが定義されており、メモリ内のアドレスを示すアンパサンド(&)演算子を使用してアクセスできます。定義された変数のアドレスを出力する次のことを考慮してください-
#include <iostream>
using namespace std;
int main () {
int var1;
char var2[10];
cout << "Address of var1 variable: ";
cout << &var1 << endl;
cout << "Address of var2 variable: ";
cout << &var2 << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Address of var1 variable: 0xbfebd5c0
Address of var2 variable: 0xbfebd5b6
ポインタとは何ですか?
A pointer値が別の変数のアドレスである変数です。他の変数や定数と同様に、ポインタを操作する前にポインタを宣言する必要があります。ポインタ変数宣言の一般的な形式は次のとおりです。
type *var-name;
ここに、 typeポインタの基本型です。有効なC ++タイプである必要があり、var-nameポインタ変数の名前です。ポインターを宣言するために使用したアスタリスクは、乗算に使用したものと同じアスタリスクです。ただし、このステートメントでは、変数をポインターとして指定するためにアスタリスクが使用されています。有効なポインタ宣言は次のとおりです-
int *ip; // pointer to an integer
double *dp; // pointer to a double
float *fp; // pointer to a float
char *ch // pointer to character
整数、浮動小数点数、文字など、すべてのポインタの値の実際のデータ型は同じであり、メモリアドレスを表す長い16進数です。異なるデータ型のポインター間の唯一の違いは、ポインターが指す変数または定数のデータ型です。
C ++でのポインターの使用
重要な操作はほとんどありませんが、ポインターを使用して頻繁に実行します。 (a) ポインタ変数を定義します。 (b) 変数のアドレスをポインタに割り当てます。 (c)最後に、ポインタ変数で使用可能なアドレスの値にアクセスします。これは、オペランドで指定されたアドレスにある変数の値を返す単項演算子*を使用して行われます。次の例では、これらの操作を利用しています-
#include <iostream>
using namespace std;
int main () {
int var = 20; // actual variable declaration.
int *ip; // pointer variable
ip = &var; // store address of var in pointer variable
cout << "Value of var variable: ";
cout << var << endl;
// print the address stored in ip pointer variable
cout << "Address stored in ip variable: ";
cout << ip << endl;
// access the value at the address available in pointer
cout << "Value of *ip variable: ";
cout << *ip << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次のような結果が生成されます。
Value of var variable: 20
Address stored in ip variable: 0xbfc601ac
Value of *ip variable: 20
C ++のポインタ
ポインタには多くの簡単な概念があり、C ++プログラミングにとって非常に重要です。C ++プログラマーにとって明確であるはずの以下のいくつかの重要なポインターの概念があります-
シニア番号 | コンセプトと説明 |
---|---|
1 | ヌルポインタ C ++は、いくつかの標準ライブラリで定義されている値がゼロの定数であるnullポインタをサポートしています。 |
2 | ポインタ演算 ポインタで使用できる算術演算子は、++、-、+、-の4つです。 |
3 | ポインタと配列 ポインタと配列の間には密接な関係があります。 |
4 | ポインタの配列 多数のポインターを保持する配列を定義できます。 |
5 | ポインタからポインタへ C ++では、ポインタの上にポインタを置くことができます。 |
6 | 関数へのポインタの受け渡し 参照またはアドレスの両方で引数を渡すと、渡された引数を、呼び出された関数によって呼び出し元の関数で変更できます。 |
7 | 関数からのポインタを返す C ++では、関数がローカル変数、静的変数、および動的に割り当てられたメモリへのポインタを返すこともできます。 |
参照変数はエイリアス、つまり既存の変数の別名です。参照が変数で初期化されると、変数名または参照名のいずれかを使用して変数を参照できます。
参照とポインタ
参照はポインタと混同されることがよくありますが、参照とポインタの3つの大きな違いは次のとおりです。
NULL参照を持つことはできません。参照が正当なストレージに接続されていると常に想定できる必要があります。
参照がオブジェクトに対して初期化されると、別のオブジェクトを参照するように変更することはできません。ポインタはいつでも別のオブジェクトを指すことができます。
参照は、作成時に初期化する必要があります。ポインタはいつでも初期化できます。
C ++での参照の作成
変数名は、メモリ内の変数の場所に付けられたラベルと考えてください。次に、参照をそのメモリ位置に付けられた2番目のラベルと考えることができます。したがって、元の変数名または参照のいずれかを介して変数の内容にアクセスできます。たとえば、次の例があるとします。
int i = 17;
iの参照変数は次のように宣言できます。
int& r = i;
これらの宣言の&を次のように読んでください reference。したがって、最初の宣言を「rはiに初期化された整数参照」として読み取り、2番目の宣言を「sはdに初期化された二重参照」として読み取ります。次の例では、intとdoubleの参照を使用しています。
#include <iostream>
using namespace std;
int main () {
// declare simple variables
int i;
double d;
// declare reference variables
int& r = i;
double& s = d;
i = 5;
cout << "Value of i : " << i << endl;
cout << "Value of i reference : " << r << endl;
d = 11.7;
cout << "Value of d : " << d << endl;
cout << "Value of d reference : " << s << endl;
return 0;
}
上記のコードをまとめてコンパイルして実行すると、次のような結果になります。
Value of i : 5
Value of i reference : 5
Value of d : 11.7
Value of d reference : 11.7
参照は通常、関数の引数リストと関数の戻り値に使用されます。したがって、以下は、C ++プログラマーにとって明らかなはずのC ++参照に関連する2つの重要な主題です。
シニア番号 | コンセプトと説明 |
---|---|
1 | パラメータとしての参照 C ++は、パラメーターよりも安全に関数パラメーターとして参照を渡すことをサポートしています。 |
2 | 戻り値としての参照 他のデータ型と同様に、C ++関数から参照を返すことができます。 |
C ++標準ライブラリは、適切な日付型を提供していません。C ++は、日付と時刻を操作するための構造体と関数をCから継承します。日付と時刻に関連する関数と構造体にアクセスするには、C ++プログラムに<ctime>ヘッダーファイルをインクルードする必要があります。
時間に関連する4つのタイプがあります。 clock_t, time_t, size_t、および tm。タイプ(clock_t、size_t、time_t)は、システムの時刻と日付をある種の整数として表すことができます。
構造タイプ tm 次の要素を持つC構造の形式で日付と時刻を保持します-
struct tm {
int tm_sec; // seconds of minutes from 0 to 61
int tm_min; // minutes of hour from 0 to 59
int tm_hour; // hours of day from 0 to 24
int tm_mday; // day of month from 1 to 31
int tm_mon; // month of year from 0 to 11
int tm_year; // year since 1900
int tm_wday; // days since sunday
int tm_yday; // days since January 1st
int tm_isdst; // hours of daylight savings time
}
以下は、CまたはC ++で日付と時刻を操作するときに使用する重要な関数です。これらの関数はすべて標準CおよびC ++ライブラリの一部であり、以下に示すC ++標準ライブラリを参照して詳細を確認できます。
シニア番号 | 機能と目的 |
---|---|
1 | time_t time(time_t *time); これは、1970年1月1日から経過した秒数でシステムの現在のカレンダー時間を返します。システムに時間がない場合は、.1が返されます。 |
2 | char *ctime(const time_t *time); これは、日、月、年、時:分:秒、年の形式の文字列へのポインタを返します\ n \ 0。 |
3 | struct tm *localtime(const time_t *time); これはへのポインタを返します tm 現地時間を表す構造。 |
4 | clock_t clock(void); これは、呼び出し側プログラムが実行されていた時間の概算値を返します。時間が利用できない場合は、値.1が返されます。 |
5 | char * asctime ( const struct tm * time ); これにより、時刻が指す構造体に格納されている情報を含む文字列へのポインタが次の形式に変換されて返されます。日月日時間:分:秒年\ n \ 0 |
6 | struct tm *gmtime(const time_t *time); これは、tm構造体の形式で時間へのポインターを返します。時間は協定世界時(UTC)で表されます。これは、基本的にグリニッジ標準時(GMT)です。 |
7 | time_t mktime(struct tm *time); これは、timeが指す構造体で見つかった時間に相当するカレンダー時間を返します。 |
8 | double difftime ( time_t time2, time_t time1 ); この関数は、time1とtime2の差を秒単位で計算します。 |
9 | size_t strftime(); この関数を使用して、日付と時刻を特定の形式でフォーマットできます。 |
現在の日時
現在のシステムの日付と時刻を現地時間または協定世界時(UTC)として取得するとします。以下は同じことを達成するための例です-
#include <iostream>
#include <ctime>
using namespace std;
int main() {
// current date/time based on current system
time_t now = time(0);
// convert now to string form
char* dt = ctime(&now);
cout << "The local date and time is: " << dt << endl;
// convert now to tm struct for UTC
tm *gmtm = gmtime(&now);
dt = asctime(gmtm);
cout << "The UTC date and time is:"<< dt << endl;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
The local date and time is: Sat Jan 8 20:07:41 2011
The UTC date and time is:Sun Jan 9 03:07:41 2011
structtmを使用して時間をフォーマットする
ザ・ tmCまたはC ++で日付と時刻を操作する場合、構造は非常に重要です。この構造体は、前述のようにC構造体の形式で日付と時刻を保持します。ほとんどの場合、関連する関数はtm構造を利用します。以下は、さまざまな日付と時刻に関連する関数とtm構造を利用する例です。
この章で構造体を使用している間、C構造体と、矢印->演算子を使用して構造体メンバーにアクセスする方法について基本的な知識があることを前提としています。
#include <iostream>
#include <ctime>
using namespace std;
int main() {
// current date/time based on current system
time_t now = time(0);
cout << "Number of sec since January 1,1970 is:: " << now << endl;
tm *ltm = localtime(&now);
// print various components of tm structure.
cout << "Year:" << 1900 + ltm->tm_year<<endl;
cout << "Month: "<< 1 + ltm->tm_mon<< endl;
cout << "Day: "<< ltm->tm_mday << endl;
cout << "Time: "<< 5+ltm->tm_hour << ":";
cout << 30+ltm->tm_min << ":";
cout << ltm->tm_sec << endl;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Number of sec since January 1,1970 is:: 1588485717
Year:2020
Month: 5
Day: 3
Time: 11:31:57
C ++標準ライブラリは、後続の章で説明する入出力機能の広範なセットを提供します。この章では、C ++プログラミングに必要な非常に基本的で最も一般的なI / O操作について説明します。
C ++ I / Oは、バイトのシーケンスであるストリームで発生します。キーボード、ディスクドライブ、ネットワーク接続などのデバイスからメインメモリにバイトが流れる場合、これはinput operation また、バイトがメインメモリからディスプレイ画面、プリンタ、ディスクドライブ、ネットワーク接続などのデバイスに流れる場合、これは呼び出されます。 output operation。
I / Oライブラリヘッダーファイル
C ++プログラムにとって重要な次のヘッダーファイルがあります-
シニア番号 | ヘッダーファイルと機能および説明 |
---|---|
1 | <iostream> このファイルは、 cin, cout, cerr そして clog オブジェクト。それぞれ、標準入力ストリーム、標準出力ストリーム、バッファリングされていない標準エラーストリーム、およびバッファリングされた標準エラーストリームに対応します。 |
2 | <iomanip> このファイルは、次のようないわゆるパラメータ化されたストリームマニピュレータを使用してフォーマットされたI / Oを実行するのに役立つサービスを宣言します。 setw そして setprecision。 |
3 | <fstream> このファイルは、ユーザー制御のファイル処理のためのサービスを宣言します。これについては、ファイルとストリームに関連する章で詳しく説明します。 |
標準出力ストリーム(cout)
事前定義されたオブジェクト cout のインスタンスです ostreamクラス。coutオブジェクトは、通常は表示画面である標準出力デバイスに「接続」されていると言われます。ザ・cout 次の例に示すように、記号より2つ小さい<<として記述されるストリーム挿入演算子と組み合わせて使用されます。
#include <iostream>
using namespace std;
int main() {
char str[] = "Hello C++";
cout << "Value of str is : " << str << endl;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Value of str is : Hello C++
C ++コンパイラは、出力される変数のデータ型も決定し、適切なストリーム挿入演算子を選択して値を表示します。<<演算子は、組み込み型integer、float、double、strings、およびpointer値のデータ項目を出力するためにオーバーロードされます。
上記のように、挿入演算子<<は、1つのステートメントで複数回使用できます。 endl 行末に改行を追加するために使用されます。
標準入力ストリーム(cin)
事前定義されたオブジェクト cin のインスタンスです istreamクラス。cinオブジェクトは、通常はキーボードである標準の入力デバイスに接続されていると言われています。ザ・cin は、次の例に示すように、記号より2つ大きい>>として記述されるストリーム抽出演算子と組み合わせて使用されます。
#include <iostream>
using namespace std;
int main() {
char name[50];
cout << "Please enter your name: ";
cin >> name;
cout << "Your name is: " << name << endl;
}
上記のコードをコンパイルして実行すると、名前の入力を求められます。値を入力してからEnterキーを押すと、次の結果が表示されます-
Please enter your name: cplusplus
Your name is: cplusplus
C ++コンパイラは、入力された値のデータ型も判別し、適切なストリーム抽出演算子を選択して値を抽出し、指定された変数に格納します。
ストリーム抽出演算子>>は、1つのステートメントで複数回使用できます。複数のデータを要求するには、次を使用できます-
cin >> name >> age;
これは、次の2つのステートメントと同等になります-
cin >> name;
cin >> age;
標準エラーストリーム(cerr)
事前定義されたオブジェクト cerr のインスタンスです ostreamクラス。cerrオブジェクトは、標準エラーデバイスに接続されていると言われます。これは、表示画面でもありますが、オブジェクトです。cerr はバッファリングされておらず、cerrへの各ストリーム挿入により、その出力がすぐに表示されます。
ザ・ cerr 次の例に示すように、ストリーム挿入演算子と組み合わせて使用することもできます。
#include <iostream>
using namespace std;
int main() {
char str[] = "Unable to read....";
cerr << "Error message : " << str << endl;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Error message : Unable to read....
標準ログストリーム(clog)
事前定義されたオブジェクト clog のインスタンスです ostreamクラス。詰まりオブジェクトは、表示画面でもあるがオブジェクトである標準エラーデバイスに接続されていると言われますclogバッファリングされます。これは、詰まりを挿入するたびに、バッファーがいっぱいになるまで、またはバッファーがフラッシュされるまで、その出力がバッファーに保持される可能性があることを意味します。
ザ・ clog 次の例に示すように、ストリーム挿入演算子と組み合わせて使用することもできます。
#include <iostream>
using namespace std;
int main() {
char str[] = "Unable to read....";
clog << "Error message : " << str << endl;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Error message : Unable to read....
これらの小さな例では、cout、cerr、clogの違いを確認することはできませんが、大きなプログラムを作成して実行すると、違いが明らかになります。したがって、cerrストリームを使用してエラーメッセージを表示することをお勧めします。他のログメッセージを表示するときは、clogを使用する必要があります。
C / C ++配列を使用すると、同じ種類の複数のデータ項目を組み合わせる変数を定義できますが、 structure は、さまざまな種類のデータ項目を組み合わせることができる別のユーザー定義データ型です。
構造はレコードを表すために使用されます。図書館で本を追跡したいとします。各本に関する次の属性を追跡することをお勧めします-
- Title
- Author
- Subject
- ブックID
構造の定義
構造体を定義するには、structステートメントを使用する必要があります。structステートメントは、プログラムの複数のメンバーを持つ新しいデータ型を定義します。構造体ステートメントの形式は次のとおりです-
struct [structure tag] {
member definition;
member definition;
...
member definition;
} [one or more structure variables];
ザ・ structure tagオプションであり、各メンバー定義はintiなどの通常の変数定義です。またはfloatf; またはその他の有効な変数定義。構造体の定義の最後、最後のセミコロンの前に、1つ以上の構造体変数を指定できますが、これはオプションです。これがあなたが本の構造を宣言する方法です-
struct Books {
char title[50];
char author[50];
char subject[100];
int book_id;
} book;
構造体メンバーへのアクセス
構造体の任意のメンバーにアクセスするには、 member access operator (.)。メンバーアクセス演算子は、構造体変数名とアクセスする構造体メンバーの間のピリオドとしてコード化されます。あなたは使用しますstruct構造タイプの変数を定義するキーワード。以下は、構造の使用法を説明する例です-
#include <iostream>
#include <cstring>
using namespace std;
struct Books {
char title[50];
char author[50];
char subject[100];
int book_id;
};
int main() {
struct Books Book1; // Declare Book1 of type Book
struct Books Book2; // Declare Book2 of type Book
// book 1 specification
strcpy( Book1.title, "Learn C++ Programming");
strcpy( Book1.author, "Chand Miyan");
strcpy( Book1.subject, "C++ Programming");
Book1.book_id = 6495407;
// book 2 specification
strcpy( Book2.title, "Telecom Billing");
strcpy( Book2.author, "Yakit Singha");
strcpy( Book2.subject, "Telecom");
Book2.book_id = 6495700;
// Print Book1 info
cout << "Book 1 title : " << Book1.title <<endl;
cout << "Book 1 author : " << Book1.author <<endl;
cout << "Book 1 subject : " << Book1.subject <<endl;
cout << "Book 1 id : " << Book1.book_id <<endl;
// Print Book2 info
cout << "Book 2 title : " << Book2.title <<endl;
cout << "Book 2 author : " << Book2.author <<endl;
cout << "Book 2 subject : " << Book2.subject <<endl;
cout << "Book 2 id : " << Book2.book_id <<endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Book 1 title : Learn C++ Programming
Book 1 author : Chand Miyan
Book 1 subject : C++ Programming
Book 1 id : 6495407
Book 2 title : Telecom Billing
Book 2 author : Yakit Singha
Book 2 subject : Telecom
Book 2 id : 6495700
関数の引数としての構造
他の変数やポインタを渡すのと非常によく似た方法で、構造体を関数の引数として渡すことができます。上記の例でアクセスしたのと同様の方法で構造変数にアクセスします-
#include <iostream>
#include <cstring>
using namespace std;
void printBook( struct Books book );
struct Books {
char title[50];
char author[50];
char subject[100];
int book_id;
};
int main() {
struct Books Book1; // Declare Book1 of type Book
struct Books Book2; // Declare Book2 of type Book
// book 1 specification
strcpy( Book1.title, "Learn C++ Programming");
strcpy( Book1.author, "Chand Miyan");
strcpy( Book1.subject, "C++ Programming");
Book1.book_id = 6495407;
// book 2 specification
strcpy( Book2.title, "Telecom Billing");
strcpy( Book2.author, "Yakit Singha");
strcpy( Book2.subject, "Telecom");
Book2.book_id = 6495700;
// Print Book1 info
printBook( Book1 );
// Print Book2 info
printBook( Book2 );
return 0;
}
void printBook( struct Books book ) {
cout << "Book title : " << book.title <<endl;
cout << "Book author : " << book.author <<endl;
cout << "Book subject : " << book.subject <<endl;
cout << "Book id : " << book.book_id <<endl;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Book title : Learn C++ Programming
Book author : Chand Miyan
Book subject : C++ Programming
Book id : 6495407
Book title : Telecom Billing
Book author : Yakit Singha
Book subject : Telecom
Book id : 6495700
構造体へのポインタ
次のように他の変数へのポインタを定義するのと非常によく似た方法で、構造体へのポインタを定義できます。
struct Books *struct_pointer;
これで、構造体変数のアドレスを上記で定義したポインタ変数に格納できます。構造体変数のアドレスを見つけるには、次のように構造体の名前の前に&演算子を置きます。
struct_pointer = &Book1;
構造体へのポインタを使用して構造体のメンバーにアクセスするには、次のように->演算子を使用する必要があります-
struct_pointer->title;
構造体ポインタを使用して上記の例を書き直してみましょう。これが概念を理解しやすいことを願っています-
#include <iostream>
#include <cstring>
using namespace std;
void printBook( struct Books *book );
struct Books {
char title[50];
char author[50];
char subject[100];
int book_id;
};
int main() {
struct Books Book1; // Declare Book1 of type Book
struct Books Book2; // Declare Book2 of type Book
// Book 1 specification
strcpy( Book1.title, "Learn C++ Programming");
strcpy( Book1.author, "Chand Miyan");
strcpy( Book1.subject, "C++ Programming");
Book1.book_id = 6495407;
// Book 2 specification
strcpy( Book2.title, "Telecom Billing");
strcpy( Book2.author, "Yakit Singha");
strcpy( Book2.subject, "Telecom");
Book2.book_id = 6495700;
// Print Book1 info, passing address of structure
printBook( &Book1 );
// Print Book1 info, passing address of structure
printBook( &Book2 );
return 0;
}
// This function accept pointer to structure as parameter.
void printBook( struct Books *book ) {
cout << "Book title : " << book->title <<endl;
cout << "Book author : " << book->author <<endl;
cout << "Book subject : " << book->subject <<endl;
cout << "Book id : " << book->book_id <<endl;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Book title : Learn C++ Programming
Book author : Chand Miyan
Book subject : C++ Programming
Book id : 6495407
Book title : Telecom Billing
Book author : Yakit Singha
Book subject : Telecom
Book id : 6495700
typedefキーワード
構造体を定義する簡単な方法があります。または、作成するタイプを「エイリアス」することもできます。例-
typedef struct {
char title[50];
char author[50];
char subject[100];
int book_id;
} Books;
さて、あなたは使用することができ書籍の変数を定義するために直接、書籍のstructキーワードを使用せずにタイプを。以下は例です-
Books Book1, Book2;
使用できます typedef 非構造体のキーワードと次のように-
typedef long int *pint32;
pint32 x, y, z;
x、y、zはすべてlongintへのポインタです。
C ++プログラミングの主な目的は、Cプログラミング言語にオブジェクト指向を追加することです。クラスは、オブジェクト指向プログラミングをサポートするC ++の中心的な機能であり、ユーザー定義型と呼ばれることがよくあります。
クラスはオブジェクトの形式を指定するために使用され、データ表現とそのデータを操作するためのメソッドを1つのきちんとしたパッケージに結合します。クラス内のデータと関数は、クラスのメンバーと呼ばれます。
C ++クラス定義
クラスを定義するときは、データ型の青写真を定義します。これは実際にはデータを定義しませんが、クラス名の意味、つまり、クラスのオブジェクトが何で構成され、そのようなオブジェクトに対してどのような操作を実行できるかを定義します。
クラス定義はキーワードで始まります classその後にクラス名が続きます。中括弧のペアで囲まれたクラス本体。クラス定義の後には、セミコロンまたは宣言のリストを続ける必要があります。たとえば、キーワードを使用してBoxデータ型を定義しましたclass 次のように-
class Box {
public:
double length; // Length of a box
double breadth; // Breadth of a box
double height; // Height of a box
};
キーワード publicそれに続くクラスのメンバーのアクセス属性を決定します。パブリックメンバーには、クラスオブジェクトのスコープ内のどこからでもクラスの外部からアクセスできます。クラスのメンバーを次のように指定することもできますprivate または protected これについては、サブセクションで説明します。
C ++オブジェクトを定義する
クラスはオブジェクトの青写真を提供するため、基本的にオブジェクトはクラスから作成されます。クラスのオブジェクトは、基本型の変数を宣言するのとまったく同じ種類の宣言で宣言します。次のステートメントは、クラスBox −の2つのオブジェクトを宣言します。
Box Box1; // Declare Box1 of type Box
Box Box2; // Declare Box2 of type Box
オブジェクトBox1とBox2の両方に、独自のデータメンバーのコピーがあります。
データメンバーへのアクセス
クラスのオブジェクトのパブリックデータメンバーには、直接メンバーアクセス演算子(。)を使用してアクセスできます。次の例を試して、状況を明確にしましょう。
#include <iostream>
using namespace std;
class Box {
public:
double length; // Length of a box
double breadth; // Breadth of a box
double height; // Height of a box
};
int main() {
Box Box1; // Declare Box1 of type Box
Box Box2; // Declare Box2 of type Box
double volume = 0.0; // Store the volume of a box here
// box 1 specification
Box1.height = 5.0;
Box1.length = 6.0;
Box1.breadth = 7.0;
// box 2 specification
Box2.height = 10.0;
Box2.length = 12.0;
Box2.breadth = 13.0;
// volume of box 1
volume = Box1.height * Box1.length * Box1.breadth;
cout << "Volume of Box1 : " << volume <<endl;
// volume of box 2
volume = Box2.height * Box2.length * Box2.breadth;
cout << "Volume of Box2 : " << volume <<endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Volume of Box1 : 210
Volume of Box2 : 1560
プライベートメンバーと保護されたメンバーには、直接メンバーアクセス演算子(。)を使用して直接アクセスできないことに注意してください。プライベートメンバーと保護されたメンバーにアクセスする方法を学習します。
クラスとオブジェクトの詳細
これまでのところ、C ++クラスとオブジェクトに関する非常に基本的な考え方があります。C ++クラスとオブジェクトに関連するさらに興味深い概念があり、以下にリストされているさまざまなサブセクションで説明します。
シニア番号 | コンセプトと説明 |
---|---|
1 | クラスメンバー関数 クラスのメンバー関数は、他の変数と同様に、クラス定義内にその定義またはプロトタイプを持つ関数です。 |
2 | クラスアクセス修飾子 クラスメンバーは、パブリック、プライベート、または保護として定義できます。デフォルトでは、メンバーはプライベートと見なされます。 |
3 | コンストラクタ&デストラクタ クラスコンストラクターは、クラスの新しいオブジェクトが作成されたときに呼び出されるクラス内の特別な関数です。デストラクタは、作成されたオブジェクトが削除されたときに呼び出される特別な関数でもあります。 |
4 | コピーコンストラクタ コピーコンストラクターは、以前に作成された同じクラスのオブジェクトでオブジェクトを初期化することによってオブジェクトを作成するコンストラクターです。 |
5 | フレンド機能 A friend 関数は、クラスのプライベートおよび保護されたメンバーへのフルアクセスを許可されます。 |
6 | インライン関数 インライン関数を使用すると、コンパイラーは、関数の呼び出しの代わりに、関数の本体のコードを展開しようとします。 |
7 | このポインタ すべてのオブジェクトには特別なポインタがあります this これはオブジェクト自体を指します。 |
8 | C ++クラスへのポインタ クラスへのポインタは、構造体へのポインタとまったく同じ方法で実行されます。実際、クラスは実際には関数を含む単なる構造です。 |
9 | クラスの静的メンバー クラスのデータメンバーと関数メンバーの両方を静的として宣言できます。 |
オブジェクト指向プログラミングで最も重要な概念の1つは、継承の概念です。継承により、別のクラスの観点からクラスを定義できるため、アプリケーションの作成と保守が容易になります。これは、コード機能と迅速な実装時間を再利用する機会も提供します。
クラスを作成するとき、プログラマーは、完全に新しいデータメンバーとメンバー関数を作成する代わりに、新しいクラスが既存のクラスのメンバーを継承するように指定できます。この既存のクラスは、base クラス、および新しいクラスはと呼ばれます derived クラス。
継承のアイデアは、 is a関係。たとえば、哺乳類IS-A動物、犬IS-A哺乳類、したがって犬IS-A動物などです。
基本クラスと派生クラス
クラスは複数のクラスから派生させることができます。つまり、複数の基本クラスからデータと関数を継承できます。派生クラスを定義するには、クラス派生リストを使用して基本クラスを指定します。クラス派生リストは、1つ以上の基本クラスに名前を付け、次の形式を持ちます。
class derived-class: access-specifier base-class
access-specifierが次のいずれかである場合 public, protected, または private、およびbase-classは、以前に定義されたクラスの名前です。アクセス指定子が使用されていない場合、デフォルトではプライベートです。
基本クラスを検討する Shape およびその派生クラス Rectangle 次のように-
#include <iostream>
using namespace std;
// Base class
class Shape {
public:
void setWidth(int w) {
width = w;
}
void setHeight(int h) {
height = h;
}
protected:
int width;
int height;
};
// Derived class
class Rectangle: public Shape {
public:
int getArea() {
return (width * height);
}
};
int main(void) {
Rectangle Rect;
Rect.setWidth(5);
Rect.setHeight(7);
// Print the area of the object.
cout << "Total area: " << Rect.getArea() << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Total area: 35
アクセス制御と継承
派生クラスは、その基本クラスのすべての非プライベートメンバーにアクセスできます。したがって、派生クラスのメンバー関数にアクセスできない基本クラスのメンバーは、基本クラスでプライベートとして宣言する必要があります。
次の方法でアクセスできるユーザーに応じて、さまざまなアクセスタイプを要約できます。
アクセス | 公衆 | 保護 | 民間 |
---|---|---|---|
同階級 | はい | はい | はい |
派生クラス | はい | はい | 番号 |
クラス外 | はい | 番号 | 番号 |
派生クラスは、次の例外を除いて、すべての基本クラスメソッドを継承します-
- 基本クラスのコンストラクタ、デストラクタ、およびコピーコンストラクタ。
- 基本クラスのオーバーロードされた演算子。
- 基本クラスのフレンド機能。
継承の種類
基本クラスからクラスを派生させる場合、基本クラスは public, protected または private継承。継承のタイプは、上で説明したようにアクセス指定子によって指定されます。
ほとんど使用しません protected または private 継承、しかし public継承が一般的に使用されます。異なるタイプの継承を使用している間、次のルールが適用されます-
Public Inheritance −からクラスを導出する場合 public 基本クラス、 public 基本クラスのメンバーは public 派生クラスのメンバーと protected 基本クラスのメンバーは protected派生クラスのメンバー。基本クラスのprivate メンバーは、派生クラスから直接アクセスすることはできませんが、への呼び出しを介してアクセスできます。 public そして protected 基本クラスのメンバー。
Protected Inheritance −から派生する場合 protected 基本クラス、 public そして protected 基本クラスのメンバーは protected 派生クラスのメンバー。
Private Inheritance −から派生する場合 private 基本クラス、 public そして protected 基本クラスのメンバーは private 派生クラスのメンバー。
多重継承
C ++クラスは、複数のクラスからメンバーを継承できます。拡張構文は次のとおりです。
class derived-class: access baseA, access baseB....
アクセスが1つである場所 public, protected, または privateそして、すべての基本クラスに与えられ、上記のようにコンマで区切られます。次の例を試してみましょう-
#include <iostream>
using namespace std;
// Base class Shape
class Shape {
public:
void setWidth(int w) {
width = w;
}
void setHeight(int h) {
height = h;
}
protected:
int width;
int height;
};
// Base class PaintCost
class PaintCost {
public:
int getCost(int area) {
return area * 70;
}
};
// Derived class
class Rectangle: public Shape, public PaintCost {
public:
int getArea() {
return (width * height);
}
};
int main(void) {
Rectangle Rect;
int area;
Rect.setWidth(5);
Rect.setHeight(7);
area = Rect.getArea();
// Print the area of the object.
cout << "Total area: " << Rect.getArea() << endl;
// Print the total cost of painting
cout << "Total paint cost: $" << Rect.getCost(area) << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Total area: 35
Total paint cost: $2450
C ++では、1つの定義を複数指定できます。 function 名前または operator 同じスコープで、 function overloading そして operator overloading それぞれ。
オーバーロードされた宣言は、同じスコープ内で以前に宣言された宣言と同じ名前で宣言された宣言です。ただし、両方の宣言の引数が異なり、定義(実装)も明らかに異なります。
オーバーロードを呼び出すとき function または operator、コンパイラは、関数または演算子の呼び出しに使用した引数タイプを定義で指定されたパラメータタイプと比較することにより、使用する最も適切な定義を決定します。最も適切なオーバーロードされた関数または演算子を選択するプロセスが呼び出されますoverload resolution。
C ++での関数のオーバーロード
同じスコープ内の同じ関数名に対して複数の定義を持つことができます。関数の定義は、引数リスト内の引数のタイプや数によって互いに異なる必要があります。戻り値の型のみが異なる関数宣言をオーバーロードすることはできません。
以下は同じ機能の例です print() さまざまなデータ型を印刷するために使用されています-
#include <iostream>
using namespace std;
class printData {
public:
void print(int i) {
cout << "Printing int: " << i << endl;
}
void print(double f) {
cout << "Printing float: " << f << endl;
}
void print(char* c) {
cout << "Printing character: " << c << endl;
}
};
int main(void) {
printData pd;
// Call print to print integer
pd.print(5);
// Call print to print float
pd.print(500.263);
// Call print to print character
pd.print("Hello C++");
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Printing int: 5
Printing float: 500.263
Printing character: Hello C++
C ++での演算子のオーバーロード
C ++で使用可能なほとんどの組み込み演算子を再定義またはオーバーロードできます。したがって、プログラマーはユーザー定義型の演算子も使用できます。
オーバーロードされた演算子は、特別な名前を持つ関数です。キーワード「operator」の後に、定義されている演算子の記号が続きます。他の関数と同様に、オーバーロードされた演算子には戻り値の型とパラメーターリストがあります。
Box operator+(const Box&);
に使用できる加算演算子を宣言します add2つのBoxオブジェクトであり、最後のBoxオブジェクトを返します。ほとんどのオーバーロードされた演算子は、通常の非メンバー関数またはクラスメンバー関数として定義できます。上記の関数をクラスの非メンバー関数として定義する場合、次のように各オペランドに2つの引数を渡す必要があります。
Box operator+(const Box&, const Box&);
以下は、メンバー関数を使用した演算子のオーバーロードの概念を示す例です。ここで、オブジェクトは、このオブジェクトを使用してプロパティにアクセスする引数として渡されます。この演算子を呼び出すオブジェクトには、を使用してアクセスできます。this 以下に説明する演算子-
#include <iostream>
using namespace std;
class Box {
public:
double getVolume(void) {
return length * breadth * height;
}
void setLength( double len ) {
length = len;
}
void setBreadth( double bre ) {
breadth = bre;
}
void setHeight( double hei ) {
height = hei;
}
// Overload + operator to add two Box objects.
Box operator+(const Box& b) {
Box box;
box.length = this->length + b.length;
box.breadth = this->breadth + b.breadth;
box.height = this->height + b.height;
return box;
}
private:
double length; // Length of a box
double breadth; // Breadth of a box
double height; // Height of a box
};
// Main function for the program
int main() {
Box Box1; // Declare Box1 of type Box
Box Box2; // Declare Box2 of type Box
Box Box3; // Declare Box3 of type Box
double volume = 0.0; // Store the volume of a box here
// box 1 specification
Box1.setLength(6.0);
Box1.setBreadth(7.0);
Box1.setHeight(5.0);
// box 2 specification
Box2.setLength(12.0);
Box2.setBreadth(13.0);
Box2.setHeight(10.0);
// volume of box 1
volume = Box1.getVolume();
cout << "Volume of Box1 : " << volume <<endl;
// volume of box 2
volume = Box2.getVolume();
cout << "Volume of Box2 : " << volume <<endl;
// Add two object as follows:
Box3 = Box1 + Box2;
// volume of box 3
volume = Box3.getVolume();
cout << "Volume of Box3 : " << volume <<endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Volume of Box1 : 210
Volume of Box2 : 1560
Volume of Box3 : 5400
オーバーロード可能/非オーバーロード可能オペレーター
以下は、オーバーロードできる演算子のリストです-
+ | - | * | / | % | ^ |
& | | | 〜 | ! | 、 | = |
< | >> | <= | > = | ++ | - |
<< | >> | == | != | && | || |
+ = | -= | / = | %= | ^ = | &= |
| = | * = | << = | >> = | [] | () |
-> | -> * | 新着 | 新着 [] | 削除 | 削除[] |
以下は、オーバーロードできない演算子のリストです-
::: | 。* | 。 | ?: |
演算子のオーバーロードの例
概念を理解するのに役立つさまざまな演算子のオーバーロードの例を次に示します。
シニア番号 | 演算子と例 |
---|---|
1 | 単項演算子のオーバーロード |
2 | 二項演算子のオーバーロード |
3 | 関係演算子のオーバーロード |
4 | 入出力演算子のオーバーロード |
5 | ++および-演算子のオーバーロード |
6 | 代入演算子のオーバーロード |
7 | 関数呼び出し()演算子のオーバーロード |
8 | 添え字[]演算子のオーバーロード |
9 | クラスメンバーアクセス演算子->オーバーロード |
言葉 polymorphism多くの形を持つことを意味します。通常、ポリモーフィズムは、クラスの階層があり、それらが継承によって関連付けられている場合に発生します。
C ++ポリモーフィズムとは、メンバー関数を呼び出すと、その関数を呼び出すオブジェクトのタイプに応じて、異なる関数が実行されることを意味します。
基本クラスが他の2つのクラスによって派生した次の例を考えてみましょう-
#include <iostream>
using namespace std;
class Shape {
protected:
int width, height;
public:
Shape( int a = 0, int b = 0){
width = a;
height = b;
}
int area() {
cout << "Parent class area :" <<endl;
return 0;
}
};
class Rectangle: public Shape {
public:
Rectangle( int a = 0, int b = 0):Shape(a, b) { }
int area () {
cout << "Rectangle class area :" <<endl;
return (width * height);
}
};
class Triangle: public Shape {
public:
Triangle( int a = 0, int b = 0):Shape(a, b) { }
int area () {
cout << "Triangle class area :" <<endl;
return (width * height / 2);
}
};
// Main function for the program
int main() {
Shape *shape;
Rectangle rec(10,7);
Triangle tri(10,5);
// store the address of Rectangle
shape = &rec;
// call rectangle area.
shape->area();
// store the address of Triangle
shape = &tri;
// call triangle area.
shape->area();
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Parent class area :
Parent class area :
誤った出力の理由は、関数area()の呼び出しが、基本クラスで定義されたバージョンとしてコンパイラーによって一度設定されているためです。これは呼ばれますstatic resolution 関数呼び出しの、または static linkage-関数呼び出しは、プログラムが実行される前に修正されます。これは時々呼ばれることもありますearly binding これは、area()関数がプログラムのコンパイル中に設定されるためです。
しかし、ここで、プログラムに少し変更を加えて、Shapeクラスのarea()の宣言の前にキーワードを付けましょう。 virtual このように見えるように-
class Shape {
protected:
int width, height;
public:
Shape( int a = 0, int b = 0) {
width = a;
height = b;
}
virtual int area() {
cout << "Parent class area :" <<endl;
return 0;
}
};
このわずかな変更の後、前のサンプルコードをコンパイルして実行すると、次の結果が生成されます。
Rectangle class area
Triangle class area
今回、コンパイラーは、ポインターのタイプではなく、ポインターの内容を調べます。したがって、triクラスとrecクラスのオブジェクトのアドレスは* shapeに格納されるため、それぞれのarea()関数が呼び出されます。
ご覧のとおり、各子クラスには、関数area()の個別の実装があります。こうやってpolymorphism一般的に使用されます。同じ名前の関数、さらには同じパラメーターを持つ異なるクラスがありますが、実装は異なります。
仮想機能
A virtual functionは、キーワードを使用して宣言された基本クラスの関数です。 virtual。基本クラスで仮想関数を定義し、派生クラスで別のバージョンを使用すると、この関数の静的リンケージが不要であることをコンパイラーに通知します。
私たちが望んでいるのは、プログラムの任意の時点で呼び出される関数を、それが呼び出されるオブジェクトの種類に基づいて選択することです。この種の操作は、dynamic linkage、または late binding。
純粋仮想関数
仮想関数を基本クラスに含めて、そのクラスのオブジェクトに合わせて派生クラスで再定義できるようにすることは可能ですが、基本クラスの関数に意味のある定義を与えることはできません。 。
基本クラスの仮想関数area()を次のように変更できます。
class Shape {
protected:
int width, height;
public:
Shape(int a = 0, int b = 0) {
width = a;
height = b;
}
// pure virtual function
virtual int area() = 0;
};
= 0は、関数に本体がなく、上記の仮想関数が呼び出されることをコンパイラーに通知します。 pure virtual function。
データの抽象化とは、重要な情報のみを外の世界に提供し、その背景の詳細を隠すこと、つまり、詳細を提示せずにプログラムで必要な情報を表すことを指します。
データの抽象化は、インターフェイスと実装の分離に依存するプログラミング(および設計)手法です。
オンとオフを切り替えたり、チャンネルを変更したり、音量を調整したり、スピーカー、VCR、DVDプレーヤーなどの外部コンポーネントを追加したりできる、テレビの実際の例を1つ取り上げましょう。ただし、内部の詳細はわかりません。つまり、無線またはケーブルを介して信号を受信する方法、信号を変換する方法、そして最終的に画面に表示する方法がわかりません。
したがって、テレビは内部実装を外部インターフェイスから明確に分離しており、内部の知識がなくても、電源ボタン、チャンネルチェンジャー、音量調節などのインターフェイスで遊ぶことができます。
C ++では、クラスは優れたレベルの data abstraction。これらは、オブジェクトの機能を操作したり、オブジェクトデータ、つまりクラスが内部でどのように実装されているかを実際に知らなくても状態を操作したりするのに十分なパブリックメソッドを外部に提供します。
たとえば、プログラムはに電話をかけることができます sort()関数が指定された値をソートするために実際に使用するアルゴリズムを知らなくても関数。実際、並べ替え機能の基本的な実装は、ライブラリのリリース間で変更される可能性があり、インターフェイスが同じである限り、関数呼び出しは引き続き機能します。
C ++では、 classes独自の抽象データ型(ADT)を定義します。あなたは使用することができますcout クラスのオブジェクト ostream このようにデータを標準出力にストリーミングするには−
#include <iostream>
using namespace std;
int main() {
cout << "Hello C++" <<endl;
return 0;
}
ここでは、方法を理解する必要はありません coutユーザーの画面にテキストを表示します。パブリックインターフェイスを知っているだけでよく、「cout」の基本的な実装は自由に変更できます。
アクセスラベルは抽象化を強制します
C ++では、アクセスラベルを使用して、クラスへの抽象インターフェイスを定義します。クラスには0個以上のアクセスラベルが含まれる場合があります-
パブリックラベルで定義されたメンバーは、プログラムのすべての部分にアクセスできます。タイプのデータ抽象化ビューは、そのパブリックメンバーによって定義されます。
プライベートラベルで定義されたメンバーは、クラスを使用するコードにアクセスできません。プライベートセクションは、型を使用するコードから実装を隠します。
アクセスラベルが表示される頻度に制限はありません。各アクセスラベルは、後続のメンバー定義のアクセスレベルを指定します。指定されたアクセスレベルは、次のアクセスラベルが検出されるか、クラス本体の右中括弧が表示されるまで有効です。
データ抽象化の利点
データの抽象化には2つの重要な利点があります-
クラスの内部は、オブジェクトの状態を破壊する可能性のある不注意なユーザーレベルのエラーから保護されています。
クラスの実装は、ユーザーレベルのコードを変更することなく、要件の変更やバグレポートに応じて時間の経過とともに進化する可能性があります。
クラスのプライベートセクションでのみデータメンバーを定義することにより、クラスの作成者はデータを自由に変更できます。実装が変更された場合は、クラスコードのみを調べて、変更がどのような影響を与えるかを確認する必要があります。データが公開されている場合、古い表現のデータメンバーに直接アクセスする関数が壊れている可能性があります。
データ抽象化の例
パブリックメンバーとプライベートメンバーでクラスを実装するC ++プログラムは、データ抽象化の例です。次の例を考えてみましょう-
#include <iostream>
using namespace std;
class Adder {
public:
// constructor
Adder(int i = 0) {
total = i;
}
// interface to outside world
void addNum(int number) {
total += number;
}
// interface to outside world
int getTotal() {
return total;
};
private:
// hidden data from outside world
int total;
};
int main() {
Adder a;
a.addNum(10);
a.addNum(20);
a.addNum(30);
cout << "Total " << a.getTotal() <<endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Total 60
上記のクラスは数値を合計し、合計を返します。パブリックメンバー-addNum そして getTotalは外界へのインターフェースであり、ユーザーはクラスを使用するためにそれらを知る必要があります。プライベートメンバーtotal これは、ユーザーが知る必要はありませんが、クラスが正しく動作するために必要なものです。
設計戦略
抽象化は、コードをインターフェースと実装に分離します。したがって、コンポーネントを設計するときは、インターフェースを実装から独立させて、基礎となる実装を変更してもインターフェースが損なわれないようにする必要があります。
この場合、これらのインターフェイスを使用しているプログラムが何であれ、影響を受けることはなく、最新の実装で再コンパイルする必要があります。
すべてのC ++プログラムは、次の2つの基本要素で構成されています-
Program statements (code) −これは、アクションを実行するプログラムの一部であり、関数と呼ばれます。
Program data −データは、プログラム機能の影響を受けるプログラムの情報です。
カプセル化は、データとデータを操作する関数を結合し、外部からの干渉や誤用から安全に保つオブジェクト指向プログラミングの概念です。データのカプセル化は、次の重要なOOP概念につながりました。data hiding。
Data encapsulation データをバンドルするメカニズムであり、それらを使用する関数と data abstraction は、インターフェイスのみを公開し、実装の詳細をユーザーから隠すメカニズムです。
C ++は、と呼ばれるユーザー定義型の作成を通じて、カプセル化とデータ隠蔽のプロパティをサポートします。 classes。クラスに含めることができることはすでに研究しましたprivate, protected そして publicメンバー。デフォルトでは、クラスで定義されているすべてのアイテムはプライベートです。例-
class Box {
public:
double getVolume(void) {
return length * breadth * height;
}
private:
double length; // Length of a box
double breadth; // Breadth of a box
double height; // Height of a box
};
変数の長さ、幅、高さは次のとおりです。 private。つまり、Boxクラスの他のメンバーだけがアクセスでき、プログラムの他の部分からはアクセスできません。これは、カプセル化を実現する1つの方法です。
クラスの一部を作るには public (つまり、プログラムの他の部分にアクセスできる)、後で宣言する必要があります publicキーワード。パブリック指定子の後に定義されたすべての変数または関数は、プログラム内の他のすべての関数からアクセスできます。
あるクラスを別のクラスの友達にすると、実装の詳細が明らかになり、カプセル化が減ります。理想は、各クラスの詳細を他のすべてのクラスからできるだけ隠しておくことです。
データカプセル化の例
パブリックメンバーとプライベートメンバーでクラスを実装するC ++プログラムは、データのカプセル化とデータの抽象化の例です。次の例を考えてみましょう-
#include <iostream>
using namespace std;
class Adder {
public:
// constructor
Adder(int i = 0) {
total = i;
}
// interface to outside world
void addNum(int number) {
total += number;
}
// interface to outside world
int getTotal() {
return total;
};
private:
// hidden data from outside world
int total;
};
int main() {
Adder a;
a.addNum(10);
a.addNum(20);
a.addNum(30);
cout << "Total " << a.getTotal() <<endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Total 60
上記のクラスは数値を合計し、合計を返します。パブリックメンバーaddNum そして getTotal は外界へのインターフェースであり、ユーザーはクラスを使用するためにそれらを知る必要があります。プライベートメンバーtotal 外の世界からは隠されているものですが、クラスが適切に機能するために必要です。
設計戦略
私たちのほとんどは、本当に公開する必要がない限り、デフォルトでクラスメンバーをプライベートにすることを学びました。それはいいですencapsulation。
これはデータメンバーに最も頻繁に適用されますが、仮想関数を含むすべてのメンバーに等しく適用されます。
インターフェイスは、C ++クラスの特定の実装にコミットすることなく、そのクラスの動作または機能を記述します。
C ++インターフェースは、 abstract classes また、これらの抽象クラスを、実装の詳細を関連データから分離するという概念であるデータ抽象化と混同しないでください。
クラスは、その関数の少なくとも1つを次のように宣言することによって抽象化されます。 pure virtual関数。純粋仮想関数は、次のように宣言に「= 0」を配置することで指定されます。
class Box {
public:
// pure virtual function
virtual double getVolume() = 0;
private:
double length; // Length of a box
double breadth; // Breadth of a box
double height; // Height of a box
};
の目的 abstract class(ABCと呼ばれることもあります)は、他のクラスが継承できる適切な基本クラスを提供することです。抽象クラスを使用してオブジェクトをインスタンス化することはできず、interface。抽象クラスのオブジェクトをインスタンス化しようとすると、コンパイルエラーが発生します。
したがって、ABCのサブクラスをインスタンス化する必要がある場合は、各仮想関数を実装する必要があります。つまり、ABCによって宣言されたインターフェイスをサポートします。派生クラスの純粋仮想関数をオーバーライドしてから、そのクラスのオブジェクトをインスタンス化しようとすると、コンパイルエラーになります。
オブジェクトをインスタンス化するために使用できるクラスは、 concrete classes。
抽象クラスの例
親クラスが基本クラスへのインターフェイスを提供して、という関数を実装する次の例について考えてみます。 getArea() −
#include <iostream>
using namespace std;
// Base class
class Shape {
public:
// pure virtual function providing interface framework.
virtual int getArea() = 0;
void setWidth(int w) {
width = w;
}
void setHeight(int h) {
height = h;
}
protected:
int width;
int height;
};
// Derived classes
class Rectangle: public Shape {
public:
int getArea() {
return (width * height);
}
};
class Triangle: public Shape {
public:
int getArea() {
return (width * height)/2;
}
};
int main(void) {
Rectangle Rect;
Triangle Tri;
Rect.setWidth(5);
Rect.setHeight(7);
// Print the area of the object.
cout << "Total Rectangle area: " << Rect.getArea() << endl;
Tri.setWidth(5);
Tri.setHeight(7);
// Print the area of the object.
cout << "Total Triangle area: " << Tri.getArea() << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Total Rectangle area: 35
Total Triangle area: 17
抽象クラスがgetArea()の観点からインターフェイスを定義し、他の2つのクラスが同じ関数を実装しましたが、形状に固有の領域を計算するためのアルゴリズムが異なることがわかります。
設計戦略
オブジェクト指向システムは、抽象基本クラスを使用して、すべての外部アプリケーションに適した共通の標準化されたインターフェースを提供する場合があります。次に、その抽象基本クラスからの継承を通じて、同様に動作する派生クラスが形成されます。
外部アプリケーションによって提供される機能(つまり、パブリック関数)は、抽象基本クラスの純粋仮想関数として提供されます。これらの純粋仮想関数の実装は、アプリケーションの特定のタイプに対応する派生クラスで提供されます。
このアーキテクチャにより、システムが定義された後でも、新しいアプリケーションをシステムに簡単に追加することができます。
これまで、 iostream 提供する標準ライブラリ cin そして cout それぞれ標準入力からの読み取りと標準出力への書き込みの方法。
このチュートリアルでは、ファイルの読み取りと書き込みの方法を説明します。これには、別の標準C ++ライブラリが必要です。fstream、3つの新しいデータ型を定義します-
シニア番号 | データ型と説明 |
---|---|
1 | ofstream このデータ型は出力ファイルストリームを表し、ファイルの作成とファイルへの情報の書き込みに使用されます。 |
2 | ifstream このデータ型は入力ファイルストリームを表し、ファイルから情報を読み取るために使用されます。 |
3 | fstream このデータ型は一般にファイルストリームを表し、ストリームとifstreamの両方の機能を備えています。つまり、ファイルの作成、ファイルへの情報の書き込み、ファイルからの情報の読み取りを行うことができます。 |
C ++でファイル処理を実行するには、ヘッダーファイル<iostream>と<fstream>をC ++ソースファイルにインクルードする必要があります。
ファイルを開く
ファイルからの読み取りまたはファイルへの書き込みを行う前に、ファイルを開く必要があります。どちらかofstream または fstreamオブジェクトは、書き込み用にファイルを開くために使用できます。また、ifstreamオブジェクトは、読み取り目的でのみファイルを開くために使用されます。
以下は、fstream、ifstream、およびofstreamオブジェクトのメンバーであるopen()関数の標準構文です。
void open(const char *filename, ios::openmode mode);
ここで、最初の引数は開くファイルの名前と場所を指定し、2番目の引数は open() メンバー関数は、ファイルを開くモードを定義します。
シニア番号 | モードフラグと説明 |
---|---|
1 | ios::app 追加モード。そのファイルへのすべての出力は、最後に追加されます。 |
2 | ios::ate 出力用にファイルを開き、読み取り/書き込みコントロールをファイルの最後に移動します。 |
3 | ios::in 読み取るためにファイルを開きます。 |
4 | ios::out 書き込み用にファイルを開きます。 |
5 | ios::trunc ファイルがすでに存在する場合、ファイルを開く前にその内容が切り捨てられます。 |
これらの値の2つ以上を次のように組み合わせることができます ORそれらを一緒に。たとえば、ファイルを書き込みモードで開き、すでに存在する場合にそれを切り捨てたい場合、次の構文になります-
ofstream outfile;
outfile.open("file.dat", ios::out | ios::trunc );
同様に、次のように読み取りと書き込みの目的でファイルを開くことができます。
fstream afile;
afile.open("file.dat", ios::out | ios::in );
ファイルを閉じる
C ++プログラムが終了すると、すべてのストリームが自動的にフラッシュされ、割り当てられたすべてのメモリが解放され、開いているすべてのファイルが閉じられます。ただし、プログラマーは、プログラムを終了する前に、開いているすべてのファイルを閉じることをお勧めします。
以下は、fstream、ifstream、およびofstreamオブジェクトのメンバーであるclose()関数の標準構文です。
void close();
ファイルへの書き込み
C ++プログラミングを実行している間、ストリーム挿入演算子(<<)を使用して情報を画面に出力するのと同じように、プログラムからファイルに情報を書き込みます。唯一の違いは、ofstream または fstream の代わりにオブジェクト cout オブジェクト。
ファイルからの読み取り
ストリーム抽出演算子(>>)を使用してキーボードから情報を入力するのと同じように、ファイルからプログラムに情報を読み込みます。唯一の違いは、ifstream または fstream の代わりにオブジェクト cin オブジェクト。
読み取りと書き込みの例
以下は、読み取りおよび書き込みモードでファイルを開くC ++プログラムです。ユーザーが入力した情報をafile.datという名前のファイルに書き込んだ後、プログラムはファイルから情報を読み取り、それを画面に出力します。
#include <fstream>
#include <iostream>
using namespace std;
int main () {
char data[100];
// open a file in write mode.
ofstream outfile;
outfile.open("afile.dat");
cout << "Writing to the file" << endl;
cout << "Enter your name: ";
cin.getline(data, 100);
// write inputted data into the file.
outfile << data << endl;
cout << "Enter your age: ";
cin >> data;
cin.ignore();
// again write inputted data into the file.
outfile << data << endl;
// close the opened file.
outfile.close();
// open a file in read mode.
ifstream infile;
infile.open("afile.dat");
cout << "Reading from the file" << endl;
infile >> data;
// write the data at the screen.
cout << data << endl;
// again read the data from the file and display it.
infile >> data;
cout << data << endl;
// close the opened file.
infile.close();
return 0;
}
上記のコードをコンパイルして実行すると、次のサンプル入力と出力が生成されます。
$./a.out
Writing to the file
Enter your name: Zara
Enter your age: 9
Reading from the file
Zara
9
上記の例では、外部から行を読み取るgetline()関数や、前のreadステートメントによって残された余分な文字を無視するignore()関数など、cinオブジェクトの追加関数を使用しています。
ファイル位置ポインタ
どちらも istream そして ostreamファイル位置ポインタを再配置するためのメンバー関数を提供します。これらのメンバー関数はseekg ( "seek get")istreamおよび seekp (「シークプット」)ostream用。
seekgとseekpの引数は通常、長整数です。2番目の引数は、シーク方向を示すために指定できます。シーク方向はios::beg (デフォルト)ストリームの開始を基準にして配置する場合、 ios::cur ストリーム内の現在の位置を基準にして配置する場合、または ios::end ストリームの終わりを基準にして配置します。
ファイル位置ポインターは、ファイルの開始位置からのバイト数としてファイル内の位置を指定する整数値です。「get」ファイル位置ポインタの配置の例は次のとおりです。
// position to the nth byte of fileObject (assumes ios::beg)
fileObject.seekg( n );
// position n bytes forward in fileObject
fileObject.seekg( n, ios::cur );
// position n bytes back from end of fileObject
fileObject.seekg( n, ios::end );
// position at end of fileObject
fileObject.seekg( 0, ios::end );
例外は、プログラムの実行中に発生する問題です。C ++例外は、ゼロ除算の試行など、プログラムの実行中に発生する例外的な状況への応答です。
例外は、プログラムのある部分から別の部分に制御を移す方法を提供します。C ++例外処理は、次の3つのキーワードに基づいて構築されています。try, catch, そして throw。
throw−問題が発生すると、プログラムは例外をスローします。これは、throw キーワード。
catch−プログラムは、問題を処理したいプログラム内の場所で例外ハンドラーを使用して例外をキャッチします。ザ・catch キーワードは、例外のキャッチを示します。
try − a tryblockは、特定の例外がアクティブ化されるコードのブロックを識別します。その後に1つ以上のキャッチブロックが続きます。
ブロックが例外を発生させると仮定すると、メソッドは、との組み合わせを使用して例外をキャッチします。 try そして catchキーワード。例外を生成する可能性のあるコードの周囲にtry / catchブロックが配置されます。try / catchブロック内のコードは保護されたコードと呼ばれ、try / catchを使用するための構文は次のとおりです。
try {
// protected code
} catch( ExceptionName e1 ) {
// catch block
} catch( ExceptionName e2 ) {
// catch block
} catch( ExceptionName eN ) {
// catch block
}
複数のリストを作成できます catch さまざまなタイプの例外をキャッチするステートメント try ブロックは、さまざまな状況で複数の例外を発生させます。
例外のスロー
例外は、を使用してコードブロック内のどこにでもスローできます。 throwステートメント。throwステートメントのオペランドは、例外のタイプを決定し、任意の式にすることができ、式の結果のタイプによって、スローされる例外のタイプが決まります。
以下は、ゼロ除算条件が発生したときに例外をスローする例です。
double division(int a, int b) {
if( b == 0 ) {
throw "Division by zero condition!";
}
return (a/b);
}
例外をキャッチする
ザ・ catch 次のブロック tryブロックは例外をキャッチします。キャッチする例外のタイプを指定できます。これは、キーワードcatchに続く括弧内に表示される例外宣言によって決定されます。
try {
// protected code
} catch( ExceptionName e ) {
// code to handle ExceptionName exception
}
上記のコードは例外をキャッチします ExceptionNameタイプ。catchブロックがtryブロックでスローされるすべてのタイプの例外を処理するように指定する場合は、次のように、例外宣言を囲む括弧の間に省略記号...を付ける必要があります。
try {
// protected code
} catch(...) {
// code to handle any exception
}
以下は、ゼロ除算例外をスローし、それをcatchブロックでキャッチする例です。
#include <iostream>
using namespace std;
double division(int a, int b) {
if( b == 0 ) {
throw "Division by zero condition!";
}
return (a/b);
}
int main () {
int x = 50;
int y = 0;
double z = 0;
try {
z = division(x, y);
cout << z << endl;
} catch (const char* msg) {
cerr << msg << endl;
}
return 0;
}
タイプの例外を発生させているため const char*したがって、この例外をキャッチしている間、catchブロックでconst char *を使用する必要があります。上記のコードをコンパイルして実行すると、次の結果が生成されます-
Division by zero condition!
C ++の標準例外
C ++は、で定義されている標準例外のリストを提供します。 <exception>プログラムで使用できます。これらは、以下に示す親子クラス階層に配置されます-
上記の階層で言及されている各例外の簡単な説明は次のとおりです-
シニア番号 | 例外と説明 |
---|---|
1 | std::exception すべての標準C ++例外の例外および親クラス。 |
2 | std::bad_alloc これはによって投げることができます new。 |
3 | std::bad_cast これはによって投げることができます dynamic_cast。 |
4 | std::bad_exception これは、C ++プログラムで予期しない例外を処理するのに便利なデバイスです。 |
5 | std::bad_typeid これはによって投げることができます typeid。 |
6 | std::logic_error コードを読み取ることで理論的に検出できる例外。 |
7 | std::domain_error これは、数学的に無効なドメインが使用された場合にスローされる例外です。 |
8 | std::invalid_argument これは、無効な引数が原因でスローされます。 |
9 | std::length_error これは、大きすぎるstd :: stringが作成されたときにスローされます。 |
10 | std::out_of_range これは、「at」メソッドによってスローできます。たとえば、std :: vectorやstd :: bitset <> :: operator []()などです。 |
11 | std::runtime_error コードを読み取っても理論的には検出できない例外。 |
12 | std::overflow_error これは、数学的なオーバーフローが発生した場合にスローされます。 |
13 | std::range_error これは、範囲外の値を保存しようとしたときに発生します。 |
14 | std::underflow_error これは、数学的アンダーフローが発生した場合にスローされます。 |
新しい例外を定義する
継承してオーバーライドすることで、独自の例外を定義できます exceptionクラスの機能。以下は、std :: exceptionクラスを使用して、標準的な方法で独自の例外を実装する方法を示す例です。
#include <iostream>
#include <exception>
using namespace std;
struct MyException : public exception {
const char * what () const throw () {
return "C++ Exception";
}
};
int main() {
try {
throw MyException();
} catch(MyException& e) {
std::cout << "MyException caught" << std::endl;
std::cout << e.what() << std::endl;
} catch(std::exception& e) {
//Other errors
}
}
これにより、次の結果が生成されます-
MyException caught
C++ Exception
ここに、 what()は例外クラスによって提供されるパブリックメソッドであり、すべての子例外クラスによってオーバーライドされています。これにより、例外の原因が返されます。
動的メモリがC ++で実際にどのように機能するかをよく理解することは、優れたC ++プログラマーになるために不可欠です。C ++プログラムのメモリは2つの部分に分かれています-
The stack −関数内で宣言されたすべての変数は、スタックからメモリを消費します。
The heap −これはプログラムの未使用メモリであり、プログラムの実行時にメモリを動的に割り当てるために使用できます。
多くの場合、定義された変数に特定の情報を格納するために必要なメモリの量を事前に認識しておらず、必要なメモリのサイズは実行時に決定できます。
割り当てられたスペースのアドレスを返すC ++の特別な演算子を使用して、特定のタイプの変数のヒープ内で実行時にメモリを割り当てることができます。この演算子はと呼ばれますnew オペレーター。
動的に割り当てられたメモリが不要になった場合は、次を使用できます。 delete 演算子。以前に新しい演算子によって割り当てられたメモリの割り当てを解除します。
新規および削除演算子
使用する一般的な構文は次のとおりです new 任意のデータ型に動的にメモリを割り当てる演算子。
new data-type;
ここに、 data-type配列を含む任意の組み込みデータ型、またはクラスまたは構造を含む任意のユーザー定義データ型にすることができます。組み込みのデータ型から始めましょう。たとえば、double型へのポインタを定義してから、実行時にメモリを割り当てるように要求できます。これは、new 次のステートメントを持つ演算子-
double* pvalue = NULL; // Pointer initialized with null
pvalue = new double; // Request memory for the variable
フリーストアが使い果たされた場合は、メモリが正常に割り当てられていない可能性があります。したがって、新しい演算子がNULLポインターを返しているかどうかを確認し、以下のように適切なアクションを実行することをお勧めします。
double* pvalue = NULL;
if( !(pvalue = new double )) {
cout << "Error: out of memory." <<endl;
exit(1);
}
ザ・ malloc()Cの関数はC ++にも存在しますが、malloc()関数の使用は避けることをお勧めします。malloc()に対するnewの主な利点は、newがメモリを割り当てるだけでなく、C ++の主な目的であるオブジェクトを構築することです。
いつでも、動的に割り当てられた変数が不要になったと感じたら、次のように「削除」演算子を使用して、空きストアで占有しているメモリを解放できます。
delete pvalue; // Release memory pointed to by pvalue
上記の概念を置き、次の例を作成して、「new」と「delete」がどのように機能するかを示しましょう。
#include <iostream>
using namespace std;
int main () {
double* pvalue = NULL; // Pointer initialized with null
pvalue = new double; // Request memory for the variable
*pvalue = 29494.99; // Store value at allocated address
cout << "Value of pvalue : " << *pvalue << endl;
delete pvalue; // free up the memory.
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が生成されます-
Value of pvalue : 29495
配列の動的メモリ割り当て
文字の配列、つまり20文字の文字列にメモリを割り当てたいとします。上で使用したのと同じ構文を使用して、以下に示すようにメモリを動的に割り当てることができます。
char* pvalue = NULL; // Pointer initialized with null
pvalue = new char[20]; // Request memory for the variable
作成したばかりの配列を削除するには、ステートメントは次のようになります。
delete [] pvalue; // Delete array pointed to by pvalue
new演算子の同様の一般的な構文に従って、次のように多次元配列に割り当てることができます。
double** pvalue = NULL; // Pointer initialized with null
pvalue = new double [3][4]; // Allocate memory for a 3x4 array
ただし、多次元配列のメモリを解放するための構文は、上記と同じままです。
delete [] pvalue; // Delete array pointed to by pvalue
オブジェクトの動的メモリ割り当て
オブジェクトは単純なデータ型と同じです。たとえば、オブジェクトの配列を使用して概念を明確にする次のコードについて考えてみます。
#include <iostream>
using namespace std;
class Box {
public:
Box() {
cout << "Constructor called!" <<endl;
}
~Box() {
cout << "Destructor called!" <<endl;
}
};
int main() {
Box* myBoxArray = new Box[4];
delete [] myBoxArray; // Delete array
return 0;
}
4つのBoxオブジェクトの配列を割り当てる場合、Simpleコンストラクターは4回呼び出され、同様にこれらのオブジェクトを削除するときに、デストラクタも同じ回数呼び出されます。
上記のコードをコンパイルして実行すると、次の結果が生成されます-
Constructor called!
Constructor called!
Constructor called!
Constructor called!
Destructor called!
Destructor called!
Destructor called!
Destructor called!
同じクラスに同じ名前の2人のZaraがいる状況を考えてみましょう。それらを明確に区別する必要があるときはいつでも、地域、別の地域に住んでいる場合、母親または父親の名前など、名前とともにいくつかの追加情報を使用する必要があります。
C ++アプリケーションでも同じ状況が発生する可能性があります。たとえば、xyz()という関数を持つコードを記述していて、同じ関数xyz()を持つ別のライブラリが利用可能であるとします。これで、コンパイラーは、コード内で参照しているxyz()関数のバージョンを知る方法がありません。
A namespaceこの問題を克服するように設計されており、異なるライブラリで使用可能な同じ名前の類似の関数、クラス、変数などを区別するための追加情報として使用されます。名前空間を使用して、名前が定義されているコンテキストを定義できます。本質的に、名前空間はスコープを定義します。
名前空間の定義
名前空間の定義はキーワードで始まります namespace 次のように名前空間名が続きます-
namespace namespace_name {
// code declarations
}
関数または変数の名前空間対応バージョンを呼び出すには、次のように名前空間名を前に付けます(::)-
name::code; // code could be variable or function.
名前空間が変数や関数を含むエンティティをどのようにスコープするかを見てみましょう-
#include <iostream>
using namespace std;
// first name space
namespace first_space {
void func() {
cout << "Inside first_space" << endl;
}
}
// second name space
namespace second_space {
void func() {
cout << "Inside second_space" << endl;
}
}
int main () {
// Calls function from first name space.
first_space::func();
// Calls function from second name space.
second_space::func();
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が生成されます-
Inside first_space
Inside second_space
usingディレクティブ
また、名前空間の先頭に追加することを回避できます。 using namespace指令。このディレクティブは、後続のコードが指定された名前空間の名前を使用していることをコンパイラーに通知します。したがって、名前空間は次のコードに含まれます-
#include <iostream>
using namespace std;
// first name space
namespace first_space {
void func() {
cout << "Inside first_space" << endl;
}
}
// second name space
namespace second_space {
void func() {
cout << "Inside second_space" << endl;
}
}
using namespace first_space;
int main () {
// This calls function from first name space.
func();
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が生成されます-
Inside first_space
'using'ディレクティブを使用して、名前空間内の特定のアイテムを参照することもできます。たとえば、使用する予定のstd名前空間の唯一の部分がcoutである場合、次のように参照できます。
using std::cout;
後続のコードは、名前空間を付加せずにcoutを参照できますが、 std 名前空間は、次のように明示的にする必要があります-
#include <iostream>
using std::cout;
int main () {
cout << "std::endl is used with std!" << std::endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が生成されます-
std::endl is used with std!
で紹介された名前 usingディレクティブは通常のスコープルールに従います。名前はのポイントから表示されますusingディレクティブが見つかったスコープの最後までのディレクティブ。外部スコープで定義された同じ名前のエンティティは非表示になります。
不連続な名前空間
名前空間はいくつかの部分で定義できるため、名前空間は個別に定義された部分の合計で構成されます。名前空間の個別の部分は、複数のファイルに分散できます。
したがって、名前空間の一部で別のファイルで定義された名前が必要な場合でも、その名前を宣言する必要があります。次の名前空間定義を作成すると、新しい名前空間が定義されるか、既存の名前空間に新しい要素が追加されます。
namespace namespace_name {
// code declarations
}
ネストされた名前空間
名前空間はネストでき、次のように別の名前空間内に1つの名前空間を定義できます。
namespace namespace_name1 {
// code declarations
namespace namespace_name2 {
// code declarations
}
}
次のように解決演算子を使用して、ネストされた名前空間のメンバーにアクセスできます。
// to access members of namespace_name2
using namespace namespace_name1::namespace_name2;
// to access members of namespace:name1
using namespace namespace_name1;
上記のステートメントでnamespace_name1を使用している場合、次のように、namespace_name2の要素がスコープで使用可能になります。
#include <iostream>
using namespace std;
// first name space
namespace first_space {
void func() {
cout << "Inside first_space" << endl;
}
// second name space
namespace second_space {
void func() {
cout << "Inside second_space" << endl;
}
}
}
using namespace first_space::second_space;
int main () {
// This calls function from second name space.
func();
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が生成されます-
Inside second_space
テンプレートはジェネリックプログラミングの基盤であり、特定のタイプに依存しない方法でコードを記述します。
テンプレートは、ジェネリッククラスまたは関数を作成するための青写真または式です。イテレータやアルゴリズムなどのライブラリコンテナは、ジェネリックプログラミングの例であり、テンプレートの概念を使用して開発されています。
各コンテナには、次のような単一の定義があります。 vector、しかし、たとえば、さまざまな種類のベクトルを定義できます。 vector <int> または vector <string>。
テンプレートを使用して、クラスだけでなく関数も定義できます。それらがどのように機能するかを見てみましょう。
関数テンプレート
テンプレート関数定義の一般的な形式を次に示します-
template <class type> ret-type func-name(parameter list) {
// body of function
}
ここで、typeは、関数で使用されるデータ型のプレースホルダー名です。この名前は、関数定義内で使用できます。
以下は、最大2つの値を返す関数テンプレートの例です-
#include <iostream>
#include <string>
using namespace std;
template <typename T>
inline T const& Max (T const& a, T const& b) {
return a < b ? b:a;
}
int main () {
int i = 39;
int j = 20;
cout << "Max(i, j): " << Max(i, j) << endl;
double f1 = 13.5;
double f2 = 20.7;
cout << "Max(f1, f2): " << Max(f1, f2) << endl;
string s1 = "Hello";
string s2 = "World";
cout << "Max(s1, s2): " << Max(s1, s2) << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が生成されます-
Max(i, j): 39
Max(f1, f2): 20.7
Max(s1, s2): World
クラステンプレート
関数テンプレートを定義できるのと同じように、クラステンプレートも定義できます。ジェネリッククラス宣言の一般的な形式を次に示します-
template <class type> class class-name {
.
.
.
}
ここに、 typeはプレースホルダータイプ名であり、クラスがインスタンス化されるときに指定されます。コンマ区切りのリストを使用して、複数の汎用データ型を定義できます。
以下は、クラスStack <>を定義し、スタックから要素をプッシュおよびポップするためのジェネリックメソッドを実装する例です。
#include <iostream>
#include <vector>
#include <cstdlib>
#include <string>
#include <stdexcept>
using namespace std;
template <class T>
class Stack {
private:
vector<T> elems; // elements
public:
void push(T const&); // push element
void pop(); // pop element
T top() const; // return top element
bool empty() const { // return true if empty.
return elems.empty();
}
};
template <class T>
void Stack<T>::push (T const& elem) {
// append copy of passed element
elems.push_back(elem);
}
template <class T>
void Stack<T>::pop () {
if (elems.empty()) {
throw out_of_range("Stack<>::pop(): empty stack");
}
// remove last element
elems.pop_back();
}
template <class T>
T Stack<T>::top () const {
if (elems.empty()) {
throw out_of_range("Stack<>::top(): empty stack");
}
// return copy of last element
return elems.back();
}
int main() {
try {
Stack<int> intStack; // stack of ints
Stack<string> stringStack; // stack of strings
// manipulate int stack
intStack.push(7);
cout << intStack.top() <<endl;
// manipulate string stack
stringStack.push("hello");
cout << stringStack.top() << std::endl;
stringStack.pop();
stringStack.pop();
} catch (exception const& ex) {
cerr << "Exception: " << ex.what() <<endl;
return -1;
}
}
上記のコードをコンパイルして実行すると、次の結果が生成されます-
7
hello
Exception: Stack<>::pop(): empty stack
プリプロセッサは、実際のコンパイルが開始される前に情報を前処理するようにコンパイラに指示するディレクティブです。
すべてのプリプロセッサディレクティブは#で始まり、行のプリプロセッサディレクティブの前に空白文字のみを表示できます。プリプロセッサディレクティブはC ++ステートメントではないため、セミコロン(;)で終わりません。
あなたはすでに見ました #includeすべての例でディレクティブ。このマクロは、ヘッダーファイルをソースファイルにインクルードするために使用されます。
C ++でサポートされているプリプロセッサディレクティブには、#include、#define、#if、#else、#lineなどがあります。重要なディレクティブを見てみましょう-
#defineプリプロセッサ
#defineプリプロセッサディレクティブはシンボリック定数を作成します。シンボリック定数はmacro ディレクティブの一般的な形式は-です。
#define macro-name replacement-text
この行がファイルに表示されると、プログラムがコンパイルされる前に、そのファイル内で出現する後続のすべてのマクロが置換テキストに置き換えられます。例-
#include <iostream>
using namespace std;
#define PI 3.14159
int main () {
cout << "Value of PI :" << PI << endl;
return 0;
}
ここで、このコードの前処理を実行して、ソースコードファイルがあると仮定して結果を確認しましょう。それでは、-Eオプションを指定してコンパイルし、結果をtest.pにリダイレクトしましょう。ここで、test.pを確認すると、多くの情報が表示され、下部に次のように値が置き換えられていることがわかります。
$gcc -E test.cpp > test.p
...
int main () {
cout << "Value of PI :" << 3.14159 << endl;
return 0;
}
関数のようなマクロ
#defineを使用して、次のように引数を取るマクロを定義できます。
#include <iostream>
using namespace std;
#define MIN(a,b) (((a)<(b)) ? a : b)
int main () {
int i, j;
i = 100;
j = 30;
cout <<"The minimum is " << MIN(i, j) << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が生成されます-
The minimum is 30
条件付きコンパイル
プログラムのソースコードの選択部分をコンパイルするために使用できるいくつかのディレクティブがあります。このプロセスは条件付きコンパイルと呼ばれます。
条件付きプリプロセッサ構造は、「if」選択構造によく似ています。次のプリプロセッサコードを検討してください-
#ifndef NULL
#define NULL 0
#endif
デバッグ目的でプログラムをコンパイルできます。次のように、単一のマクロを使用してデバッグをオンまたはオフにすることもできます。
#ifdef DEBUG
cerr <<"Variable x = " << x << endl;
#endif
これにより、 cerrシンボリック定数DEBUGがディレクティブ#ifdefDEBUGの前に定義されている場合に、プログラムでコンパイルされるステートメント。次のように、#if 0ステートメントを使用して、プログラムの一部をコメントアウトできます。
#if 0
code prevented from compiling
#endif
次の例を試してみましょう-
#include <iostream>
using namespace std;
#define DEBUG
#define MIN(a,b) (((a)<(b)) ? a : b)
int main () {
int i, j;
i = 100;
j = 30;
#ifdef DEBUG
cerr <<"Trace: Inside main function" << endl;
#endif
#if 0
/* This is commented part */
cout << MKSTR(HELLO C++) << endl;
#endif
cout <<"The minimum is " << MIN(i, j) << endl;
#ifdef DEBUG
cerr <<"Trace: Coming out of main function" << endl;
#endif
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が生成されます-
The minimum is 30
Trace: Inside main function
Trace: Coming out of main function
#および##演算子
#および##プリプロセッサ演算子はC ++およびANSI / ISO Cで使用できます。#演算子を使用すると、置換テキストトークンが引用符で囲まれた文字列に変換されます。
次のマクロ定義を検討してください-
#include <iostream>
using namespace std;
#define MKSTR( x ) #x
int main () {
cout << MKSTR(HELLO C++) << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が生成されます-
HELLO C++
それがどのように機能したか見てみましょう。C ++プリプロセッサが次の行を変えることを理解するのは簡単です-
cout << MKSTR(HELLO C++) << endl;
上記の行は次の行になります-
cout << "HELLO C++" << endl;
##演算子は、2つのトークンを連結するために使用されます。ここに例があります-
#define CONCAT( x, y ) x ## y
CONCATがプログラムに表示されると、その引数が連結され、マクロを置き換えるために使用されます。たとえば、CONCAT(HELLO、C ++)は、プログラム内で次のように「HELLO C ++」に置き換えられます。
#include <iostream>
using namespace std;
#define concat(a, b) a ## b
int main() {
int xy = 100;
cout << concat(x, y);
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が生成されます-
100
それがどのように機能したか見てみましょう。C ++プリプロセッサが変換することを理解するのは簡単です-
cout << concat(x, y);
上記の行は次の行に変換されます-
cout << xy;
事前定義されたC ++マクロ
C ++は、以下に説明するいくつかの事前定義されたマクロを提供します-
シニア番号 | マクロと説明 |
---|---|
1 | __LINE__ これには、コンパイル時のプログラムの現在の行番号が含まれます。 |
2 | __FILE__ これには、コンパイル時のプログラムの現在のファイル名が含まれます。 |
3 | __DATE__ これには、ソースファイルがオブジェクトコードに変換された日付であるmonth / day / yearの形式の文字列が含まれます。 |
4 | __TIME__ これには、プログラムがコンパイルされた時刻である時間:分:秒の形式の文字列が含まれます。 |
上記のすべてのマクロの例を見てみましょう-
#include <iostream>
using namespace std;
int main () {
cout << "Value of __LINE__ : " << __LINE__ << endl;
cout << "Value of __FILE__ : " << __FILE__ << endl;
cout << "Value of __DATE__ : " << __DATE__ << endl;
cout << "Value of __TIME__ : " << __TIME__ << endl;
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が生成されます-
Value of __LINE__ : 6
Value of __FILE__ : test.cpp
Value of __DATE__ : Feb 28 2011
Value of __TIME__ : 18:52:48
シグナルは、オペレーティングシステムによってプロセスに配信される割り込みであり、プログラムを途中で終了させる可能性があります。UNIX、LINUX、Mac OS X、またはWindowsシステムでCtrl + Cを押すと、割り込みを生成できます。
プログラムでキャッチできないシグナルがありますが、プログラムでキャッチでき、シグナルに基づいて適切なアクションを実行できるシグナルのリストは次のとおりです。これらのシグナルは、C ++ヘッダーファイル<csignal>で定義されています。
シニア番号 | 信号と説明 |
---|---|
1 | SIGABRT の呼び出しなど、プログラムの異常終了 abort。 |
2 | SIGFPE ゼロ除算やオーバーフローを引き起こす演算など、誤った算術演算。 |
3 | SIGILL 不正な命令の検出。 |
4 | SIGINT インタラクティブな注意信号の受信。 |
5 | SIGSEGV ストレージへの無効なアクセス。 |
6 | SIGTERM プログラムに送信された終了要求。 |
signal()関数
C ++シグナル処理ライブラリは機能を提供します signal予期しないイベントをトラップします。以下はsignal()関数の構文です-
void (*signal (int sig, void (*func)(int)))(int);
簡単に言うと、この関数は2つの引数を受け取ります。最初の引数は信号番号を表す整数として、2番目の引数は信号処理関数へのポインターとして受け取ります。
signal()関数を使用してSIGINTシグナルをキャッチする簡単なC ++プログラムを書いてみましょう。プログラムでキャッチしたいシグナルが何であれ、を使用してそのシグナルを登録する必要があります。signal関数を作成し、シグナルハンドラーに関連付けます。次の例を調べてください-
#include <iostream>
#include <csignal>
using namespace std;
void signalHandler( int signum ) {
cout << "Interrupt signal (" << signum << ") received.\n";
// cleanup and close up stuff here
// terminate program
exit(signum);
}
int main () {
// register signal SIGINT and signal handler
signal(SIGINT, signalHandler);
while(1) {
cout << "Going to sleep...." << endl;
sleep(1);
}
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
Going to sleep....
Going to sleep....
Going to sleep....
ここで、Ctrl + cを押してプログラムを中断すると、プログラムが信号をキャッチし、次のように出力することで出力されることがわかります。
Going to sleep....
Going to sleep....
Going to sleep....
Interrupt signal (2) received.
raise()関数
機能別に信号を生成できます raise()、これは引数として整数の信号番号を取り、次の構文を持ちます。
int raise (signal sig);
ここに、 sigは、SIGINT、SIGABRT、SIGFPE、SIGILL、SIGSEGV、SIGTERM、SIGHUPのいずれかのシグナルを送信するためのシグナル番号です。以下は、次のように、raise()関数を使用して内部的にシグナルを発生させる例です。
#include <iostream>
#include <csignal>
using namespace std;
void signalHandler( int signum ) {
cout << "Interrupt signal (" << signum << ") received.\n";
// cleanup and close up stuff here
// terminate program
exit(signum);
}
int main () {
int i = 0;
// register signal SIGINT and signal handler
signal(SIGINT, signalHandler);
while(++i) {
cout << "Going to sleep...." << endl;
if( i == 3 ) {
raise( SIGINT);
}
sleep(1);
}
return 0;
}
上記のコードをコンパイルして実行すると、次の結果が生成され、自動的に出力されます-
Going to sleep....
Going to sleep....
Going to sleep....
Interrupt signal (2) received.
マルチスレッドはマルチタスクの特殊な形式であり、マルチタスクはコンピューターが2つ以上のプログラムを同時に実行できるようにする機能です。一般に、マルチタスクには、プロセスベースとスレッドベースの2種類があります。
プロセスベースのマルチタスクは、プログラムの同時実行を処理します。スレッドベースのマルチタスクは、同じプログラムの一部を同時に実行します。
マルチスレッドプログラムには、同時に実行できる2つ以上の部分が含まれています。このようなプログラムの各部分はスレッドと呼ばれ、各スレッドは個別の実行パスを定義します。
C ++には、マルチスレッドアプリケーションの組み込みサポートは含まれていません。代わりに、この機能を提供するためにオペレーティングシステムに完全に依存しています。
このチュートリアルは、Linux OSで作業していて、POSIXを使用してマルチスレッドC ++プログラムを作成することを前提としています。POSIXスレッド(Pthreads)は、FreeBSD、NetBSD、GNU / Linux、Mac OS X、Solarisなどの多くのUnixライクなPOSIXシステムで利用できるAPIを提供します。
スレッドの作成
次のルーチンは、POSIXスレッドを作成するために使用されます-
#include <pthread.h>
pthread_create (thread, attr, start_routine, arg)
ここに、 pthread_create新しいスレッドを作成し、実行可能にします。このルーチンは、コード内のどこからでも何度でも呼び出すことができます。パラメータの説明は次のとおりです-
シニア番号 | パラメータと説明 |
---|---|
1 | thread サブルーチンによって返される新しいスレッドの不透明で一意の識別子。 |
2 | attr スレッド属性の設定に使用できる不透明な属性オブジェクト。スレッド属性オブジェクトを指定するか、デフォルト値にNULLを指定できます。 |
3 | start_routine スレッドが作成されると実行されるC ++ルーチン。 |
4 | arg start_routineに渡すことができる単一の引数。void型のポインタキャストとして参照で渡す必要があります。引数を渡さない場合は、NULLを使用できます。 |
プロセスによって作成される可能性のあるスレッドの最大数は、実装によって異なります。作成されるスレッドはピアであり、他のスレッドを作成する場合があります。スレッド間に暗黙の階層や依存関係はありません。
スレッドの終了
POSIXスレッドを終了するために使用する次のルーチンがあります-
#include <pthread.h>
pthread_exit (status)
ここに pthread_exitスレッドを明示的に終了するために使用されます。通常、pthread_exit()ルーチンは、スレッドがその作業を完了し、存在する必要がなくなった後に呼び出されます。
main()が作成したスレッドの前に終了し、pthread_exit()で終了した場合、他のスレッドは引き続き実行されます。それ以外の場合は、main()が終了すると自動的に終了します。
Example
この単純なサンプルコードは、pthread_create()ルーチンを使用して5つのスレッドを作成します。各スレッドは「HelloWorld!」を印刷します。メッセージが表示され、pthread_exit()の呼び出しで終了します。
#include <iostream>
#include <cstdlib>
#include <pthread.h>
using namespace std;
#define NUM_THREADS 5
void *PrintHello(void *threadid) {
long tid;
tid = (long)threadid;
cout << "Hello World! Thread ID, " << tid << endl;
pthread_exit(NULL);
}
int main () {
pthread_t threads[NUM_THREADS];
int rc;
int i;
for( i = 0; i < NUM_THREADS; i++ ) {
cout << "main() : creating thread, " << i << endl;
rc = pthread_create(&threads[i], NULL, PrintHello, (void *)i);
if (rc) {
cout << "Error:unable to create thread," << rc << endl;
exit(-1);
}
}
pthread_exit(NULL);
}
次のように-lpthreadライブラリを使用して次のプログラムをコンパイルします-
$gcc test.cpp -lpthread
ここで、次の出力を与えるプログラムを実行します-
main() : creating thread, 0
main() : creating thread, 1
main() : creating thread, 2
main() : creating thread, 3
main() : creating thread, 4
Hello World! Thread ID, 0
Hello World! Thread ID, 1
Hello World! Thread ID, 2
Hello World! Thread ID, 3
Hello World! Thread ID, 4
スレッドへの引数の受け渡し
この例は、構造体を介して複数の引数を渡す方法を示しています。次の例で説明するように、voidを指すため、スレッドコールバックで任意のデータ型を渡すことができます。
#include <iostream>
#include <cstdlib>
#include <pthread.h>
using namespace std;
#define NUM_THREADS 5
struct thread_data {
int thread_id;
char *message;
};
void *PrintHello(void *threadarg) {
struct thread_data *my_data;
my_data = (struct thread_data *) threadarg;
cout << "Thread ID : " << my_data->thread_id ;
cout << " Message : " << my_data->message << endl;
pthread_exit(NULL);
}
int main () {
pthread_t threads[NUM_THREADS];
struct thread_data td[NUM_THREADS];
int rc;
int i;
for( i = 0; i < NUM_THREADS; i++ ) {
cout <<"main() : creating thread, " << i << endl;
td[i].thread_id = i;
td[i].message = "This is message";
rc = pthread_create(&threads[i], NULL, PrintHello, (void *)&td[i]);
if (rc) {
cout << "Error:unable to create thread," << rc << endl;
exit(-1);
}
}
pthread_exit(NULL);
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
main() : creating thread, 0
main() : creating thread, 1
main() : creating thread, 2
main() : creating thread, 3
main() : creating thread, 4
Thread ID : 3 Message : This is message
Thread ID : 2 Message : This is message
Thread ID : 0 Message : This is message
Thread ID : 1 Message : This is message
Thread ID : 4 Message : This is message
スレッドの結合と切り離し
スレッドを結合または切り離すために使用できる次の2つのルーチンがあります-
pthread_join (threadid, status)
pthread_detach (threadid)
pthread_join()サブルーチンは、指定された「threadid」スレッドが終了するまで、呼び出し元のスレッドをブロックします。スレッドが作成されると、その属性の1つが、スレッドが結合可能か分離可能かを定義します。参加可能として作成されたスレッドのみが参加できます。スレッドが切り離された状態で作成された場合、そのスレッドを結合することはできません。
この例は、Pthread結合ルーチンを使用してスレッドの完了を待機する方法を示しています。
#include <iostream>
#include <cstdlib>
#include <pthread.h>
#include <unistd.h>
using namespace std;
#define NUM_THREADS 5
void *wait(void *t) {
int i;
long tid;
tid = (long)t;
sleep(1);
cout << "Sleeping in thread " << endl;
cout << "Thread with id : " << tid << " ...exiting " << endl;
pthread_exit(NULL);
}
int main () {
int rc;
int i;
pthread_t threads[NUM_THREADS];
pthread_attr_t attr;
void *status;
// Initialize and set thread joinable
pthread_attr_init(&attr);
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);
for( i = 0; i < NUM_THREADS; i++ ) {
cout << "main() : creating thread, " << i << endl;
rc = pthread_create(&threads[i], &attr, wait, (void *)i );
if (rc) {
cout << "Error:unable to create thread," << rc << endl;
exit(-1);
}
}
// free attribute and wait for the other threads
pthread_attr_destroy(&attr);
for( i = 0; i < NUM_THREADS; i++ ) {
rc = pthread_join(threads[i], &status);
if (rc) {
cout << "Error:unable to join," << rc << endl;
exit(-1);
}
cout << "Main: completed thread id :" << i ;
cout << " exiting with status :" << status << endl;
}
cout << "Main: program exiting." << endl;
pthread_exit(NULL);
}
上記のコードをコンパイルして実行すると、次の結果が得られます。
main() : creating thread, 0
main() : creating thread, 1
main() : creating thread, 2
main() : creating thread, 3
main() : creating thread, 4
Sleeping in thread
Thread with id : 0 .... exiting
Sleeping in thread
Thread with id : 1 .... exiting
Sleeping in thread
Thread with id : 2 .... exiting
Sleeping in thread
Thread with id : 3 .... exiting
Sleeping in thread
Thread with id : 4 .... exiting
Main: completed thread id :0 exiting with status :0
Main: completed thread id :1 exiting with status :0
Main: completed thread id :2 exiting with status :0
Main: completed thread id :3 exiting with status :0
Main: completed thread id :4 exiting with status :0
Main: program exiting.
CGIとは何ですか?
Common Gateway Interface(CGI)は、Webサーバーとカスタムスクリプトの間で情報を交換する方法を定義する一連の標準です。
CGI仕様は現在NCSAによって維持されており、NCSAはCGIを次のように定義しています。
Common Gateway Interface(CGI)は、外部ゲートウェイプログラムがHTTPサーバーなどの情報サーバーとインターフェイスするための標準です。
現在のバージョンはCGI / 1.1であり、CGI /1.2は進行中です。
Webブラウジング
CGIの概念を理解するために、ハイパーリンクをクリックして特定のWebページまたはURLを参照するとどうなるかを見てみましょう。
ブラウザがHTTPWebサーバーに接続し、URLを要求します。ファイル名。
WebサーバーはURLを解析し、ファイル名を探します。要求されたファイルが見つかった場合、Webサーバーはそのファイルをブラウザーに送り返します。それ以外の場合は、間違ったファイルを要求したことを示すエラーメッセージを送信します。
WebブラウザはWebサーバーから応答を受け取り、受信したファイルまたは受信した応答に基づいてエラーメッセージを表示します。
ただし、特定のディレクトリ内のファイルが要求されたときにそのファイルが返送されないようにHTTPサーバーを設定することは可能です。代わりに、プログラムとして実行され、プログラムから生成された出力がブラウザに返送されて表示されます。
Common Gateway Interface(CGI)は、アプリケーション(CGIプログラムまたはCGIスクリプトと呼ばれる)がWebサーバーおよびクライアントと対話できるようにするための標準プロトコルです。これらのCGIプログラムは、Python、PERL、シェル、C、C ++などで記述できます。
CGIアーキテクチャ図
次の簡単なプログラムは、CGIの簡単なアーキテクチャを示しています。
Webサーバーの構成
CGIプログラミングを続行する前に、WebサーバーがCGIをサポートし、CGIプログラムを処理するように構成されていることを確認してください。HTTPサーバーによって実行されるすべてのCGIプログラムは、事前設定されたディレクトリに保存されます。このディレクトリはCGIディレクトリと呼ばれ、慣例により/ var / www / cgi-binという名前が付けられています。慣例により、CGIファイルの拡張子は次のようになります。.cgi、C ++実行可能ファイルですが。
デフォルトでは、ApacheWebサーバーは/ var / www / cgi-binでCGIプログラムを実行するように構成されています。CGIスクリプトを実行するために他のディレクトリを指定する場合は、httpd.confファイルの次のセクションを変更できます。
<Directory "/var/www/cgi-bin">
AllowOverride None
Options ExecCGI
Order allow,deny
Allow from all
</Directory>
<Directory "/var/www/cgi-bin">
Options All
</Directory>
ここでは、Webサーバーが正常に稼働しており、PerlやShellなどの他のCGIプログラムを実行できることを前提としています。
最初のCGIプログラム
次のC ++プログラムの内容を検討してください-
#include <iostream>
using namespace std;
int main () {
cout << "Content-type:text/html\r\n\r\n";
cout << "<html>\n";
cout << "<head>\n";
cout << "<title>Hello World - First CGI Program</title>\n";
cout << "</head>\n";
cout << "<body>\n";
cout << "<h2>Hello World! This is my first CGI program</h2>\n";
cout << "</body>\n";
cout << "</html>\n";
return 0;
}
上記のコードをコンパイルし、実行可能ファイルにcplusplus.cgiという名前を付けます。このファイルは/ var / www / cgi-binディレクトリに保存されており、次の内容が含まれています。CGIプログラムを実行する前に、を使用してファイルのモードを変更していることを確認してくださいchmod 755 cplusplus.cgi ファイルを実行可能にするUNIXコマンド。
私の最初のCGIプログラム
上記のC ++プログラムは、STDOUTファイル(画面)に出力を書き込んでいる単純なプログラムです。利用可能な重要で追加の機能が1つあります。それは、最初の行の印刷です。Content-type:text/html\r\n\r\n。この行はブラウザに返送され、ブラウザ画面に表示されるコンテンツタイプを指定します。これで、CGIの基本概念を理解し、Pythonを使用して多くの複雑なCGIプログラムを作成できるようになりました。C ++ CGIプログラムは、RDBMSなどの他の外部システムと対話して、情報を交換できます。
HTTPヘッダー
この線 Content-type:text/html\r\n\r\nはHTTPヘッダーの一部であり、コンテンツを理解するためにブラウザに送信されます。すべてのHTTPヘッダーは次の形式になります-
HTTP Field Name: Field Content
For Example
Content-type: text/html\r\n\r\n
CGIプログラミングで頻繁に使用する重要なHTTPヘッダーは他にもいくつかあります。
シニア番号 | ヘッダーと説明 |
---|---|
1 | Content-type: 返されるファイルの形式を定義するMIME文字列。例はContent-type:text / htmlです。 |
2 | Expires: Date 情報が無効になる日付。これは、ページを更新する必要がある時期を決定するためにブラウザが使用する必要があります。有効な日付文字列は、1998年1月1日12:00:00GMTの形式である必要があります。 |
3 | Location: URL 要求されたURLの代わりに返されるURL。このファイルを使用して、リクエストを任意のファイルにリダイレクトできます。 |
4 | Last-modified: Date リソースが最後に変更された日付。 |
5 | Content-length: N 返されるデータの長さ(バイト単位)。ブラウザはこの値を使用して、ファイルの推定ダウンロード時間を報告します。 |
6 | Set-Cookie: String 文字列を介して渡されるCookieを設定します。 |
CGI環境変数
すべてのCGIプログラムは、次の環境変数にアクセスできます。これらの変数は、CGIプログラムを作成する際に重要な役割を果たします。
シニア番号 | 変数名と説明 |
---|---|
1 | CONTENT_TYPE クライアントが添付コンテンツをサーバーに送信するときに使用されるコンテンツのデータ型。たとえば、ファイルのアップロードなど。 |
2 | CONTENT_LENGTH POST要求でのみ使用可能な照会情報の長さ。 |
3 | HTTP_COOKIE 設定されたCookieをキーと値のペアの形式で返します。 |
4 | HTTP_USER_AGENT User-Agent request-headerフィールドには、リクエストを発信したユーザーエージェントに関する情報が含まれています。Webブラウザの名前です。 |
5 | PATH_INFO CGIスクリプトのパス。 |
6 | QUERY_STRING GETメソッドリクエストで送信されるURLエンコードされた情報。 |
7 | REMOTE_ADDR リクエストを行っているリモートホストのIPアドレス。これは、ロギングまたは認証の目的で役立ちます。 |
8 | REMOTE_HOST 要求を行っているホストの完全修飾名。この情報が利用できない場合は、REMOTE_ADDRを使用してIRアドレスを取得できます。 |
9 | REQUEST_METHOD リクエストの作成に使用されたメソッド。最も一般的なメソッドはGETとPOSTです。 |
10 | SCRIPT_FILENAME CGIスクリプトへのフルパス。 |
11 | SCRIPT_NAME CGIスクリプトの名前。 |
12 | SERVER_NAME サーバーのホスト名またはIPアドレス。 |
13 | SERVER_SOFTWARE サーバーが実行しているソフトウェアの名前とバージョン。 |
これは、すべてのCGI変数を一覧表示する小さなCGIプログラムです。
#include <iostream>
#include <stdlib.h>
using namespace std;
const string ENV[ 24 ] = {
"COMSPEC", "DOCUMENT_ROOT", "GATEWAY_INTERFACE",
"HTTP_ACCEPT", "HTTP_ACCEPT_ENCODING",
"HTTP_ACCEPT_LANGUAGE", "HTTP_CONNECTION",
"HTTP_HOST", "HTTP_USER_AGENT", "PATH",
"QUERY_STRING", "REMOTE_ADDR", "REMOTE_PORT",
"REQUEST_METHOD", "REQUEST_URI", "SCRIPT_FILENAME",
"SCRIPT_NAME", "SERVER_ADDR", "SERVER_ADMIN",
"SERVER_NAME","SERVER_PORT","SERVER_PROTOCOL",
"SERVER_SIGNATURE","SERVER_SOFTWARE" };
int main () {
cout << "Content-type:text/html\r\n\r\n";
cout << "<html>\n";
cout << "<head>\n";
cout << "<title>CGI Environment Variables</title>\n";
cout << "</head>\n";
cout << "<body>\n";
cout << "<table border = \"0\" cellspacing = \"2\">";
for ( int i = 0; i < 24; i++ ) {
cout << "<tr><td>" << ENV[ i ] << "</td><td>";
// attempt to retrieve value of environment variable
char *value = getenv( ENV[ i ].c_str() );
if ( value != 0 ) {
cout << value;
} else {
cout << "Environment variable does not exist.";
}
cout << "</td></tr>\n";
}
cout << "</table><\n";
cout << "</body>\n";
cout << "</html>\n";
return 0;
}
C ++ CGIライブラリ
実際の例では、CGIプログラムで多くの操作を行う必要があります。C ++プログラム用に作成されたCGIライブラリがあり、ftp://ftp.gnu.org/gnu/cgicc/からダウンロードして、手順に従ってライブラリをインストールできます。
$tar xzf cgicc-X.X.X.tar.gz
$cd cgicc-X.X.X/ $./configure --prefix=/usr
$make $make install
'C ++ CGI LibDocumentationで入手可能な関連ドキュメントを確認できます。
GETメソッドとPOSTメソッド
ブラウザからWebサーバーに、そして最終的にはCGIプログラムに情報を渡す必要がある場合、多くの状況に遭遇したに違いありません。ほとんどの場合、ブラウザは2つの方法を使用してこの情報をWebサーバーに渡します。これらのメソッドは、GETメソッドとPOSTメソッドです。
GETメソッドを使用した情報の受け渡し
GETメソッドは、ページリクエストに追加されたエンコードされたユーザー情報を送信します。ページとエンコードされた情報は?で区切られます 次のような文字-
http://www.test.com/cgi-bin/cpp.cgi?key1=value1&key2=value2
GETメソッドは、ブラウザからWebサーバーに情報を渡すためのデフォルトのメソッドであり、ブラウザのLocation:boxに表示される長い文字列を生成します。サーバーに渡すパスワードやその他の機密情報がある場合は、GETメソッドを使用しないでください。GETメソッドにはサイズ制限があり、リクエスト文字列で最大1024文字を渡すことができます。
GETメソッドを使用する場合、情報はQUERY_STRING httpヘッダーを使用して渡され、QUERY_STRING環境変数を介してCGIプログラムでアクセスできます。
キーと値のペアを任意のURLと連結するだけで情報を渡すか、HTML <FORM>タグを使用してGETメソッドを使用して情報を渡すことができます。
単純なURLの例:Getメソッド
これは、GETメソッドを使用してhello_get.pyプログラムに2つの値を渡す単純なURLです。
/cgi-bin/cpp_get.cgi?first_name=ZARA&last_name=ALI以下は生成するプログラムです cpp_get.cgiWebブラウザからの入力を処理するCGIプログラム。渡された情報に非常に簡単にアクセスできるC ++ CGIライブラリを使用します-
#include <iostream>
#include <vector>
#include <string>
#include <stdio.h>
#include <stdlib.h>
#include <cgicc/CgiDefs.h>
#include <cgicc/Cgicc.h>
#include <cgicc/HTTPHTMLHeader.h>
#include <cgicc/HTMLClasses.h>
using namespace std;
using namespace cgicc;
int main () {
Cgicc formData;
cout << "Content-type:text/html\r\n\r\n";
cout << "<html>\n";
cout << "<head>\n";
cout << "<title>Using GET and POST Methods</title>\n";
cout << "</head>\n";
cout << "<body>\n";
form_iterator fi = formData.getElement("first_name");
if( !fi->isEmpty() && fi != (*formData).end()) {
cout << "First name: " << **fi << endl;
} else {
cout << "No text entered for first name" << endl;
}
cout << "<br/>\n";
fi = formData.getElement("last_name");
if( !fi->isEmpty() &&fi != (*formData).end()) {
cout << "Last name: " << **fi << endl;
} else {
cout << "No text entered for last name" << endl;
}
cout << "<br/>\n";
cout << "</body>\n";
cout << "</html>\n";
return 0;
}
ここで、上記のプログラムを次のようにコンパイルします-
$g++ -o cpp_get.cgi cpp_get.cpp -lcgicc
cpp_get.cgiを生成してCGIディレクトリに置き、次のリンクを使用してアクセスしてみてください-
/cgi-bin/cpp_get.cgi?first_name=ZARA&last_name=ALIこれにより、次の結果が生成されます-
First name: ZARA
Last name: ALI
単純なFORMの例:GETメソッド
これは、HTMLFORMと送信ボタンを使用して2つの値を渡す簡単な例です。同じCGIスクリプトcpp_get.cgiを使用してこの入力を処理します。
<form action = "/cgi-bin/cpp_get.cgi" method = "get">
First Name: <input type = "text" name = "first_name"> <br />
Last Name: <input type = "text" name = "last_name" />
<input type = "submit" value = "Submit" />
</form>
上記のフォームの実際の出力は次のとおりです。姓名を入力し、送信ボタンをクリックして結果を確認します。
POSTメソッドを使用した情報の受け渡し
CGIプログラムに情報を渡す一般的により信頼性の高い方法はPOST方法です。これは、GETメソッドとまったく同じ方法で情報をパッケージ化しますが、?の後にテキスト文字列として送信する代わりに URLでは、別のメッセージとして送信します。このメッセージは、標準入力の形式でCGIスクリプトに送られます。
同じcpp_get.cgiプログラムがPOSTメソッドも処理します。上記と同じ例を見てみましょう。HTMLFORMと送信ボタンを使用して2つの値を渡しますが、今回は次のようにPOSTメソッドを使用します。
<form action = "/cgi-bin/cpp_get.cgi" method = "post">
First Name: <input type = "text" name = "first_name"><br />
Last Name: <input type = "text" name = "last_name" />
<input type = "submit" value = "Submit" />
</form>
上記のフォームの実際の出力は次のとおりです。姓名を入力し、送信ボタンをクリックして結果を確認します。
チェックボックスデータをCGIプログラムに渡す
チェックボックスは、複数のオプションを選択する必要がある場合に使用されます。
これは、2つのチェックボックスがあるフォームのHTMLコードの例です-
<form action = "/cgi-bin/cpp_checkbox.cgi" method = "POST" target = "_blank">
<input type = "checkbox" name = "maths" value = "on" /> Maths
<input type = "checkbox" name = "physics" value = "on" /> Physics
<input type = "submit" value = "Select Subject" />
</form>
このコードの結果は次の形式になります-
以下はC ++プログラムで、チェックボックスボタンを介してWebブラウザからの入力を処理するcpp_checkbox.cgiスクリプトを生成します。
#include <iostream>
#include <vector>
#include <string>
#include <stdio.h>
#include <stdlib.h>
#include <cgicc/CgiDefs.h>
#include <cgicc/Cgicc.h>
#include <cgicc/HTTPHTMLHeader.h>
#include <cgicc/HTMLClasses.h>
using namespace std;
using namespace cgicc;
int main () {
Cgicc formData;
bool maths_flag, physics_flag;
cout << "Content-type:text/html\r\n\r\n";
cout << "<html>\n";
cout << "<head>\n";
cout << "<title>Checkbox Data to CGI</title>\n";
cout << "</head>\n";
cout << "<body>\n";
maths_flag = formData.queryCheckbox("maths");
if( maths_flag ) {
cout << "Maths Flag: ON " << endl;
} else {
cout << "Maths Flag: OFF " << endl;
}
cout << "<br/>\n";
physics_flag = formData.queryCheckbox("physics");
if( physics_flag ) {
cout << "Physics Flag: ON " << endl;
} else {
cout << "Physics Flag: OFF " << endl;
}
cout << "<br/>\n";
cout << "</body>\n";
cout << "</html>\n";
return 0;
}
ラジオボタンデータをCGIプログラムに渡す
ラジオボタンは、1つのオプションのみを選択する必要がある場合に使用されます。
これは、2つのラジオボタンがあるフォームのHTMLコードの例です-
<form action = "/cgi-bin/cpp_radiobutton.cgi" method = "post" target = "_blank">
<input type = "radio" name = "subject" value = "maths" checked = "checked"/> Maths
<input type = "radio" name = "subject" value = "physics" /> Physics
<input type = "submit" value = "Select Subject" />
</form>
このコードの結果は次の形式になります-
以下はC ++プログラムで、ラジオボタンを介してWebブラウザからの入力を処理するcpp_radiobutton.cgiスクリプトを生成します。
#include <iostream>
#include <vector>
#include <string>
#include <stdio.h>
#include <stdlib.h>
#include <cgicc/CgiDefs.h>
#include <cgicc/Cgicc.h>
#include <cgicc/HTTPHTMLHeader.h>
#include <cgicc/HTMLClasses.h>
using namespace std;
using namespace cgicc;
int main () {
Cgicc formData;
cout << "Content-type:text/html\r\n\r\n";
cout << "<html>\n";
cout << "<head>\n";
cout << "<title>Radio Button Data to CGI</title>\n";
cout << "</head>\n";
cout << "<body>\n";
form_iterator fi = formData.getElement("subject");
if( !fi->isEmpty() && fi != (*formData).end()) {
cout << "Radio box selected: " << **fi << endl;
}
cout << "<br/>\n";
cout << "</body>\n";
cout << "</html>\n";
return 0;
}
テキストエリアデータをCGIプログラムに渡す
TEXTAREA要素は、複数行のテキストをCGIプログラムに渡す必要がある場合に使用されます。
TEXTAREAボックスのあるフォームのHTMLコードの例を次に示します-
<form action = "/cgi-bin/cpp_textarea.cgi" method = "post" target = "_blank">
<textarea name = "textcontent" cols = "40" rows = "4">
Type your text here...
</textarea>
<input type = "submit" value = "Submit" />
</form>
このコードの結果は次の形式になります-
以下はC ++プログラムで、テキスト領域を介してWebブラウザからの入力を処理するcpp_textarea.cgiスクリプトを生成します。
#include <iostream>
#include <vector>
#include <string>
#include <stdio.h>
#include <stdlib.h>
#include <cgicc/CgiDefs.h>
#include <cgicc/Cgicc.h>
#include <cgicc/HTTPHTMLHeader.h>
#include <cgicc/HTMLClasses.h>
using namespace std;
using namespace cgicc;
int main () {
Cgicc formData;
cout << "Content-type:text/html\r\n\r\n";
cout << "<html>\n";
cout << "<head>\n";
cout << "<title>Text Area Data to CGI</title>\n";
cout << "</head>\n";
cout << "<body>\n";
form_iterator fi = formData.getElement("textcontent");
if( !fi->isEmpty() && fi != (*formData).end()) {
cout << "Text Content: " << **fi << endl;
} else {
cout << "No text entered" << endl;
}
cout << "<br/>\n";
cout << "</body>\n";
cout << "</html>\n";
return 0;
}
ドロップダウンボックスデータをCGIプログラムに渡す
ドロップダウンボックスは、利用可能なオプションが多数あるが、1つまたは2つだけが選択される場合に使用されます。
ドロップダウンボックスが1つあるフォームのHTMLコードの例を次に示します-
<form action = "/cgi-bin/cpp_dropdown.cgi" method = "post" target = "_blank">
<select name = "dropdown">
<option value = "Maths" selected>Maths</option>
<option value = "Physics">Physics</option>
</select>
<input type = "submit" value = "Submit"/>
</form>
このコードの結果は次の形式になります-
以下はC ++プログラムで、ドロップダウンボックスを介してWebブラウザからの入力を処理するcpp_dropdown.cgiスクリプトを生成します。
#include <iostream>
#include <vector>
#include <string>
#include <stdio.h>
#include <stdlib.h>
#include <cgicc/CgiDefs.h>
#include <cgicc/Cgicc.h>
#include <cgicc/HTTPHTMLHeader.h>
#include <cgicc/HTMLClasses.h>
using namespace std;
using namespace cgicc;
int main () {
Cgicc formData;
cout << "Content-type:text/html\r\n\r\n";
cout << "<html>\n";
cout << "<head>\n";
cout << "<title>Drop Down Box Data to CGI</title>\n";
cout << "</head>\n";
cout << "<body>\n";
form_iterator fi = formData.getElement("dropdown");
if( !fi->isEmpty() && fi != (*formData).end()) {
cout << "Value Selected: " << **fi << endl;
}
cout << "<br/>\n";
cout << "</body>\n";
cout << "</html>\n";
return 0;
}
CGIでのCookieの使用
HTTPプロトコルはステートレスプロトコルです。ただし、商用Webサイトの場合、異なるページ間でセッション情報を維持する必要があります。たとえば、1人のユーザー登録は、多くのページを完了した後に終了します。しかし、すべてのWebページにわたってユーザーのセッション情報を維持する方法。
多くの場合、Cookieを使用することは、訪問者のエクスペリエンスやサイトの統計を向上させるために必要な設定、購入、手数料、およびその他の情報を記憶および追跡するための最も効率的な方法です。
使い方
サーバーは、Cookieの形式で訪問者のブラウザにデータを送信します。ブラウザはCookieを受け入れる場合があります。含まれている場合は、訪問者のハードドライブにプレーンテキストレコードとして保存されます。これで、訪問者がサイトの別のページにアクセスすると、Cookieを取得できるようになります。取得されると、サーバーは何が保存されたかを認識/記憶します。
Cookieは、5つの可変長フィールドのプレーンテキストデータレコードです-
Expires−これはCookieの有効期限が切れる日付を示します。これが空白の場合、訪問者がブラウザを終了するとCookieは期限切れになります。
Domain −これはあなたのサイトのドメイン名を示しています。
Path−これは、Cookieを設定したディレクトリまたはWebページへのパスを示します。任意のディレクトリまたはページからCookieを取得する場合は、これを空白にすることができます。
Secure−このフィールドに「セキュア」という単語が含まれている場合、Cookieはセキュアサーバーでのみ取得できます。このフィールドが空白の場合、そのような制限はありません。
Name = Value − Cookieは、キーと値のペアの形式で設定および取得されます。
クッキーの設定
ブラウザにCookieを送信するのは非常に簡単です。これらのCookieは、Content-typeが提出される前にHTTPヘッダーとともに送信されます。ユーザーIDとパスワードをCookieとして設定するとします。したがって、Cookieの設定は次のように行われます。
#include <iostream>
using namespace std;
int main () {
cout << "Set-Cookie:UserID = XYZ;\r\n";
cout << "Set-Cookie:Password = XYZ123;\r\n";
cout << "Set-Cookie:Domain = www.tutorialspoint.com;\r\n";
cout << "Set-Cookie:Path = /perl;\n";
cout << "Content-type:text/html\r\n\r\n";
cout << "<html>\n";
cout << "<head>\n";
cout << "<title>Cookies in CGI</title>\n";
cout << "</head>\n";
cout << "<body>\n";
cout << "Setting cookies" << endl;
cout << "<br/>\n";
cout << "</body>\n";
cout << "</html>\n";
return 0;
}
この例から、Cookieの設定方法を理解している必要があります。を使用しておりますSet-Cookie Cookieを設定するためのHTTPヘッダー。
ここでは、Expires、Domain、PathなどのCookie属性を設定することはオプションです。マジックラインを送信する前にCookieが設定されていることは注目に値します"Content-type:text/html\r\n\r\n。
上記のプログラムをコンパイルしてsetcookies.cgiを生成し、次のリンクを使用してCookieを設定してみてください。それはあなたのコンピュータに4つのクッキーを設定します-
/cgi-bin/setcookies.cgi
クッキーの取得
設定されたすべてのCookieを簡単に取得できます。CookieはCGI環境変数HTTP_COOKIEに保存され、次の形式になります。
key1 = value1; key2 = value2; key3 = value3....
これは、Cookieを取得する方法の例です。
#include <iostream>
#include <vector>
#include <string>
#include <stdio.h>
#include <stdlib.h>
#include <cgicc/CgiDefs.h>
#include <cgicc/Cgicc.h>
#include <cgicc/HTTPHTMLHeader.h>
#include <cgicc/HTMLClasses.h>
using namespace std;
using namespace cgicc;
int main () {
Cgicc cgi;
const_cookie_iterator cci;
cout << "Content-type:text/html\r\n\r\n";
cout << "<html>\n";
cout << "<head>\n";
cout << "<title>Cookies in CGI</title>\n";
cout << "</head>\n";
cout << "<body>\n";
cout << "<table border = \"0\" cellspacing = \"2\">";
// get environment variables
const CgiEnvironment& env = cgi.getEnvironment();
for( cci = env.getCookieList().begin();
cci != env.getCookieList().end();
++cci ) {
cout << "<tr><td>" << cci->getName() << "</td><td>";
cout << cci->getValue();
cout << "</td></tr>\n";
}
cout << "</table><\n";
cout << "<br/>\n";
cout << "</body>\n";
cout << "</html>\n";
return 0;
}
次に、上記のプログラムをコンパイルしてgetcookies.cgiを生成し、コンピューターで使用可能なすべてのCookieのリストを取得してみます-
/cgi-bin/getcookies.cgi
これにより、前のセクションで設定された4つのCookieすべてと、コンピューターに設定された他のすべてのCookieのリストが作成されます。
UserID XYZ
Password XYZ123
Domain www.tutorialspoint.com
Path /perl
ファイルアップロードの例
ファイルをアップロードするには、HTMLフォームでenctype属性をに設定する必要があります multipart/form-data。ファイルタイプの入力タグは「参照」ボタンを作成します。
<html>
<body>
<form enctype = "multipart/form-data" action = "/cgi-bin/cpp_uploadfile.cgi"
method = "post">
<p>File: <input type = "file" name = "userfile" /></p>
<p><input type = "submit" value = "Upload" /></p>
</form>
</body>
</html>
このコードの結果は次の形式になります-
Note−上記の例は、サーバーへのファイルのアップロードを停止するために意図的に無効にされています。ただし、サーバーで上記のコードを試すことができます。
これがスクリプトです cpp_uploadfile.cpp ファイルのアップロードを処理する-
#include <iostream>
#include <vector>
#include <string>
#include <stdio.h>
#include <stdlib.h>
#include <cgicc/CgiDefs.h>
#include <cgicc/Cgicc.h>
#include <cgicc/HTTPHTMLHeader.h>
#include <cgicc/HTMLClasses.h>
using namespace std;
using namespace cgicc;
int main () {
Cgicc cgi;
cout << "Content-type:text/html\r\n\r\n";
cout << "<html>\n";
cout << "<head>\n";
cout << "<title>File Upload in CGI</title>\n";
cout << "</head>\n";
cout << "<body>\n";
// get list of files to be uploaded
const_file_iterator file = cgi.getFile("userfile");
if(file != cgi.getFiles().end()) {
// send data type at cout.
cout << HTTPContentHeader(file->getDataType());
// write content at cout.
file->writeToStream(cout);
}
cout << "<File uploaded successfully>\n";
cout << "</body>\n";
cout << "</html>\n";
return 0;
}
上記の例は、でコンテンツを書き込むためのものです cout ストリームですが、ファイルストリームを開いて、アップロードしたファイルのコンテンツをファイルの目的の場所に保存できます。
このチュートリアルを楽しんでいただけたでしょうか。はいの場合は、フィードバックをお送りください。