※前スレ
C++相談室 part154
https://mevius.5ch.net/test/read.cgi/tech/1610096040/
テンプレここまで
探検
C++相談室 part155
■ このスレッドは過去ログ倉庫に格納されています
2021/03/24(水) 12:07:15.39ID:R+oM8cup
497デフォルトの名無しさん
2021/04/27(火) 10:39:34.03ID:zzzFrqtR 例?
整数リテラル
整数リテラル
498デフォルトの名無しさん
2021/04/27(火) 12:29:20.94ID:ul4gccCl 関数
499デフォルトの名無しさん
2021/04/27(火) 12:34:12.25ID:rYx8lJmb const
500デフォルトの名無しさん
2021/04/27(火) 14:47:15.34ID:V9b4VlmB >>496
・定数リテラル、const属性が付くもの、は置けない。
・関数呼び出しの 関数()は、戻り値は、伝統的に右辺値扱いになるので左辺に置くとエラー
になる様になっている。
戻り値は、構造体型(クラス型/union型含む)/整数型/浮動小数点型/ポインタ型/列挙型の場合
を想定した。
・*ptr のようなものは左辺値になるので置ける。
・変数名はconst 修飾されていないなら置ける。
・構造体変数名.データメンバ名 は const 修飾されていないなら置ける。
・「関数名」は置けない。これは関数呼び出しの関数()とは別の話。
・&x は置けない。理由としては右辺値であるから。constでもあるが。
・x + y は、組み込み演算子の場合でも、関数呼び出しに置き換わった場合でも
置けない。後者の場合は、関数の戻り値が置いてあるということになるが右辺値だから。
前者の場合は、右辺値だからだと思う。
・定数リテラル、const属性が付くもの、は置けない。
・関数呼び出しの 関数()は、戻り値は、伝統的に右辺値扱いになるので左辺に置くとエラー
になる様になっている。
戻り値は、構造体型(クラス型/union型含む)/整数型/浮動小数点型/ポインタ型/列挙型の場合
を想定した。
・*ptr のようなものは左辺値になるので置ける。
・変数名はconst 修飾されていないなら置ける。
・構造体変数名.データメンバ名 は const 修飾されていないなら置ける。
・「関数名」は置けない。これは関数呼び出しの関数()とは別の話。
・&x は置けない。理由としては右辺値であるから。constでもあるが。
・x + y は、組み込み演算子の場合でも、関数呼び出しに置き換わった場合でも
置けない。後者の場合は、関数の戻り値が置いてあるということになるが右辺値だから。
前者の場合は、右辺値だからだと思う。
501デフォルトの名無しさん
2021/04/27(火) 14:48:40.00ID:V9b4VlmB >>500
>・構造体変数名.データメンバ名 は const 修飾されていないなら置ける。
これについては、構造体変数名が、右辺値の場合は、
構造体変数名.データメンバ名も右辺値になるため、置けない。
>・構造体変数名.データメンバ名 は const 修飾されていないなら置ける。
これについては、構造体変数名が、右辺値の場合は、
構造体変数名.データメンバ名も右辺値になるため、置けない。
502デフォルトの名無しさん
2021/04/27(火) 15:10:33.44ID:V9b4VlmB [続き]
・(cast)x は置けない。理由は、右辺値になるから。
なので、int i; (char)i = 5; はエラーになる。
そうしたい場合は、*(char *)&i = 5; と書く。
・(cast)x は置けない。理由は、右辺値になるから。
なので、int i; (char)i = 5; はエラーになる。
そうしたい場合は、*(char *)&i = 5; と書く。
503デフォルトの名無しさん
2021/04/27(火) 17:53:11.42ID:x+a+UXmv 配列を要素展開して、可変長引数の関数に渡したいんですけど、どうかけばいいか分かりません
template<typename... Ts>
void g(Ts... ts){}
template<typename T,size_t N>
void f(T (&a)[N])
{
g(a[0],a[1],a[2]/* なんて書くのか*/);
}
template<typename... Ts>
void g(Ts... ts){}
template<typename T,size_t N>
void f(T (&a)[N])
{
g(a[0],a[1],a[2]/* なんて書くのか*/);
}
504デフォルトの名無しさん
2021/04/27(火) 18:30:39.66ID:z5odOJ3h505デフォルトの名無しさん
2021/04/27(火) 19:37:01.40ID:2o2XkKHN >>504
ありがとう出来ました
ありがとう出来ました
506デフォルトの名無しさん
2021/04/27(火) 19:58:19.96ID:1Ls3FsW9 通常クラスのコンストラクタにテンプレート引数がある場合ってどう呼び出したらよいでしょう?
class Test{
public:
template<class T>
Test() {}
};
Test test = Test::Test<int>();
で呼び出せるかと思ったのですが出来ず...
class Test{
public:
template<class T>
Test(T dummy) {}
};
コンストラクタに引数を持たせると、msvcでは一応できました。
Test test(0);
class Test{
public:
template<class T>
Test() {}
};
Test test = Test::Test<int>();
で呼び出せるかと思ったのですが出来ず...
class Test{
public:
template<class T>
Test(T dummy) {}
};
コンストラクタに引数を持たせると、msvcでは一応できました。
Test test(0);
507デフォルトの名無しさん
2021/04/27(火) 20:39:05.42ID:eX4df2SV 代入演算子ってメンバ関数じゃないとだめなんだな
thisを返すから当たり前だが、オーバーロードしたいとき困る
a = b を意味する assign(a, b) を作るしかない?
thisを返すから当たり前だが、オーバーロードしたいとき困る
a = b を意味する assign(a, b) を作るしかない?
508デフォルトの名無しさん
2021/04/27(火) 21:47:32.08ID:rFiajegR 別にthisを返す必要もないけど
509デフォルトの名無しさん
2021/04/28(水) 00:01:05.47ID:pTBAhwEs thisがどうのじゃなくてコピー代入演算子とムーブ代入演算子が特別扱いだから
510デフォルトの名無しさん
2021/04/28(水) 02:07:16.16ID:LEZnE3AK >>506 https://timsong-cpp.github.io/cppwp/n4861/temp.arg.explicit#8
> [Note: Because the explicit template argument list follows the function template name, and because constructor templates ([class.ctor]) are named
> without using a function name ([class.qual]), there is no way to provide an explicit template argument list for these function templates. - end note]
> [Note: Because the explicit template argument list follows the function template name, and because constructor templates ([class.ctor]) are named
> without using a function name ([class.qual]), there is no way to provide an explicit template argument list for these function templates. - end note]
511デフォルトの名無しさん
2021/04/28(水) 03:41:16.44ID:v8E9sca8 unique_ptrって、unique_ptr<T>とuniqu_ptr<T[]>が、1つのテンプレートではなく、
テンプレート自体が別に用意されてるんだよね?
そもそも前者の規則と後者は使う時の記号としても違っていて、前者は、
unique_ptr<int> a = new int;
*a = 5;
と書くのだから、a は、int*、つまり、intへのポインタのように振舞う。
この規則のままであるなら、
unique_ptr<int[]> b = new int[10];
と書いた場合、b は、int[10] へのポインタ、つまり、int (*b)[10] のように
振舞わなければならない。
となると、
b[idx] = value;
とは書けずに、(*b)[idx] = value; と書かねば成らないが、実際にはそうではない。
テンプレート自体が別に用意されてるんだよね?
そもそも前者の規則と後者は使う時の記号としても違っていて、前者は、
unique_ptr<int> a = new int;
*a = 5;
と書くのだから、a は、int*、つまり、intへのポインタのように振舞う。
この規則のままであるなら、
unique_ptr<int[]> b = new int[10];
と書いた場合、b は、int[10] へのポインタ、つまり、int (*b)[10] のように
振舞わなければならない。
となると、
b[idx] = value;
とは書けずに、(*b)[idx] = value; と書かねば成らないが、実際にはそうではない。
512はちみつ餃子 ◆8X2XSCHEME
2021/04/28(水) 05:20:55.28ID:cpOEbmvB >>511
> テンプレート自体が別に用意されてるんだよね?
特殊化で配列の場合は特別に用意されている。
> unique_ptr<int[]> b = new int[10];
> と書いた場合、b は、int[10] へのポインタ、つまり、int (*b)[10] のように
> 振舞わなければならない。
(そのように b を初期化することは出来ない (生ポインタを受け取るコンストラクタには explicit が付いてる) が、意図はわかるのでとりあえずわきに置く。)
new int[10]; という式の型は int* なので、この時点で配列の大きさに関する情報は失われている。
(配列の先頭要素を指す生ポインタではなく) 配列を指す生ポインタのようにスマートポインタを抽象化する意味がない。
どうしてもやりたければ std::array と組み合わせればいいし。
> テンプレート自体が別に用意されてるんだよね?
特殊化で配列の場合は特別に用意されている。
> unique_ptr<int[]> b = new int[10];
> と書いた場合、b は、int[10] へのポインタ、つまり、int (*b)[10] のように
> 振舞わなければならない。
(そのように b を初期化することは出来ない (生ポインタを受け取るコンストラクタには explicit が付いてる) が、意図はわかるのでとりあえずわきに置く。)
new int[10]; という式の型は int* なので、この時点で配列の大きさに関する情報は失われている。
(配列の先頭要素を指す生ポインタではなく) 配列を指す生ポインタのようにスマートポインタを抽象化する意味がない。
どうしてもやりたければ std::array と組み合わせればいいし。
513デフォルトの名無しさん
2021/04/28(水) 12:25:48.23ID:jQpDsyge >>512
すまん。正しい書き方は:
unique_ptr<int[]> b(new int[10]); //(1)
だったようだ。
ちょっと意図が伝わらなかったのか、言いたかったのは、
unique_ptr<T> a; //(2)
の場合、a の型は、T* のようになるのに、
unique_ptr<U[]> b;
の型は、U* のようになるので、数学の様に最初のT=U[]
すまん。正しい書き方は:
unique_ptr<int[]> b(new int[10]); //(1)
だったようだ。
ちょっと意図が伝わらなかったのか、言いたかったのは、
unique_ptr<T> a; //(2)
の場合、a の型は、T* のようになるのに、
unique_ptr<U[]> b;
の型は、U* のようになるので、数学の様に最初のT=U[]
514デフォルトの名無しさん
2021/04/28(水) 12:39:35.77ID:jQpDsyge すまん、まちがって送信してしまった。
>>512
正しい書き方は:
unique_ptr<int[]> b(new int[10]); //(1)
だったようだ。それはともかく、ちょっと意図が伝わらなかったのか、言いたかったのは、
unique_ptr<T> a; //(2)
の場合、a の型は T * であるかのように振舞う(T *a と宣言していたかのように振舞う)のに、
unique_ptr<U[]> b; //(3)
の場合、b の型は U* のように振舞うが、数学の様に(2)にT=U[]を代入してみると、
b の型は本来、T *b とした場合のように振舞うはずなので、数学的には U (*b) [] とした
場合の型になっていなければならないはずなのに、実際には、U *b のようにした場合
の型になっているということで、(3)は数学的には (2)の特殊形とはみなせないということ。
なので、unique_ptr<U[]>のテンプレートは、unique_ptr<T>のテンプレートとは別に人間が
意図的に専用のコードを書いて「特殊化」していることの証拠となるということ。
もちろんそれが「テンプレート特殊化」という仕組みで行われていることは知っているが、
数学的な意味で(2)を一般形とみなした場合の自動的な特殊形にはなってないということ
(だからこそ「テンプレート特殊化」で特殊な場合だけを例外的に特記するのではあるが)。
これは、unique_ptr<T>は、Tが配列型の場合は、Tがその他の場合と比べて
「一般性を失っている」と言える。
一般性を失っていることは、言語として分かりにくくなってしまうと俺は思うんだ。
>>512
正しい書き方は:
unique_ptr<int[]> b(new int[10]); //(1)
だったようだ。それはともかく、ちょっと意図が伝わらなかったのか、言いたかったのは、
unique_ptr<T> a; //(2)
の場合、a の型は T * であるかのように振舞う(T *a と宣言していたかのように振舞う)のに、
unique_ptr<U[]> b; //(3)
の場合、b の型は U* のように振舞うが、数学の様に(2)にT=U[]を代入してみると、
b の型は本来、T *b とした場合のように振舞うはずなので、数学的には U (*b) [] とした
場合の型になっていなければならないはずなのに、実際には、U *b のようにした場合
の型になっているということで、(3)は数学的には (2)の特殊形とはみなせないということ。
なので、unique_ptr<U[]>のテンプレートは、unique_ptr<T>のテンプレートとは別に人間が
意図的に専用のコードを書いて「特殊化」していることの証拠となるということ。
もちろんそれが「テンプレート特殊化」という仕組みで行われていることは知っているが、
数学的な意味で(2)を一般形とみなした場合の自動的な特殊形にはなってないということ
(だからこそ「テンプレート特殊化」で特殊な場合だけを例外的に特記するのではあるが)。
これは、unique_ptr<T>は、Tが配列型の場合は、Tがその他の場合と比べて
「一般性を失っている」と言える。
一般性を失っていることは、言語として分かりにくくなってしまうと俺は思うんだ。
515デフォルトの名無しさん
2021/04/28(水) 12:44:49.64ID:P6pu+tTf 「数学的に」
数学を知らないヤツがよく使う言葉
類義語
「物理的に」
数学を知らないヤツがよく使う言葉
類義語
「物理的に」
516デフォルトの名無しさん
2021/04/28(水) 12:56:00.95ID:VWIud7ZL 規格的にってのも仕事ができない言語厨がよく使ってるw
517デフォルトの名無しさん
2021/04/28(水) 12:57:27.64ID:jQpDsyge >>515
「数学的に」と書いたのは、法則に従った「規則変化」しているということだ。
言語仕様で決まっているとかではなく、記号パターンで「導出」されるというか。
「代入」の概念というか。
優先順位のために、U (*b)[] のような 記号になっているが、これも数学的に
演算子が演算される順序に従って「平坦」に書くと
b --> * --> [] --> U
となる。読み方は、一番左の b の部分以外が右から順に
bの型は「Uの配列へのポインタ」
となる。ちなみに、
T b の場合は、
b --> T
となり、読み方は、
bの型は「T」
となる。
「数学的に」と書いたのは、法則に従った「規則変化」しているということだ。
言語仕様で決まっているとかではなく、記号パターンで「導出」されるというか。
「代入」の概念というか。
優先順位のために、U (*b)[] のような 記号になっているが、これも数学的に
演算子が演算される順序に従って「平坦」に書くと
b --> * --> [] --> U
となる。読み方は、一番左の b の部分以外が右から順に
bの型は「Uの配列へのポインタ」
となる。ちなみに、
T b の場合は、
b --> T
となり、読み方は、
bの型は「T」
となる。
518デフォルトの名無しさん
2021/04/28(水) 12:58:27.74ID:jQpDsyge519デフォルトの名無しさん
2021/04/28(水) 13:01:24.71ID:jQpDsyge >>517
もう少し噛み砕いて書くと、コンパイラ内部では、
U (*b)[];
という宣言は、優先順位に従って、
b --> * --> [] --> U
となり、コンパイラ内部では左から順に
「bは、ポインタ(*)であり、その元の型は、配列([])であり、その元の型は、
U である」
と理解している。
もう少し噛み砕いて書くと、コンパイラ内部では、
U (*b)[];
という宣言は、優先順位に従って、
b --> * --> [] --> U
となり、コンパイラ内部では左から順に
「bは、ポインタ(*)であり、その元の型は、配列([])であり、その元の型は、
U である」
と理解している。
520デフォルトの名無しさん
2021/04/28(水) 13:04:48.90ID:jQpDsyge >>519
b --> * --> [] --> U
は、英語で読むと、左から順に、
「b is a pointer(*) to array([]) to type U.」
と読めて、言葉と記号の順序が一致する。
b --> * --> [] --> U
は、英語で読むと、左から順に、
「b is a pointer(*) to array([]) to type U.」
と読めて、言葉と記号の順序が一致する。
521デフォルトの名無しさん
2021/04/28(水) 14:25:22.22ID:uUyjbKVX void hoge(){
vector<int> a(10);
vector<int*> p(10);
for(int i=0; i<10; i++) p[i] = &a[i];
// play with a and p
}
っていう関数で a とか p に対してやったらやばい操作ってどんなんですかね
vector<int> a(10);
vector<int*> p(10);
for(int i=0; i<10; i++) p[i] = &a[i];
// play with a and p
}
っていう関数で a とか p に対してやったらやばい操作ってどんなんですかね
522デフォルトの名無しさん
2021/04/28(水) 14:41:21.61ID:P6pu+tTf 数学記号に関するIQて何?
523デフォルトの名無しさん
2021/04/28(水) 15:09:55.27ID:pxclvZlf524デフォルトの名無しさん
2021/04/28(水) 15:12:59.93ID:GWxUY3yT すごい
クソゲボゴミ老害の意見
頼まれたことはできない、自分なりに行動しても何も産まない、家族と職場の全員から疎まれてるバカ
天晴
クソゲボゴミ老害の意見
頼まれたことはできない、自分なりに行動しても何も産まない、家族と職場の全員から疎まれてるバカ
天晴
525デフォルトの名無しさん
2021/04/28(水) 15:14:07.47ID:UpJQEntB >>510
ありがとうございます!できないんですね…
ありがとうございます!できないんですね…
526デフォルトの名無しさん
2021/04/28(水) 15:21:22.38ID:aIuZlW8v527デフォルトの名無しさん
2021/04/28(水) 15:26:21.46ID:+7CWSFOZ >>521
pのサイズが変わるような操作はアドレスが変わる可能性があるので全てアウト
pのサイズが変わるような操作はアドレスが変わる可能性があるので全てアウト
528デフォルトの名無しさん
2021/04/28(水) 15:27:17.73ID:+7CWSFOZ >>527
おっと、aのサイズが〜の間違い
おっと、aのサイズが〜の間違い
529デフォルトの名無しさん
2021/04/28(水) 15:51:27.64ID:P5vpmJVI >>515
そうか、禿Stroustrupは数学を知らないのか
そうか、禿Stroustrupは数学を知らないのか
530デフォルトの名無しさん
2021/04/28(水) 15:52:12.82ID:Kf9DoDRw pのサイズ変わるのもまずいよね?
勝手に new/delete されて使用不能になりえるので
勝手に new/delete されて使用不能になりえるので
531デフォルトの名無しさん
2021/04/28(水) 16:03:18.23ID:P6pu+tTf >>529
どこからそういう結論になった?
どこからそういう結論になった?
532デフォルトの名無しさん
2021/04/28(水) 16:04:05.01ID:P6pu+tTf533デフォルトの名無しさん
2021/04/28(水) 16:06:25.57ID:7RK+jwPd >>530
晒しage
晒しage
534デフォルトの名無しさん
2021/04/28(水) 16:18:30.85ID:eLEqCP2l >>530
ダメな例をコードで示してくれ
ダメな例をコードで示してくれ
535デフォルトの名無しさん
2021/04/28(水) 16:28:13.84ID:c0w2coaF 自作クラスや構造体に対して範囲forを使うための条件って、そのクラスや構造体が
・メンバ関数としてbegin()、end()を持つ
・begin()、end()が間接参照演算子、インクリメント演算子、不等価演算子を持つクラスか構造体を返す
で合ってますか?
cpprefjp見てたら一個目の条件しか書いてなかったんですが、二個目の条件要りますよね?
・メンバ関数としてbegin()、end()を持つ
・begin()、end()が間接参照演算子、インクリメント演算子、不等価演算子を持つクラスか構造体を返す
で合ってますか?
cpprefjp見てたら一個目の条件しか書いてなかったんですが、二個目の条件要りますよね?
536はちみつ餃子 ◆8X2XSCHEME
2021/04/28(水) 16:35:30.69ID:cpOEbmvB537デフォルトの名無しさん
2021/04/28(水) 18:23:34.44ID:c0w2coaF もう一個質問させてください
範囲for文ってfor文スコープ外の変数を使えないんですか?
int i;
vector<int> v;
// v を初期化
for(i: v){
...
}
みたいなことをしたいです
範囲for文ってfor文スコープ外の変数を使えないんですか?
int i;
vector<int> v;
// v を初期化
for(i: v){
...
}
みたいなことをしたいです
538デフォルトの名無しさん
2021/04/28(水) 18:36:49.45ID:7AKt1vSf539デフォルトの名無しさん
2021/04/29(木) 06:49:31.89ID:1rAkIDNr >>531
ISBN4-7561-1895のP.500 16.1.1 設計上の制約
ISBN4-7561-1895のP.500 16.1.1 設計上の制約
540デフォルトの名無しさん
2021/04/29(木) 06:50:25.72ID:+7uc9ATw >>538
最近こういうゴミ返しするバカめっちゃ増えたね、このスレ
「intじゃダメ」は「間接参照演算子、インクリメント演算子、不等価演算子を持つクラスか構造体である必要がある」を全く意味しないだろ……
一方で、ではイテレータを返せば良いのかというとそこまで条件が厳しいわけでもなくて、イテレータ「らしき」性質を持ったクラスを返せば良いということなので、この話はそこまで単純じゃない
そういうこと全く分かってないバカが場当たり的な回答をしてるのは甚だ不快ですね
最近こういうゴミ返しするバカめっちゃ増えたね、このスレ
「intじゃダメ」は「間接参照演算子、インクリメント演算子、不等価演算子を持つクラスか構造体である必要がある」を全く意味しないだろ……
一方で、ではイテレータを返せば良いのかというとそこまで条件が厳しいわけでもなくて、イテレータ「らしき」性質を持ったクラスを返せば良いということなので、この話はそこまで単純じゃない
そういうこと全く分かってないバカが場当たり的な回答をしてるのは甚だ不快ですね
541デフォルトの名無しさん
2021/04/29(木) 07:36:35.34ID:fshfa4aU えっっ
Tをイテレータ様のふるまいを示す型として
T begin()だけではダメで
T end()も最低限定義いないといけないんじゃ……
Tをイテレータ様のふるまいを示す型として
T begin()だけではダメで
T end()も最低限定義いないといけないんじゃ……
542デフォルトの名無しさん
2021/04/29(木) 07:50:14.64ID:1rAkIDNr543デフォルトの名無しさん
2021/04/29(木) 08:34:23.25ID:3mmNht9g544デフォルトの名無しさん
2021/04/29(木) 08:51:08.82ID:fshfa4aU 質問者がbegin()とend()の定義の必要性を把握しているのに対して
begin()のみではNGだとレスするのがそんなにドヤるほどのことなのかどうか……
begin()のみではNGだとレスするのがそんなにドヤるほどのことなのかどうか……
545デフォルトの名無しさん
2021/04/29(木) 08:55:45.00ID:3mmNht9g546デフォルトの名無しさん
2021/04/29(木) 09:04:34.25ID:6rdxVFZp >>542
ありがとうございます
v は必ずしも vector<int> ではなく vector<vector<int>> とかで、かついっぱい回るループを想定してるのでできるだけ定数倍を軽くしたいのですが、j を右辺値参照にすればコピーは起きないので相当マシと思って良いですかね
ありがとうございます
v は必ずしも vector<int> ではなく vector<vector<int>> とかで、かついっぱい回るループを想定してるのでできるだけ定数倍を軽くしたいのですが、j を右辺値参照にすればコピーは起きないので相当マシと思って良いですかね
547デフォルトの名無しさん
2021/04/29(木) 09:07:08.51ID:1rAkIDNr 右辺値参照つってもauto&&は左辺値参照も兼ねるぞ
548デフォルトの名無しさん
2021/04/29(木) 09:16:30.33ID:fshfa4aU549デフォルトの名無しさん
2021/04/29(木) 09:22:13.70ID:3mmNht9g 二個目の条件に反してる
550デフォルトの名無しさん
2021/04/29(木) 09:41:24.58ID:fshfa4aU551デフォルトの名無しさん
2021/04/29(木) 10:20:12.63ID:gzXxIopT552デフォルトの名無しさん
2021/04/29(木) 10:26:57.88ID:6rdxVFZp >>547
for()のカッコの中でiをjの参照として定義できたらそれでも良いんですが、そういうことはできるんでしたっけ?
for()のカッコの中でiをjの参照として定義できたらそれでも良いんですが、そういうことはできるんでしたっけ?
553デフォルトの名無しさん
2021/04/29(木) 10:28:45.87ID:6rdxVFZp すみません「定義」というよりは、forのスコープの外にiの宣言はあって、for(ここ)でiがjの参照であることにできたら良いってことです
554デフォルトの名無しさん
2021/04/29(木) 10:29:35.81ID:fshfa4aU555デフォルトの名無しさん
2021/04/29(木) 10:35:45.46ID:gzXxIopT556デフォルトの名無しさん
2021/04/29(木) 11:07:16.11ID:1rAkIDNr557デフォルトの名無しさん
2021/04/29(木) 11:10:24.28ID:6rdxVFZp558デフォルトの名無しさん
2021/04/29(木) 11:52:35.69ID:1rAkIDNr559デフォルトの名無しさん
2021/04/29(木) 12:28:01.48ID:vjsl7cGC ポインタと参照の決定的な違いはそこだね
560デフォルトの名無しさん
2021/04/29(木) 12:46:42.20ID:yUiVUiFp >>541
begin()とend()の型が一致してる必要はない(C++17〜)
begin()とend()の型が一致してる必要はない(C++17〜)
561デフォルトの名無しさん
2021/04/29(木) 13:03:31.68ID:K/HFYMcp https://negation.hatenadiary.org/entry/20111203/1322876171
↑で、簡単に書けば、
class Book {・・・};
class Novel : public Book{・・・};
class Comic : public Book{・・・};
class Shelf { public: std::vector<Book> list; ・・・ };
Shelf g_shelf;
int main(void) {
Novel n = Novel("Hoshio wo tugumono");
Comic c = Comic("Kimetsu no Yaiba");
Shelf s = Shelf();
g_shelft.list.push_back(n);
g_shelft.list.push_back(c);
}
のようになっているところがあるけど、
vector<Book>って、Bookの実体の動的配列で、
Book a[100];
とにら様なものだと思うんだけど、NovelやComicのクラスのバイトサイズが
Bookを越えたら、入りきれないと思うんだけど、これで合ってる?
合ってるとしたら、どういう仕組み?
もしかして、vector<Book>って、
Book *p[100];
みたいな配列なの?
↑で、簡単に書けば、
class Book {・・・};
class Novel : public Book{・・・};
class Comic : public Book{・・・};
class Shelf { public: std::vector<Book> list; ・・・ };
Shelf g_shelf;
int main(void) {
Novel n = Novel("Hoshio wo tugumono");
Comic c = Comic("Kimetsu no Yaiba");
Shelf s = Shelf();
g_shelft.list.push_back(n);
g_shelft.list.push_back(c);
}
のようになっているところがあるけど、
vector<Book>って、Bookの実体の動的配列で、
Book a[100];
とにら様なものだと思うんだけど、NovelやComicのクラスのバイトサイズが
Bookを越えたら、入りきれないと思うんだけど、これで合ってる?
合ってるとしたら、どういう仕組み?
もしかして、vector<Book>って、
Book *p[100];
みたいな配列なの?
562デフォルトの名無しさん
2021/04/29(木) 13:05:10.26ID:K/HFYMcp >>561
すまん。間違った。正しくはこう :
int main(void) {
Novel n = Novel("Hoshio wo tugumono");
Comic c = Comic("Kimetsu no Yaiba");
Shelf s = Shelf();
s.list.push_back(n);
s.list.push_back(c);
}
すまん。間違った。正しくはこう :
int main(void) {
Novel n = Novel("Hoshio wo tugumono");
Comic c = Comic("Kimetsu no Yaiba");
Shelf s = Shelf();
s.list.push_back(n);
s.list.push_back(c);
}
563デフォルトの名無しさん
2021/04/29(木) 13:10:29.61ID:yUiVUiFp564デフォルトの名無しさん
2021/04/29(木) 13:10:34.24ID:K/HFYMcp https://stackoverflow.com/questions/16126578/vectors-and-polymorphism-in-c
↑によれば、やっぱり、>>561 や >>562 のようなやり方は間違いで、
std::vector<T> の T は、Bookではなく、shared_ptr<Book> のようなものを入れるべきで、
以下の様になっている。だから、>>561 や >>562 は間違いだよね?
class Instruction {・・・};
class Add: public Instruction{・・・};
typedef shared_ptr<Instruction> PInstruction;
vector<PInstruction> v;
v.emplace_back(make_shared<Add>());
↑によれば、やっぱり、>>561 や >>562 のようなやり方は間違いで、
std::vector<T> の T は、Bookではなく、shared_ptr<Book> のようなものを入れるべきで、
以下の様になっている。だから、>>561 や >>562 は間違いだよね?
class Instruction {・・・};
class Add: public Instruction{・・・};
typedef shared_ptr<Instruction> PInstruction;
vector<PInstruction> v;
v.emplace_back(make_shared<Add>());
565デフォルトの名無しさん
2021/04/29(木) 13:16:01.96ID:yUiVUiFp566デフォルトの名無しさん
2021/04/29(木) 13:19:09.72ID:K/HFYMcp >>563
なるほど、では、正しい書き方としては、たとえば、こうかな:
class Book {・・・};
class Novel : public Book{・・・};
class Comic : public Book{・・・};
typedef shared_ptr<Book> PBOOK;
class Shelf : public std::vector<PBOOK> {・・・};
Shelf g_shelf;
int main(void) {
std::shared_ptr<Novel> n = make_shared<Novel>("Hoshio wo tugumono");
std::shared_ptr<Comic> c = make_shared<Comic>("Kimetsu no Yaiba");
g_shelf.push_back(n);
g_shelf.push_back(c);
}
なるほど、では、正しい書き方としては、たとえば、こうかな:
class Book {・・・};
class Novel : public Book{・・・};
class Comic : public Book{・・・};
typedef shared_ptr<Book> PBOOK;
class Shelf : public std::vector<PBOOK> {・・・};
Shelf g_shelf;
int main(void) {
std::shared_ptr<Novel> n = make_shared<Novel>("Hoshio wo tugumono");
std::shared_ptr<Comic> c = make_shared<Comic>("Kimetsu no Yaiba");
g_shelf.push_back(n);
g_shelf.push_back(c);
}
567はちみつ餃子 ◆8X2XSCHEME
2021/04/29(木) 18:07:20.75ID:x0Vd7BP9 パブリック継承している型のオブジェクトが基底クラスへ暗黙に型変換されるのは抑止する方法がないんだよな。
568デフォルトの名無しさん
2021/04/30(金) 01:30:09.20ID:7VhEvZ/Q569デフォルトの名無しさん
2021/04/30(金) 08:17:30.86ID:tsrXe0Ut それは別に問題ない
func(Animal a)だとやばい
func(Animal a)だとやばい
570デフォルトの名無しさん
2021/04/30(金) 09:32:28.19ID:W8up1rh0 funcがAnimalの情報しか使わないんなら別にいいんだけどね
単にポリモーフィズムにはポインタか参照が必要ってだけ
単にポリモーフィズムにはポインタか参照が必要ってだけ
571デフォルトの名無しさん
2021/04/30(金) 09:52:57.56ID:dyTEtgxA shared_ptr<T>もいけるでしょ
スタックを使った60バイト近いメモリーコピーが発生するから最適ではないけど
スタックを使った60バイト近いメモリーコピーが発生するから最適ではないけど
572デフォルトの名無しさん
2021/04/30(金) 10:40:50.56ID:7VhEvZ/Q >>569
class Animal {・・・};
class Dog : publoic Animal {・・・};
std::vector<Animal> g_list;
void func( Animal &a )
{
g_list.push_back(a);
}
int main()
{
Dog d;
func(d); // コンパイルエラーにはならないのに、スライシングが発生。駄目な例。
return 0;
}
class Animal {・・・};
class Dog : publoic Animal {・・・};
std::vector<Animal> g_list;
void func( Animal &a )
{
g_list.push_back(a);
}
int main()
{
Dog d;
func(d); // コンパイルエラーにはならないのに、スライシングが発生。駄目な例。
return 0;
}
573デフォルトの名無しさん
2021/04/30(金) 11:02:34.65ID:Qm/DlA0m >>572
スライシングが発生してるのは g_list.push_back(a) であって、 func( Animal &a ) は関係ないでしょ。
スライシングが発生してるのは g_list.push_back(a) であって、 func( Animal &a ) は関係ないでしょ。
574デフォルトの名無しさん
2021/04/30(金) 11:21:35.01ID:qs5VuYRE なんで範囲for文ってBOOST_FOREACHを完全に置き換えれるようにしなかったの?
>>537,542はそれで解決するし、逆順のループとかもできるのに
>>537,542はそれで解決するし、逆順のループとかもできるのに
575デフォルトの名無しさん
2021/04/30(金) 11:34:29.27ID:vL4rFdIy ポリモーフィズム前提の時は、コピーコンストラクタはdeleteするのが良いのかね
576デフォルトの名無しさん
2021/04/30(金) 11:43:26.05ID:qs5VuYRE577デフォルトの名無しさん
2021/04/30(金) 11:48:46.39ID:7VhEvZ/Q >>573
この場合、スライシングが発生しても問題ない、という観点もあるかも。
この場合、スライシングが発生しても問題ない、という観点もあるかも。
578デフォルトの名無しさん
2021/04/30(金) 12:01:43.38ID:W8up1rh0 >>575
そしたら派生クラス同士でコピー出来んやろ、コピー禁止にしたいのでなければ書くべき(で派生のコピーコンストラクタで基底のコピーコンストラクタを初期化リストから呼ぶ
必ず派生させて使うのが前提ならコンストラクタをprotectedにするとかその辺のテクニックは大昔から色々出てる
そしたら派生クラス同士でコピー出来んやろ、コピー禁止にしたいのでなければ書くべき(で派生のコピーコンストラクタで基底のコピーコンストラクタを初期化リストから呼ぶ
必ず派生させて使うのが前提ならコンストラクタをprotectedにするとかその辺のテクニックは大昔から色々出てる
579デフォルトの名無しさん
2021/04/30(金) 12:18:33.85ID:7VhEvZ/Q スライシングが起きるとき、基本クラス部分だけをコピーすることになるが、
その際、仮想テーブルへのポインタが継承クラス用のままだと、コピー後の
操作が危険になる可能性があるが、もし、仮想テーブルへのポインタを
基本クラス用に変えてしてしまえば、コピー後の動作は完全に基本クラス的に
なってしまうけれど、その反面、メンバー関数を呼び出しても特にメモリーの
破壊などは起きないはず。
そのようにしてしまえばスライシングが起きても問題ないような気が。
その際、仮想テーブルへのポインタが継承クラス用のままだと、コピー後の
操作が危険になる可能性があるが、もし、仮想テーブルへのポインタを
基本クラス用に変えてしてしまえば、コピー後の動作は完全に基本クラス的に
なってしまうけれど、その反面、メンバー関数を呼び出しても特にメモリーの
破壊などは起きないはず。
そのようにしてしまえばスライシングが起きても問題ないような気が。
580デフォルトの名無しさん
2021/04/30(金) 12:31:22.41ID:W8up1rh0 >>578て書いたけど実際ポリモーフィズム目的の継承ツリーだとコピー禁止にしてること多いな・・
どうせポインタ(orスマポ)で管理するから確かにコピー禁止のが無難かもね
どうせポインタ(orスマポ)で管理するから確かにコピー禁止のが無難かもね
581デフォルトの名無しさん
2021/04/30(金) 12:38:12.34ID:5yvxXz0O >>570
それぞれのコピコンをプログラマが自由に記述できる以上、例えAnimalの情報しか参照しないとしても問題が起こることはあるんじゃね?
例えば、DogのコピコンはAnimal::cuterThanCat変数を強制的にtrueに書き換えてるかもしれない
それぞれのコピコンをプログラマが自由に記述できる以上、例えAnimalの情報しか参照しないとしても問題が起こることはあるんじゃね?
例えば、DogのコピコンはAnimal::cuterThanCat変数を強制的にtrueに書き換えてるかもしれない
582デフォルトの名無しさん
2021/04/30(金) 12:51:39.69ID:W8up1rh0583デフォルトの名無しさん
2021/04/30(金) 14:49:19.98ID:qs5VuYRE 誰かBOOST_FOREACHを自作クラスに使う条件教えてください
584はちみつ餃子 ◆8X2XSCHEME
2021/04/30(金) 16:01:39.38ID:Efpw+/b3585デフォルトの名無しさん
2021/04/30(金) 20:24:33.87ID:QZYh0z4M >>579
>スライシングが起きるとき、基本クラス部分だけをコピーすることになるが、
>その際、仮想テーブルへのポインタが継承クラス用のままだと、コピー後の
>操作が危険
そんな恐ろしいことが!
と思って試したが、少なくともデフォルトのコピコンは仮想関数テーブルへのポインタを
コピー先の型に合わせて付け替えてくれるらしい
↓そうでなけれはfunc_with_ref(Animal a)の中でa2.get()がDogの値でなくAnimalの値を返す説明がつかない
https://ideone.com/eJMsYG
規格でどうなっているかは知らん
>スライシングが起きるとき、基本クラス部分だけをコピーすることになるが、
>その際、仮想テーブルへのポインタが継承クラス用のままだと、コピー後の
>操作が危険
そんな恐ろしいことが!
と思って試したが、少なくともデフォルトのコピコンは仮想関数テーブルへのポインタを
コピー先の型に合わせて付け替えてくれるらしい
↓そうでなけれはfunc_with_ref(Animal a)の中でa2.get()がDogの値でなくAnimalの値を返す説明がつかない
https://ideone.com/eJMsYG
規格でどうなっているかは知らん
586デフォルトの名無しさん
2021/04/30(金) 20:25:50.19ID:QZYh0z4M 訂正orz、
誤: そうでなけれはfunc_with_ref(Animal a)の中で
正: そうでなけれはfunc_with_copy(Animal a)の中で
誤: そうでなけれはfunc_with_ref(Animal a)の中で
正: そうでなけれはfunc_with_copy(Animal a)の中で
587デフォルトの名無しさん
2021/05/01(土) 07:00:40.64ID:2eVlBBCY ていうかよく考えたらAnimalのコピコンは
Animalのコンストラクタでもあるのだから>>585の挙動は当然かorz
Animalのコンストラクタでもあるのだから>>585の挙動は当然かorz
588デフォルトの名無しさん
2021/05/01(土) 09:21:42.94ID:C4kuj/yW コピコン
589デフォルトの名無しさん
2021/05/01(土) 09:28:33.15ID:18idEqJd コピコン
たまに見る
頭悪そう
たまに見る
頭悪そう
590デフォルトの名無しさん
2021/05/01(土) 10:19:04.33ID:tHuso9oJ でもコピーコンストラクターって長いよね
591デフォルトの名無しさん
2021/05/01(土) 10:26:47.50ID:2eVlBBCY コンストラクタはコンストラ
コピーコンストラクタはコピコン
デストラクタはデストラ
アルゴリズムはアルゴ
と略すのが効率的
コピーコンストラクタはコピコン
デストラクタはデストラ
アルゴリズムはアルゴ
と略すのが効率的
592デフォルトの名無しさん
2021/05/01(土) 10:31:05.38ID:Jen9oEOj 取引先がそんな言葉を使ってきたら
今後の契約を考え直すかも
少なくとも評価はマイナス
取引先とそんな細かい内容を話す打合せも無いだろうけど
今後の契約を考え直すかも
少なくとも評価はマイナス
取引先とそんな細かい内容を話す打合せも無いだろうけど
593デフォルトの名無しさん
2021/05/01(土) 11:21:44.35ID:fSkONWKY >>591
気持ち悪い略語だな
気持ち悪い略語だな
594デフォルトの名無しさん
2021/05/01(土) 12:04:39.12ID:zZz/KCNF デストラw
595デフォルトの名無しさん
2021/05/01(土) 12:14:30.82ID:yzll/vyD ctor,dtorは一般的な略語?
596デフォルトの名無しさん
2021/05/01(土) 12:14:41.90ID:toT74GP1 機能を引き継ぐために継承して、インスタンス化して使うために移譲もしたい
継承も移譲もするのってありですか?
継承も移譲もするのってありですか?
■ このスレッドは過去ログ倉庫に格納されています
ニュース
- 【地震速報】青森県で震度6強 沿岸部に津波警報 ★6 [ぐれ★]
- 「日の丸にバツ印」掲げた大学生 あいまいな国旗損壊罪に「怖い」 The Mainichi [少考さん★]
- 【音楽】BARBEE BOYS・KONTAが事故で四肢麻痺を公表、新体制で活動は継続 [少考さん★]
- 【野球】野球の未来に危機感「マイナースポーツになる」 宮本慎也氏が開催…学童大会 [尺アジ★]
- 中国「捜索レーダー起動は各国の通常の手法」 火器管制用か回答せず [蚤の市★]
- 【訃報】声優・西村知道さん死去 「SLAM DUNK」安西先生役 9月に体調不良のため一時休業 [少考さん★]
- 中国「日本のネトウヨを根絶してやるからな」。ネトウヨ、人生が終わる [805596214]
- おはようございます [577451214]
- (´・ω・`)刑務所での暮らしwwwwwwwwwwwwwwww
- ぺこーら、地震で同僚が次々配信を止めるなか強行し続けるので悪目立ちするwww [268244553]
- なぜ人間は架空の人物に感情移入するのか
- 【愛国者速報】山上徹也、金に困りTwitterのお金配り垢に応募していた。犯行もお金があったら暫くやらなかったと供述 [856698234]
