アフィリエイト広告を利用しています

広告

posted by fanblog

2017年10月05日

《その66》 返却値型が const char* の関数(p.315演習8-15)


本ブログ《その64》の 演習8-13 で作った関数は、
    const char* strchr_ptr(const char* s, char c);

そして、今回の 演習8-15 で作る関数は、
    const char* str_match(char* s1, char* s2);

両方とも返却値型に const が付いています。

 関数の仮引数の宣言のときに付ける const は、その値が関数内で書きかえられることがない
ということを保証します。
では、関数の返却値型に const を付ける意味は・・・。



 こんなストーリーを考えてみました(見当はずれの可能性ありです)。


#include <iostream>
using namespace std;

char* f(char* c) /* 受け取った文字を 'f' に変える関数です。
const が付いていません。*/
{
*c = 'f';
return c;
}

int main(void)
{
char character = 'a'; // 文字データ character があるとします。

char* p = f(&character); /* 文字データ character を関数 f で変更します。
変更した character は大切なので、返却値を
p に代入して、以後はこれを使うことにします。
*/

cout << *p << '\n'; // *p は確かに 'f' になっています。
cout << character << '\n'; /* 関数 f で変更した大切な character は、もちろん
'f' です。
*/

*p = 'p'; // *p を利用する過程で、 'p' に変更したとします。

cout << *p << '\n'; // *p は確かに 'p' になりました。
cout << character << '\n'; /* 大切な character は・・・。
'p' になってしまった! (゚ω゚;)
const を付けて const char* f(char* c) にして
おけばよかったのに
(ノ_-。) */
}

c08_1501.png

「ちがうだろ〜」という声が聞こえてきたような・・・。 まあ取りあえず、const を付ける意味ということで。

こんなことを考えてみたのは、きょうの 演習8-15 の課題が
    const char* str_match(char* s1, char*s2);
なもので、自分なりに const を付ける理由が欲しかったからなんです。



新版明解C++入門編 p.315 演習8-15
 文字列 s1 の中に含まれる最も先頭に位置する文字列 s2 の先頭文字へのポインタを返す関数
str_match を作成せよ。
    const char* str_match(char* s1, char* s2);
 たとえば、受け取った文字列 s1 が "abcabcdef" で s2 が "abcd" であれば、返却するのは &s1[3] の値である。
文字列 s2 が s1 に含まれない場合は NULL を返却すること。

// p315_演習8-15
#include <iostream>
using namespace std;

const char* str_match(char* s1, char* s2)
{
char* p1 = s1; char* p2 = s2;

int n1 = 0; while (*s1++) n1++; // n1 … s1 の文字数
int n2 = 0; while (*s2++) n2++; // n2 … s2 の文字数

int i, f;
for (i = 0; i <= n1 - n2; i++) { // s1 の1文字目から順に・・・
f = 1; // とりあえず f = 1 にしておいて・・・
for (int j = 0; j < n2; j++) // s2 が含まれているかどうか・・・
if (p1[i + j] != p2[j]) {
f = 0; break; // 違う文字があったら f = 0
}
if (f == 1) // f = 1 のままなら探索成功
break;
}

return f == 1 ? &p1[i] : NULL;
}

void disp(char* s1, char* s2, const char* p) // 結果表示用の関数
{
cout << "対象文字列 : " << s1 << '\n';
cout << "探索文字列 : " << s2 << '\n';
if (p)
cout << "探索文字列は対象文字列の " << p - s1 + 1
<< " 文字目にあります。" << '\n';
else
cout << "探索文字列は対象文字列に含まれていません。" << '\n';
}

int main(void)
{
char str1[] = "abcabcdef";
char str2[] = "abcd";
char str3[] = "abc";
char str4[] = "cdf";
char str5[] = "cdef";

const char* ptr2 = str_match(str1, str2);
disp(str1, str2, ptr2);

cout << '\n';
const char* ptr3 = str_match(str1, str3);
disp(str1, str3, ptr3);

cout << '\n';
const char* ptr4 = str_match(str1, str4);
disp(str1, str4, ptr4);

cout << '\n';
const char* ptr5 = str_match(str1, str5);
disp(str1, str5, ptr5);
}

c08_15.png


新版 明解C 入門編 (明解シリーズ)

新品価格
¥2,916から
(2017/8/30 21:02時点)









--
この記事へのコメント
コメントを書く

お名前:

メールアドレス:


ホームページアドレス:

コメント:

※ブログオーナーが承認したコメントのみ表示されます。

この記事へのトラックバックURL
https://fanblogs.jp/tb/6776412
※ブログオーナーが承認したトラックバックのみ表示されます。

この記事へのトラックバック

 たまに、クリック お願いします m(_ _)m

 AA にほんブログ村 IT技術ブログ C/C++へ

こうすけ:メール kousuke_cpp@outlook.jp

【1】★★C++ 記事目次★★ ← 利用可能です。
・新版明解C++入門編 / 新版明解C++中級編
・その他 C++ 関連記事

【2】★★こうすけ@C#★★
・C# の初歩的な記事


検索
<< 2018年08月 >>
      1 2 3 4
5 6 7 8 9 10 11
12 13 14 15 16 17 18
19 20 21 22 23 24 25
26 27 28 29 30 31  
プロフィール
こうすけさんの画像
こうすけ

 たまに、クリック お願いします m(_ _)m

 AA にほんブログ村 IT技術ブログ C/C++へ

こうすけ:メール kousuke_cpp@outlook.jp

【1】★★C++ 記事目次★★ ← 利用可能です。
・新版明解C++入門編 / 新版明解C++中級編
・その他 C++ 関連記事

【2】★★こうすけ@C#★★
・C# の初歩的な記事


×

この広告は30日以上新しい記事の更新がないブログに表示されております。