#include <libgen.h> char *dirname(char *path); char *basename(char *path);
dirname() と basename() は、ヌルで終端されたパス名の文字列を、 ディレクトリ部分・ファイル名部分に分割する。 通常は、 dirname() は最後の '/' までの部分 (最後の '/' は含まない) を返し、 basename() は最後の '/' 以降の部分を返す。 文字列の末尾についた '/' 文字は、パス名の一部とはみなされない。
path に '/' 文字がない場合は、 dirname() は文字列 "." を返し、 basename() は path と同じ内容を返す。 path が文字列 "/" に等しい場合は、 dirname() も basename() も文字列 "/" を返す。 path が ヌルポインターだったり、空の文字列を指していた場合は、 dirname() も basename() も文字列 "." を返す。
dirname() の返した文字列、 "/"、 basename() の返した文字列、 を順に結合すると、完全なパス名が得られる。
dirname() と basename() は、いずれも path の内容を変更することがある。 したがって、これらの関数を呼び出す際には コピーを渡すのが望ましい。
これらの関数は、静的に割り当てられたメモリーへのポインターを返すことがあり、 これらの領域は後の関数呼び出しで上書きされるかもしれない。 また、これらの関数は path の一部分を指すポインターを返すこともある。そのため、 path で参照される文字列は、関数が返すポインターが不要になるまでは 変更したり free したりすべきではない。
以下の一連の例 (SUSv2 から引用) は、 いろいろな path に対して dirname() と basename() が返す文字列を表したものである。
path | dirname | basename |
|
/usr/lib | /usr | lib |
|
/usr/ | / | usr |
|
usr | . | usr |
|
/ | / | / |
|
. | . | . |
|
.. | . | .. |
インターフェース | 属性 | 値 |
basename(), dirname() | Thread safety | MT-Safe |
#define _GNU_SOURCE /* feature_test_macros(7) 参照 */ #include <string.h>
GNU バージョンは引数を変更することはなく、 path の末尾が '/'の場合は空の文字列を返す。 特に path が "/" の場合も空文字列を返す。 dirname() には GNU バージョンはない。
glibc では、 <libgen.h> をインクルードすると POSIX バージョンの basename() が使用され、それ以外の場合は GNU バージョンとなる。
バージョン 2.2.1 以前の glibc では、 glibc の dirname() は末尾が '/' 文字になっているパス名を正しく扱えず、 引数が NULL だとセグメンテーションフォールトを起こした。
dirc = strdup(path); basec = strdup(path); dname = dirname(dirc); bname = basename(basec); printf("dirname=%s, basename=%s\n", dname, bname);
[man1]
[man2]
[man3]
[man4]
[man5]
[man6]
[man7]
[man8]
[a]
[b]
[c]
[d]
[e]
[f]
[g]
[h]
[i]
[j]
[k]
[l]
[m]
[n]
[o]
[p]
[q]
[r]
[s]
[t]
[u]
[v]
[w]
[x]
[y]
[z]