C++ primer 第十六章

2024-03-29 03:28
文章标签 c++ primer 第十六章

本文主要是介绍C++ primer 第十六章,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

模板是C++中泛型编程的基础,一个模板就是一个创建类或函数的蓝图。

1.定义模板

模板适用于唯一的差异是参数的类型,函数体完全一致的情况。

1.1、函数模板

我们可以定义一个通用的函数模板用来生成针对特定类型的函数版本

模板定义以关键字template开始,后跟模板参数列表

template<typename T> int compare(const T& v1, const T& v2)
{...}

在模板定义中,模板参数列表不能为空。 

模板参数列表类似于函数参数列表,模板参数表示在类或函数定义中用到的类型或值

当使用模板时,我们(隐式或显式)指定模板实参将其绑定到模板参数上。

当我们调用一个函数模板时,编译器通常用函数实参为我们推断模板类型。

编译器用推断出的模板参数来为我们实例化一个特定版本的函数。

compare(1,0); //实参类型为int,编译器推断出的类型为int,将它绑定到模板参数T上

这些由编译器生成的特定版本的函数通常被称为模板的实例。 

一般来说,可以将类型参数看作类型说明符,类型参数可以用来指定返回类型或函数的参数类型,以及在函数体内用于变量声明或类型转换

template<typename T> T foo(T* p)  //指定返回类型和函数参数类型
{T tmp = *p;  //声明变量return tmp;
}

每个类型参数前必须使用关键字class或typename。 

允许在模板中定义非类型参数一个非类型参数表示一个值而非一个类型

当一个模板被实例化时,非类型参数会被用户所提供的或编译器推断出的值(常量表达式)替代。

template<unsigned N, unsigned M> int compare(const char(&p1)[N],const char(&p2)[M]) {}
//N和M都是char数组的长度,是字面常量

一个非类型参数可以是一个整数,或者是一个指向对象或函数类型的指针或左值引用。

绑定到指针或引用非类型参数的实参必须具有静态的生存期(static变量)。

函数模板可以声明为inline或constexpr的,说明符放在模板参数列表之后,返回类型之前。

template<typename T> inline T min(const T&, const T&);

模板程序应该尽量减少实参类型的要求。 

只有当我们实例化出模板的一个特定版本时,编译器才会生成代码。

我们将类定义和函数声明放在头文件中,而普通函数和类的成员函数的定义放在源文件中。

函数模板和类模板成员函数的定义通常放在头文件中。

模板直到实例化时才会生成代码,因此大多数编译错误在实例化期间报告。

作为调用者应保证传递给模板的实参支持模板所要求的操作,以及操作能在模板内正确运行。

1.2、类模板

与函数模板不同,编译器不能为类模板推断模板参数类型,我们需要指定模板的具体类型。

template<typename T> class blob{...
};blob<int> a;   //空blob<int>
blob<string> b = {"hi","hello","good"}; //有3个元素的blob<string>

类模板的每个实例都形成一个独立的类,不会与其他类有任何关联,也不会有特殊访问权限。

我们可以在类模板内部,也可以在类模板外部为其定义成员函数,且定义在内部的函数被隐式声明为内联函数。

类模板的成员函数本身是一个普通函数,但类模板的每个实例都有自己版本的成员函数。

定义在类模板之外的成员函数必须以关键字template开始,后接类模板参数列表。

template<typename T> void blob<T>::compare(const T&,constT&) {...}
//定义在类外部的成员函数compare

与其他任何定义在类模板外的成员一样,构造函数的定义要以模板参数开始。

template<typename T> blob<T>::blob() : data(std::make_shared<std::vector<T>>()) {}

默认情况下,一个类模板的成员函数只有当程序用到它时才会进行实例化。 

当我们使用一个类模板类型时必须提供模板实参,但在类模板自己的作用域中,我们可以直接使用模板名而不提供实参。

template<typename T> class blobstr{//...blobstr& operator++(); //返回的是blobstr&,而不是blobstr<T>&blobstr& operator--();
};

当我们在类模板外定义其成员时,遇到类名时表示进入类的作用域,因此在函数体内可直接使用模板名而不必指明模板实参。

template<typename T> blobstr<T> blobstr<T>::operator(int)
{  blobstr ret = *this;   //直接使用模板名++*this;return ret;
}

 若一个类模板包含一个非模板友元,则友元被授权可以访问所有的模板实例。

若友元本身为模板,则类可以授权所有的友元模板实例,也可以只授权给特定实例。 

友元的声明用类模板的模板形参作为自己的模板实参,则友元关系被限定在用相同类型实例化的版本。

template<typename> class blobstr;  //前置声明,在blob中声明友元所需要的
template<typename> class blob;
template<typename T> class blob{friend class blobstr<T>;   //友元关系限定在相同类型T的版本...
};

友元的声明使用与类本身不同的模板参数,则类将类模板的每个实例声明为自己的友元。 

class total{template<typename T> friend class pal;//pal的所有实例都是total的友元,这种情况无须前置声明
};

在新标准中,我们可以将模板类型参数声明为友元。

template<typename type> class bar{friend type;   //将访问权限授予用来实例化bar的类型...
};
//对于某个类型名,该类将成为bar<type>的友元

类模板的一个实例定义了一个类类型,因此可以通过typedef来引用已实例化的类。

由于模板不是一个类型,我们不能定义一个typedef来引用一个模板。

typedef blob<T> strblob; //错误

 新标准允许我们为类模板定义一个类型别名,一个模板类型别名是一族类的别名。

template<typename T> using twin = pair<T,T>;
twin<string> authors; //authors是一个pair<string,string>

当我们定义一个模板类型别名时,可以固定一个或多个模板参数。

template<typename T> using part = pair<T,unsigned>;
part<string> book;  //book是一个pair<string,unsigned>

类模板可以声明static成员。

模板类的每个static数据成员必须有且只有一个定义,但是类模板的每个实例都有一个独有的static对象。

template<typename T> class foo{
private:static std::size_t n;
};foo<string> fs;  //实例化static成员foo<string>::n
foo<int> is;  //实例化static成员foo<int>::n

我们可以通过类类型对象来访问一个类模板的static成员,使用作用域运算符直接访问成员。

foo<int> fi;
auto ct = foo<int>::n; //实例化foo<int>::n
ct = foo.n; 

1.3、模板参数 

一个模板参数名的可用范围是在其声明之后,至模板声明或定义结束之前。

模板参数会隐藏外层作用域中生命的相同的名字,在模板内不能重用模板参数名

typedef double A;
template<typename A,typename B> void f(A a,B b)
{A tmp = a;   //tmp的类型为模板参数A的类型,而非doubledouble B;   //错误,重声明模板参数B
}

模板的声明必须包含模板参数,定义中的模板参数的名字不必与声明中相同。 

template<typename T> int compare(const T&, const T&); //模板声明template<typename type> type compare(const type& a,const type& b) {...} //模板定义

一个给定模板的每个声明和定义必须有相同数量和种类(类型或非类型)的参数。

一个特定文件所需要的所有模板的声明通常一起放置在文件开始位置。

为了处理模板,编译器必须知道名字表示一个类型还是一个数据成员。

T::size_type * p
//编译器不知道size_type是模板T的类型成员还是一个数据成员

 为了解决上述问题,C++语言默认假定通过作用域运算符访问的名字是数据成员

 若我们希望通知编译器一个名字表示类型时,必须使用关键字typename来实现。

template<typename T> template T::value_type top(const T& c) //显式表示value_type是类型
{ return typename T::value_type(); }

在新标准中,我们可以为函数和类模板提供默认实参。

template<typename T, typename F = less<T>> int compare(const T& v1,const T& v2,F f = F())
{} //给类型F提供默认实参

与函数默认实参一致,对于一个模板参数,只有当它右侧的所有参数都有默认实参时,它才能有默认实参。

无论何时使用一个类模板,我们都必须在模板名之后接上尖括号。

若我们希望使用默认实参,则在模板名之后跟一个空尖括号即可。

template<class T = int> class numbers {...};numbers<> average;   //T为int
numbers<double> prec; //T为double

 1.4、成员模板

 类(普通类或类模板)可以包含本身是模板的成员函数(成员模板),成员模板不能是虚函数

class debugdelete{
public:template<typename T> void operator()(T* p) const{ os << "delete ptr" << std::endl;delete p;}
};double* p = new double;  
debugdelete d;  //创建对象
d(p);  //实例化operator()(double*)版本,释放p
int* ip = new int;
debugdelete()(ip); //在一个临时的debugdelete对象上调用operator()(int*)

 对于类模板,我们可以为其定义成员模板。

template<typename T> class blob{template<typename It> blob{It b,It e); //构造函数接受不同类型的迭代器
};

当我们在类模板外定义一个成员模板时,必须同时为类模板和成员模板提供模板参数列表。 

template<typename T> template<typename It>
blob<T>::blob(It b,It e) : data(std::make_shared<std::vector<T>>(b,e)) {}
//类模板的参数列表在前,后跟成员自己的模板参数列表

为了实例化一个类模板的成员模板,我们必须同时提供类和函数模板的实参。

int ia[] = {1,2,3,4,5,6};
blob<int> al(begin(ia),end(ia)); //a1的定义实例化了blob<int>::blob(int*,int*)的版本

 1.5、控制实例化

当两个或多个独立编译的源文件使用了相同的模板,并提供了相同的模板参数时,每个文件中就都会有该模板的一个实例,造成了额外的开销。

在新标准中,我们可以通过显式实例化来避免这种问题的发生。

当编译器遇到extern模板声明时,它不会在本文件中生成实例化代码,但这些模板类型必须在程序其他位置(文件)进行实例化。

extern template class blob<string>;  //blob<string>的代码不会在本文件生成
blob<string> sa1,sa2; //实例化会在其他位置出现

对于一个给定的实例化版本,可能有多个extern声明,但必须只有一个定义。 

 对于每个实例化声明,在程序中某个位置必须有其显式的实例化定义。

一个类模板的实例化定义会实例化该模板的所有成员,包括内联的成员函数。

2.模板实参推断

对于函数模板,编译器从函数实参来确定模板实参的过程被称为模板实参推断

2.1、类型转换与模板类型参数

编译器通常不是对实参进行类型转换,而是直接生成一个新的模板实例。

在调用时自动应用于函数模板的类型转换有如下两项:

  • const转换:可以将一个非const对象的引用(指针)传递给一个const的引用(指针)形参。
  • 数组或函数指针转换:若函数形参不是引用类型,则可以对数组或函数类型的实参应用到正常的指针转换。(数组实参转换为指向首元素的指针、函数实参转换为函数类型的指针)
template<typename T> T foa(T,T);
template<Typename T> T fob(const T&,const T&);string s1("hello");
const string s2("world");
foa(s1,s2); //调用foa(string,string);const被忽略
fob(s1,s2); //调用fob(const string&,const string&); s1转换为const的int a[5];
int b[10];
foa(a,b);  //调用foa(int*,int*); 数组实参转换为指向首元素的指针
fob(a,b);  //错误,两个数组大小不同,因此类型不同,数组类型不匹配

 一个模板类型参数可以用作多个函数形参的类型。

 由于只允许有限的几种类型转换,因此传递给函数形参的实参必须具有相同的类型。

long lg;
compare(lg,1024); //错误,不能实例化compare(long,int)

若希望对函数实参进行正常的类型转换,可以将函数模板定义为两个类型参数。 

template<typename A, typename B> int compare(const A& v1, const B& v2) {...}long lg;
compare(lg,1024); //正确,调用compare(long,int)

若函数参数类型不是模板参数,则对实参进行正常的类型转换。

template<typename T> ostream& print(ostream& os,const T& obj)
{ return os << obj; }print(cout,42);  //实例化print(ostream&,int)
ofstream f("output");
print(f,10); //将ofstream转换为ostream&

2.2、函数模板显式实参 

当函数返回类型与参数列表中任何类型都不相同时,编译器将无法推断出模板实参的类型。

我们可以通过定义表示返回类型的第三个模板参数,从而允许用户控制返回类型。

template<typename T1,typename T2,typename T3> T1 sum(T2,T3);
//T1作为返回类型,编译器无法推断T1,它未出现在函数参数列表中

在sum中,没有任何函数实参的类型可用来推断T1的类型,因此每次调用sum时都必须为T1提供了一个显式模板实参

auto val = sum<long long>(1024,2.30); //long long sum(int,double)

显式模板实参按由左至右的顺序与对应的模板参数匹配,只有最右侧参数的显式模板参数可忽略。

对于模板类型参数已经显式指定了的函数实参,也可以进行正常的类型转换。

long lg;
compare(lg,1024); //错误,模板参数不匹配
compare<long>(lg,1024); //正确,显式指定模板类型参数,实例化compare(long,long);

2.3、尾置返回类型与类型转换

当我们知道函数的返回类型与所处理的序列的元素类型相同,可以使用尾置操作来指定返回类型。

尾置返回出现在参数列表之后,它可以直接使用函数的参数。

//尾置返回允许我们在参数列表之后声明返回类型
template<typename T> auto fcn(It beg,It end)-> decltype(*beg)
{ return *beg; }

所有迭代器操作都不会生成元素,只能生成元素的引用,为了得到元素类型,可以使用标准库中的类型转换模板

类型转换模板定义在头文件type_traits中。

标准类型转换模板
对mod<T>,其中mod为若T为则mod<T>::type为
remove_reference

X&或X&&

否则

X

T

add_const

X&,const X或函数

否则

T

const T

add_lvalue_reference

X&

X&&

否则

T

X&
T&

add_rvalue_reference

X&或X&&

否则

T

T&&

remove_pointer

X*

否则

X

T

add_pointer

X&或X&&

否则

X*

T*

make_signed

unsigned X

否则

X
T
make_unsigned

带符号类型

否则

unsigned X

T

remove_extent

X[n]

否则

X

T

remove_all_extents

X[n1][n2]

否则

X

T

template<typename It> 
auto fcn2(It beg,It end) -> typename remove_reference<decltype(*beg)>::type
{ return *beg; }
//remove_reference::type脱去引用,剩下元素的类型本身
//在返回类型的声明中使用typename来告知编译器,type是一个类型

 2.4、函数指针和实参推断

用一个函数模板初始化一个函数指针或为一个函数指针赋值时,编译器使用指针的类型来推断模板实参。

template<typename T> int compare(const T&,const T&);
int (*p)(const int&,const int&) = compare;
//p中参数的类型决定了T的模板实参的类型,T的模板实参类型为int

若不能从函数指针类型确定模板实参,则将产生错误。

当参数是一个函数模板实例的地址时,对于每个模板参数,必须唯一确定其类型或值

void func(int(*)(const int&, const int&));
void func(int(*)(const string&, const string&));func(compare); //错误,通过func的参数类型无法确定模板实参的唯一类型func(compare<int>); //正确,通过使用显式模板实参来消除func调用的歧义

2.5、模板实参推断和引用 

编译器会应用正常的引用绑定规则。

当一个函数参数时模板类型参数的一个普通左值引用时,只能传递给它一个左值。

template<typename T> void f1(T&); //实参必须是一个左值
f1(i);  //i是一个int,模板参数类型T为int
f1(ci); //ci是一个const int,模板参数类型为const int
f1(5);  //错误,实参不是左值

当函数参数本身是const时,T的类型推断的结果不会是一个const类型,const已经是函数参数类型的一部分。 

template<typename T> void f2(const T&);
f2(i);  //i是一个int,模板参数T推断为int
f2(ci); //ci是一个const int,模板参数T推断为int
f2(5);  //一个const引用参数可以绑定到一个右值;T是int

当一个函数参数是一个右值引用,我们可以传递一个右值给它,推断出的T的类型是该右值实参的类型。

template<typename T> void f3(T&&);
f3(5); //模板参数T是int

 通常我们不能将一个右值引用绑定到一个左值上,但C++语言定义了两个例外规则:

  • 当我们将一个右值传递给函数的右值引用参数,且该右值引用指向模板类型参数时,编译器推断模板类型参数为实参的右值引用类型。(形成了一个引用的引用 int& &&)
  • 若我们间接创建了一个引用的引用,这些引用形成了“折叠”,通过折叠规则来处理上述问题。

 折叠规则:

  • X& &、X& &&、X&& &  ---> X&
  • 类型X&& && ---> X&&

推断出的模板类型参数 + 自带的函数参数 = 实际上用到的实例化类型参数

引用折叠只能应用于间接创建的引用的引用,例如类型别名或模板参数。

template<typename T> void f3(T&&);f3(i);   //实参是一个左值,模板参数T是int&
f3(ci);  //实参是一个左值,模板参数T是const int&

这两个例外规则暗示着可以将任意类型的实参传递给T&&类型的函数参数。

2.6、理解std::move

标准库中的move函数接受一个左值,并获得一个绑定到左值上的右值引用。

标准库中的move函数的定义:

//通过引用折叠,函数参数可以与左值或右值的实参匹配
template<typename T> typename remove_reference<T>::type&& move(T&& t)
{//typename表示作用域运算符后的type是类型,而不是数据return static_cast<typename remove_reference<T>::type&&>(t);//typename remove_reference<T>::type&& 去除引用,并加上右值引用//static_cast 改变t的引用类型
}

不能隐式地将一个左值转换成右值引用,但可以用static_cast显式地将一个左值转换成一个右值引用。

2.7、转发 

某些函数需要将其一个或多个实参连同类型不变地转发给其他函数,需要保持被转发实参的所有性质。

若一个函数参数是指向模板类型参数的右值引用,它对应的实参的const属性和左值/右值属性都会得到保持。

template<typename F,typename T1,typename T2> void flip(F f,T1&& t1,T2&& t2)
{ f(t2,t1); } void g(int&& i,int& j) { cout << i << " " << j << endl; }

当我们试图通过flip来调用g时,参数t2将被传递给g的右值引用参数,将会发生错误。

flip(g,i,42); //错误,不能从一个左值实例化int&&
//t2是一个有名称的变量,有身份的右值表达式是左值

在调用中使用std::forward的新标准库设施来传递flip的参数,它能保持原始实参的类型

forward必须通过显式模板实参来调用。

当用于一个指向模板参数类型的右值引用函数参数时,forward会保持实参类型的所有细节。

template<typename F,typename T1,typename T2>void flip(F f,T1&& t1,T2&& t2)
{ f(std::forward<T2>(t2),std::forward<T1>(t1));
}flip(g,i,42);
//i将以int&类型传递给g,42将以int&&类型传递给g

3.重载与模板 

函数模板可以被另一个模板或一个普通非模板函数重载。

若有多个函数提供同样好的匹配,则:

  • 如果同样好的函数只有一个非模板函数,则选择此函数。(非模板版本优先
  • 如果同样好的函数没有非模板函数,而有多个函数模板,则其中一个模板比其他模板更特例化,则选择该模板。(特例版本优先
template<typename T> string debug_rep(const T& t) {...} //接受一个const对象的引用
template<typename T> string debug_rep(T* p) {...}  //接受一个模板参数T的指针string s("hello");
cout << debug_rep(&s) << endl; 
//第一个版本提供debug_rep(const string*&); 第二个版本提供debug_rep(string*)
//第一个版本需要进行普通指针到const指针的转换,而第二个版本是精确匹配

数组到指针的转换的操作对于函数匹配来说,这种转换被认为是精确匹配的。

在定义任何函数之前,记得声明所有重载的函数版本,这样就不必担心编译器由于未遇到你希望调用的函数而实例化一个并非你所需的版本。

4.可变参数模板

一个可变参数模板就是一个接受可变数量参数的模板函数或模板类。

可变数量参数被称为参数包(模板参数包或函数参数包)。

使用一个省略号来指出一个模板参数或函数参数表示一个包。

在函数参数列表中,若一个参数的类型是一个模板参数包,则此参数也是一个函数参数包。

template<typename T,typename ...arg> void foo(const T& t,const arg& ...rest);
//arg是一个模板参数包,rest是一个函数参数包

对于一个可变参数模板,编译器会推断包中参数的数目。

当我们需要知道包中有多少元素时,可以使用sizeof...运算符来计算包中的元素。

cout << sizeof...(arg) << endl;  //类型参数的数目
cout << sizeof..(rest) << endl; //函数参数的数目

4.1、编写可变参数函数模板 

可变参数函数通常是递归的。

先调用处理包中第一个实参,然后用剩余实参调用自身,为了终止递归,还需定义非可变参数的函数。

template<typename T> ostream& print(ostream& os,const T& t) //用来终止递归
{  return os << t; }template<typename T,typename ...args>
ostream& print(ostream& os,const T& t,const args&...rest)
{os << t << " ";  //打印第一个实参return print(os,rest...);  //递归调用,打印其他实参//rest中的第一个实参被绑定t,剩余实参形成下一个print调用的参数包
}

当定义可变参数版本的print时,非可变参数版本的声明必须在作用域中。否则可变参数将会无限递归。

4.2、包扩展

对于一个参数包,只有获取其大小的操作和扩展它的操作。

扩展一个包就是将它分解为构成的元素,对每个元素应用模式,获得扩展后的列表。

在模式右边放置一个省略号(...)来触发扩展操作。

return print(os,rest...); //扩展rest

C++语言允许更复杂的扩展模式,编写第二个可变参数函数,对每个实参调用这个可变参数函数

return print(os,debug_rep(rest)...);
//上式等于 print(os,debug_rep(a1),debug_rep(a2),...,debug_rep(an))
//扩展结果将是一个逗号分隔的debug_rep调用列表

扩展中的模式会独立地应用于包中的每个元素。 

 4.3、转发参数包

在新标准下,我们可以组合使用可变参数模板forward机制来编写函数,实现其实参不变地传递给其他函数

//每个函数参数都是一个指向其对应实参的右值引用
template<class ...args> inline emplace_back(args&&...arg)
{  alloc.construct(first_free++,std::forward<args>(arg)...);  }
//forward<args>(arg)... 即扩展了模板参数包args,也扩展了函数参数包argemplace_back(10,'c');
//扩展出 std::forward<int>(10)  std::forward<char>(c)

可变参数函数通常将它们的参数转发给其他函数。

5.模板特例化 

当我们不希望使用模板版本时,可以定义类或函数模板的一个特例化版本。

template<size_t n,size_t m> int compare(const char(&)[n],const char(&)[m]);
//专门处理字符串字面常量,函数参数都是非类型模板参数

一个模板特例化版本就是模板的一个独立的定义,在其中一个或多个模板参数被指定为特定的类型。 

当我们特例化一个函数模板时,必须为原模板中的每个模板参数都提供实参。

使用关键字template后跟一个空尖括号来指出正在实例化一个模板,这操作使得该特例化版本与模板保持相同的优先级。 

template<> int compare(const char* const &p1,const char* const &p2) {}
//compare的特例化版本,处理字符数组的指针

一个特例化版本本质上是一个模板的实例,而非函数名的一个重载版本。 

模板及其特例化版本应该声明在同一个头文件中。所有同名模板的声明应放在前面,然后是这些模板的特例化版本。

除了特例化函数模板,还可以特例化类模板。

template<> struct hash<sales_data> {......};//特例化一个sales_data版本的hash

类似其他任何类,我们可以在类内或类外定义特例化版本的成员。

类模板的特例化不必为所有模板参数提供实参,因此类模板的部分特例化本身是一个模板。 

template<class T> struct remove_reference {...};  //通用模板
template<class T> struct remove_reference<T&> {...}; //左值引用版本
template<class T> struct remove_reference<T&&> {...}; //右值引用版本

部分特例化版本的模板参数列表是原始模板的参数列表的一个子集或是一个特例化版本。

允许只特例化特定成员函数而不是特例化整个模板。

template<typename T> struct foo
{foo(struct T& t = T()):mem(t) {}void bar() {...}
};template<> void foo<int>::bar() {...} //先特例化int模板,再特例化foo<int>的成员bar

这篇关于C++ primer 第十六章的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/857472

相关文章

C++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)