字符串模板参数c

前端之家收集整理的这篇文章主要介绍了字符串模板参数c前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
在c中使用模板时,我有时需要传递字符串作为值模板参数.

我发现很难理解为什么某些参数是允许的,而其他参数是不允许的.

例如,如果一个类的静态成员,const char *可以作为模板参数给出,如果在外部定义,则不能.

我做了一个小程序来测试所有这些,评论不编译的行.我也根据编译器输出做了几个假设,但是它们可能是错误的.

模板参数值的规则是什么?我看到对象需要外部链接,但是bool是授权的,虽然它显然没有任何形式的联动.

  1. #include <iostream>
  2. using namespace std;
  3.  
  4. struct tag {
  5. static char array[];
  6. static const char carray[];
  7. static char *ptr;
  8. static const char *cptr;
  9. static const char *const cptrc;
  10. static string str;
  11. static const string cstr;
  12. };
  13. char tag::array[] = "array";
  14. const char tag::carray[] = "carray";
  15. char *tag::ptr = (char*)"ptr"; // cast because deprecated conversion
  16. const char *tag::cptr = "cptr";
  17. const char *const tag::cptrc = "cptrc";
  18. string tag::str = "str";
  19. const string tag::cstr = "cstr";
  20.  
  21.  
  22. namespace ntag {
  23. char array[] = "array";
  24. const char carray[] = "carray";
  25. char *ptr = (char *)"ptr"; // cast because deprecated conversion
  26. const char *cptr = "cptr";
  27. const char *const cptrc = "cptrc";
  28. string str = "str";
  29. const string cstr = "cstr";
  30. };
  31.  
  32. template <class T,T t>
  33. void print() { cout << t << endl; };
  34.  
  35. int main()
  36. {
  37. cout << "-- class --" << endl;
  38. // Works
  39. print<char *,tag::array>();
  40. print<const char *,tag::carray>();
  41.  
  42. // Does not work because it is a lvalue ?
  43. // print<char *,tag::ptr>();
  44. // print<const char *,tag::cptr>();
  45. // print<const char *const,tag::cptrc>();
  46.  
  47. // Template type param must be a basic type ?
  48. // print<string,tag::str>();
  49. // print<const string*,tag::cstr>();
  50.  
  51. cout << "-- namespace --" << endl;
  52. // Works
  53. print<char *,ntag::array>();
  54.  
  55. // No external linkage ?
  56. // print<const char *,ntag::carray>();
  57.  
  58. // Does not work because it is an lvalue ?
  59. // print<char *,ntag::ptr>();
  60. // print<const char *,ntag::cptr>();
  61. // print<const char *const,ntag::cptrc>();
  62.  
  63. // The type of a template value param must a basic type
  64. // print<string,ntag::str>();
  65. // print<const string*,ntag::cstr>();
  66. }

解决方法

当使用非类型的模板参数时,需要指定一个常量.当非类型模板参数是指针或引用时,只要指定可以在链接时确定的常数即可.在任何情况下,编译器将不接受在链接时间之后可能被突变的任何东西.在链接时间初始化的偶变量初始化太晚了:
  1. print<char *,tag::array>(); // OK: the address of the array won't change
  2. print<const char *,tag::carray>(); // OK: the address of the array won't change
  3. print<char *,tag::ptr>(); // not OK: tag::ptr can change
  4. print<const char *,tag::cptr>(); // not OK: tag::ptr can change
  5. print<const char *const,tag::cptrc>(); // not OK: a [run-time initialized] variable
  6. print<string,tag::str>(); // not OK: few types are supported (*)
  7. print<const string*,tag::cstr>(); // not OK: tag::cstr has a different type
  8. print<const string*,&tag::cstr>(); // (added) OK: address won't change
  9.  
  10. print<char *,ntag::array>(); // OK: address of array won't change
  11. print<const char *,ntag::carray>(); // OK: address of array won't change (**)
  12. print<char *,ntag::ptr>(); // not OK: ntag::ptr can change
  13. print<const char *,ntag::cptr>(); // not OK: ntag::cptr can change
  14. print<const char *const,ntag::cptrc>(); // not OK: a [run-time initialized] variable
  15.  
  16. print<string,ntag::str>(); // not OK: few types are supported (*)
  17. print<const string*,ntag::cstr>(); // not OK: ntag::cstr has a different type
  18. print<const string*,&ntag::cstr>(); // (added) OK: address won't change

笔记:

>(*)只能使用非类型模板参数的整型类型,指针和引用.没有用户定义常量的概念,可以用作模板参数.
>(**)gcc不喜欢这个用法,而clang喜欢它. gcc不接受这个代码似乎是一个错误!我看不到任何禁止使用const char []作为模板参数的限制.相反,在14.3.2 [temp.arg.nontype]段落2中有一个例子,它完全相同:

  1. template<class T,const char* p> class X {
  2. / ... /
  3. };
  4. X<int,"Studebaker"> x1; // error: string literal as template-argument
  5. const char p[] = "Vivisectionist";
  6. X<int,p> x2; // OK

>将字符串文字转换为非const指向char的指针是OK,但是,尝试更改其中一个值是未定义的行为.我强烈建议不要使用这个演员!
> Don’t overuse std::endl:在你的代码中,根本不用std :: endl.

猜你在找的C&C++相关文章