世の中には怖いもの知らずというか、とんでもないコードを書く人たちがいます。例えば、
|
0 1 2 3 4 5 6 7 8 9 10 11 12 |
struct foo { std::string str; int a; int b; foo() { std::memset(this, 0, sizeof(*this)); } }; |
のようなコードです。さすがにこれはクラッシュする可能性が高いので、実行させてみれば間違いに気付くことでしょう。しかし、こんなのはどうでしょうか?
|
0 1 2 3 4 5 6 7 8 9 10 11 12 |
struct bar { std::pair<long, double> x; int a; int b; bar() { std::memset(this, 0, sizeof(*this)); } }; |
これだと動いてしまう処理系のほうが多そうです。std::pairのようなクラスをstd::memsetでゼロクリアするのはもちろん反則ですので未定義の動作になるわけですが、たまたま動いてしまうわけです。
では、std::stringやstd::pairのようなクラスを一切メンバに持たない、すなわちC互換型だけをメンバに持つような構造体であればコンストラクタでこのようなゼロクリアを行ってもよいのでしょうか?
そんなはずはありませんね。明示的なコンストラクタを定義した時点で、その構造体はもはやC互換構造体ではなくなります。C互換型ではないオブジェクトをstd::memsetでクリアすることはできません。多くの場合は動くかもしれませんが、それはあくまでもたまたまです。メンバの構成が変更されれば実害が出ることも十分考えられるので、こんなコードを書くのは避けるべきです。

![[迷信] 入力データ格納用配列のサイズはBUFSIZ](https://www.kijineko.co.jp/wp-content/uploads/2021/06/buffer.png)
![[迷信] 0xe-0xeはゼロ](https://www.kijineko.co.jp/wp-content/uploads/2021/05/4195259_s.jpg)
![[迷信] 識別子に使える文字は英数字と下線のみ](https://www.kijineko.co.jp/wp-content/uploads/2021/06/1210817_s.jpg)
![[迷信] sizeofは定数式](https://www.kijineko.co.jp/wp-content/uploads/2021/06/4882007_s.jpg)
![[迷信] 構造体はクラスではない](https://www.kijineko.co.jp/wp-content/uploads/2021/06/2023246_s.jpg)
![[迷信] オブジェクトの動的生成に失敗するとメモリリークする](https://www.kijineko.co.jp/wp-content/uploads/2021/06/4673249_s.jpg)