質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Q&A

解決済

2回答

1185閲覧

C++20 コンセプトを使った部分特殊化のテンプレートについて

bjnes

総合スコア113

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

0グッド

1クリップ

投稿2020/08/06 09:03

前提・実現したいこと

基本型のMIN,MAX値が.NETのようにまとまってないことに不便を感じたので、コンセプトとテンプレートの練習も兼ねて書いてみることにしました。

基本的なコンストラクタなどは同じ振る舞いなので、共通化して、BoolとINTで演算系が違うのでそれぞれ特殊化する方針で考えています。

それが、以下のソースになります。

該当のソースコード

C++

1namespace ClassedValue{ 2 /** 3 * @brief 整数型(Boolを除く)であることを示すコンセプトです。 4 * 5 * @tparam T 調査対象クラス 6 */ 7 template <class T> 8 concept IsIntegerWithoutBool = std::is_integral_v<T> && !std::is_same_v<bool, T>; 9 10 11   /*基本テンプレート*/ 12   template <class T> 13 class BasicValueObject final 14 { 15 private: 16 T value; 17 18 constexpr BasicValueObject() noexcept : value(T{}){}; 19 constexpr BasicValueObject(const BasicValueObject& obj) noexcept : value(obj.value){}; 20 constexpr BasicValueObject(const T& obj) noexcept : value(obj){}; 21 }; 22 23 24 template <> 25 class BasicValueObject<bool> 26 { 27 private: 28 bool value; 29     /*中略 BOOL系演算定義 */ 30 }; 31 32  /*整数型はこの部分特殊化を経由してほしい*/ 33 template <IsIntegerWithoutBool T, T& Min, T& Max> 34 class BasicValueObject<T>              //ここでエラー 35 { 36     public: 37 constexpr T Minvalue = Min; 38 constexpr T Maxvalue = Max; 39    /*中略 整数系演算定義 */ 4041 42 //こんなかんじで定義したい 43 using Bool = BasicValueObject<bool>; 44 using Int8 = BasicValueObject<int8_t, INT8_MIN, INT8_MAX>; 45 using Uint8 = BasicValueObject<uint8_t, UINT8_MIN, UINT8_MAX>; 46 using Byte = Uint8; 47 using Int16 = BasicValueObject<int16_t, INT16_MIN, INT16_MAX>; 48 using Uint16 = BasicValueObject<uint16_t, UINT16_MIN, UINT16_MAX>; 49 using Int32 = BasicValueObject<int32_t, INT32_MIN, INT32_MAX>; 50 using Uint32 = BasicValueObject<uint32_t, UINT32_MIN, UINT32_MAX>; 51 using Int32 = BasicValueObject<int32_t, INT32_MIN, INT32_MAX>; 52 using Uint32 = BasicValueObject<uint32_t, UINT32_MIN, UINT32_MAX>; 53}

発生している問題・エラーメッセージ

template parameters not deducible in partial specialization:GCC

参考URL

以下のページを参考にしています。
https://programming-place.net/ppp/contents/cpp/language/023.html#partial_specialization

補足情報(FW/ツールのバージョンなど)

COLLECT_LTO_WRAPPER=/usr/lib/gcc/x86_64-pc-linux-gnu/10.1.0/lto-wrapper
ターゲット: x86_64-pc-linux-gnu
configure 設定: /build/gcc/src/gcc/configure --prefix=/usr --libdir=/usr/lib --libexecdir=/usr/lib --mandir=/usr/share/man --infodir=/usr/share/info --with-bugurl=https://bugs.archlinux.org/ --enable-languages=c,c++,ada,fortran,go,lto,objc,obj-c++,d --with-isl --with-linker-hash-style=gnu --with-system-zlib --enable-__cxa_atexit --enable-cet=auto --enable-checking=release --enable-clocale=gnu --enable-default-pie --enable-default-ssp --enable-gnu-indirect-function --enable-gnu-unique-object --enable-install-libiberty --enable-linker-build-id --enable-lto --enable-multilib --enable-plugin --enable-shared --enable-threads=posix --disable-libssp --disable-libstdcxx-pch --disable-libunwind-exceptions --disable-werror gdc_include_dir=/usr/include/dlang/gdc
スレッドモデル: posix
Supported LTO compression algorithms: zlib zstd
gcc バージョン 10.1.0 (GCC)

よろしくおねがいします。

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

bjnes

2020/08/06 09:37 編集

numeric_limitsの情報ありがとうございます。 テンプレートとコンセプトの練習も兼ねているので、 できれば、なぜこのエラーが出ているのか知りたいです。 部分特殊化ができない人にはなりたくないですので;;;
guest

回答2

0

ご指摘ありがとうございました。

まず、”部分”特殊化について、私の理解が間違っていました。

1 .
誤:プライマリーテンプレート実装の”一部分”をテンプレート引数に基づいて、改変してくれる。
正:プライマリーテンプレート引数の”一部分”に基づいて、実装の中身すべてを改変する。⇒中身に互換性はない(参考:URL

2 .
コンストラクタなどを共通化させたい場合は継承を用いる。
参考:継承コンストラクタ

3 .以下のような可変長の引数を扱いたい場合は先にプライマリー
テンプレートを定義する。

c++

1 2template <class T, int ...args>class BasicValueObject; 3 4template<> 5class BasicValueObject<bool>{/*中略*/}; 6 7template<std::integral T,int min,int max> 8class BasicValueObject<T,min,max>{/*中略*/}; 9 10 using Bool = BasicValueObject<bool>; 11 using Int8 = BasicValueObject<int8_t, INT8_MIN, INT8_MAX>

投稿2020/08/07 06:13

bjnes

総合スコア113

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

0

ベストアンサー

質問文中ソースコードの問題を再現する最小化したコードです。
https://wandbox.org/permlink/foQ46es3IbRPKGip

C++

1#include <concepts> 2 3// プライマリテンプレート 4template <typename T> 5struct S {}; 6 7// 特殊化 8template <> 9struct S<bool> {}; 10 11// 部分特殊化 12template <typename E, std::size_t N> 13struct S<E[N]> {}; 14 15// エラー!! 16template <std::integral T, T Value> 17struct S<T> {}; 18// error: template parameters not deducible in partial specialization: 19// note: 'Value'

テンプレートの特殊化では、Valueのように独立したテンプレートパラメータを追加することはできません。

投稿2020/08/06 09:51

編集2020/08/06 09:52
yohhoy

総合スコア6191

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

bjnes

2020/08/06 10:18

ということは、 using Bool = BasicValueObject<bool>; using Int8 = BasicValueObject<int8_t, INT8_MIN, INT8_MAX>; を定義できないということでしょうか?
bjnes

2020/08/06 12:34

個人的な感覚なのですが、BOOLがMIN,MAXを持つのが気持ち悪いのです。 あと、ToStringとか、演算とか実装しだすとなんか違う気がします。
yohhoy

2020/08/07 02:23 編集

> ということは、 > using Bool = BasicValueObject<bool>; > using Int8 = BasicValueObject<int8_t, INT8_MIN, INT8_MAX>; > を定義できないということでしょうか? そもそもクラステンプレート定義がエラーとなっていますから、結果的にそうですね。 質問者さんの実現したいことを理解していないので確かなことは言えませんが、yumetodoさんも指摘するように部分特殊化が適切なケースには見えません。 > 基本的なコンストラクタなどは同じ振る舞いなので、共通化して とありますが、プライマリテンプレートと部分特殊化されたテンプレートの間で "共通化される実装(処理)" は何もありません。 部分特殊化の主眼は、プライマリテンプレートとは異なる実装やインタフェースを与えることにあります。 実装(処理)の共通化が目的であれば、クラス継承(is-a, is-implemented-in-terms-of)ないしコンポジット(has-a)が自然です。
yumetodo

2020/08/07 00:53

>BOOLがMIN,MAXを持つのが気持ち悪いのです。 そこまでいうならBooleanは算術演算できないってことでそもそも除外してしまうほうがいいのでは。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問