web-dev-qa-db-ja.com

他のコンパイラのMSVCの_countofと同等ですか?

他のコンパイラ、特にGCCとClangによって提供される _countof に相当する組み込みのものはありますか?非マクロ形式はありますか?

30
Matt Joiner

私はGCC用のものを知りませんが、Linuxは GCCの__builtin_types_compatible_p組み込み を使用してARRAY_SIZE()マクロをより安全にします(ポインターに適用するとビルドが中断します):

/* &a[0] degrades to a pointer: a different type from an array */
#define __must_be_array(a) \
 BUILD_BUG_ON_ZERO(__builtin_types_compatible_p(typeof(a), typeof(&a[0])))

#define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]) + __must_be_array(arr))

注:BUILD_BUG_ON_ZERO()マクロの名前は誤解を招くと思います(式がnotゼロの場合、ビルドが失敗し、0が返されます)さもないと):

/* Force a compilation error if condition is true, but also produce a
   result (of value 0 and type size_t), so the expression can be used
   e.g. in a structure initializer (or where-ever else comma expressions
   aren't permitted). */
#define BUILD_BUG_ON_ZERO(e) (sizeof(struct { int:-!!(e); }))

このマクロの名前は、2つの部分で見ることに由来すると思います。BUILD_BUG_ONは式がtrueの場合にマクロが実行することであり、ZEROはマクロによって「返される」値です(ビルドがない場合)。ブレーク)。

6
Michael Burr

C++ 11を使用する場合、非マクロ形式は次のとおりです。

char arrname[5];
size_t count = std::extent< decltype( arrname ) >::value;

また、extenttype_traitsヘッダーにあります。

または、少し見栄えを良くしたい場合は、次のようにラップします。

template < typename T, size_t N >
size_t countof( T ( & arr )[ N ] )
{
    return std::extent< T[ N ] >::value;
}

そしてそれは次のようになります:

char arrname[5];
size_t count = countof( arrname );

char arrtwo[5][6];
size_t count_fst_dim = countof( arrtwo );    // 5
size_t count_snd_dim = countof( arrtwo[0] ); // 6

編集:「C++」ではなく「C」フラグに気づきました。したがって、Cのためにここにいる場合は、この投稿を無視してください。ありがとう。

13
Kurt Hutchinson

この?

#define _countof(a) (sizeof(a)/sizeof(*(a)))

5
Mud

更新:C++ 17のサポート std::size() (ヘッダー_<iterator>_で定義)

代わりに boost::size() を使用できます:

_#include <boost/range.hpp>

int my_array[10];
boost::size(my_array);
_
4
KindDragon