C言語には大文字小文字を区別せずに文字列を比較する関数やマクロは存在しません.したがってプログラマが自分でこの機能を実装する必要があります.
実装方法としては,文字列を構成する文字を 1 文字ずつ大文字に変換して比較を行っていけば OK です.以下に実装例を示します.
/**
* 文字列を大文字小文字を区別せずに比較する
* @param[in] s1 文字列1
* @param[in] s2 文字列2
* @retval 0 s1とs2が同じ文字列であるとき
* @retval 正の整数 s1 > s2であるとき
* @retval 負の整数 s2 < s2であるとき
*/
int strcmp_ignorecase(const char *s1, const char *s2) {
int i = 0;
/* 文字が等しい間繰り返す */
while (toupper((unsigned char)s1[i]) == toupper((unsigned char)s2[i])) {
if (s1[i] == '\0') {
return 0;
}
i++;
}
return toupper((unsigned char)s1[i]) - toupper((unsigned char)s2[i]);
}
なお,上記の例では文字を大文字に変換するために ctype.h の toupper 関数を使用しています.これにつきましては,アルファベットの小文字を大文字に変換する をご覧下さい.
以下に,大文字小文字を区別せずに文字列を比較する関数の実装例を示します.
/* header files */
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
/* macros */
#define N 256
/* functions */
int strcmp_ignorecase(const char *, const char *);
int strncmp_ignorecase(const char *, const char *, size_t);
/* main */
int main(void) {
char s1[N] = {'\0'};
char s2[N] = {'\0'};
/* 入力 */
printf("文字列1を入力してください.\n> ");
scanf("%s", s1);
printf("文字列2を入力してください.\n> ");
scanf("%s", s2);
puts("全文字比較");
if ( strcmp_ignorecase(s1, s2) == 0 ) {
printf("%sと%sは同じ文字列です.\n", s1, s2);
} else {
printf("%sと%sは異なる文字列です.\n", s1, s2);
}
puts("先頭の4文字分比較");
if ( strncmp_ignorecase(s1, s2, 4) == 0 ) {
printf("%sと%sは同じ文字列です.\n", s1, s2);
} else {
printf("%sと%sは異なる文字列です.\n", s1, s2);
}
return EXIT_SUCCESS;
}
/**
* 文字列を大文字小文字を区別せずに比較する
* @param[in] s1 文字列1
* @param[in] s2 文字列2
* @retval 0 s1とs2が同じ文字列であるとき
* @retval 正の整数 s1 > s2であるとき
* @retval 負の整数 s2 < s2であるとき
*/
int strcmp_ignorecase(const char *s1, const char *s2) {
int i = 0;
/* 文字が等しい間繰り返す */
while (toupper((unsigned char)s1[i]) == toupper((unsigned char)s2[i])) {
if (s1[i] == '\0') {
return 0;
}
i++;
}
return toupper((unsigned char)s1[i]) - toupper((unsigned char)s2[i]);
}
/**
* 大文字小文字を区別せずに文字列をn文字分比較する
* @param[IN] s1 文字列1
* @param[IN] s2 文字列2
* @param[IN] n 比較する文字数
* @retval 0 s1とs2が同じ文字列であるとき
* @retval 正の整数 s1 > s2であるとき
* @retval 負の整数 s2 < s2であるとき
*/
int strncmp_ignorecase(const char *s1, const char *s2, size_t n) {
int i = 0;
/* 文字が等しい間繰り返す */
while (toupper((unsigned char)s1[i]) == toupper((unsigned char)s2[i])) {
if ( s1[i] == '\0' || i >= (int)(n - 1) ) {
return 0;
}
i++;
}
return toupper((unsigned char)s1[i]) - toupper((unsigned char)s2[i]);
}
サンプルプログラムの実行結果は以下のようになります.
文字列1を入力してください. > snoopy 文字列2を入力してください. > SNOOPYPY 全文字比較 snoopyとSNOOPYは同じ文字列です. 先頭の6文字分比較 snoopyとSNOOPYは同じ文字列です. 文字列1を入力してください. > snoopy! 文字列2を入力してください. > SNOOPY! 全文字比較 snoopy!とSNOOPYは異なる文字列です. 先頭の6文字分比較 snoopy!とSNOOPYは同じ文字列です.
たくさんあるC言語関連の書籍の中でも特に役に立った本です.よかったら参考にしてみてください.
C言語の実践的参考書.少々値段は張りますが初心者を脱しようとしている人は絶対に読むべきです.
文法だけでなく,コーディングスタイルやデバッグなど文字通り「実践的」なことが書かれているので非常にためになります.
オライリーの本は,読みにくい本が多いのですが本書はとても読みやすくオススメです.
ポインタの解説書としては最高の書籍です.
この1冊でポインタを完全に理解することができます.全くの初学者が読むには敷居が高いですが,入門書を読み終えた後に読むと非常に有益です.