C++基础-(模板及标准模板库)

C++基础

模板及标准模板库

  • 模板的作用
    • 模板使程序员能够快速的建立具有类型安全得库集合和函数集合,它的实现,方便了大规模的软件开发。
  • 简单的求最大值
#include <iostream>
using namespace std;
template <class X>
X Max(X a,X b)
{
    return (a>b?a:b);
}
int main()
{
    int x1=20,x2=100;
    cout<<"max="<<Max<double>(x1,x2)<<endl;
    double y1=22.3,y2=100.2;
    cout<<"max="<<Max<double>(y1,y2)<<endl;
    char z1='A',z2='z';
    cout<<"max="<<Max<char>(z1,z2)<<endl;
}
  • 模板定义
    • 模板的定义很特殊,由 template<…> 处理的任何东西都意味着编译器在当时不为它分配存储空间,它一直处于等待状态直到被一个模板实例告知。所以为了容易使用,几乎总是在头文件中放置全部的模板声明和定义
#include <iostream>
using namespace std;
template <class X,class Y>
class Test
{
    X m_t1;
    Y m_t2;
    public:
        Test(X t1,Y t2)
        {
            m_t1=t1;
            m_t2=t2;
        }
        void show()
        {
            cout<<"T1="<<m_t1<<"T2="<<m_t2<<endl;
        }
        void print();
};
template <class X,class Y>
void Test<X,Y>::print()
{
    cout<<"t1="<<m_t1<<"t2="<<m_t2<<endl;
}
int main()
{
    Test<int,char> t(10,'s');
    t.show();
    t.print();
}
  • 类模板

    • 使用类模板使用户可以为类声明一种模式,使得类中的某些数据成员、某些成员函数的参数、某些成员函数的返回值,能取任意类型(包括基本类型的和用户自定义类型)。
    • 定义
    template <模板参数表>
    class 类名
    {类成员声明}
    
    • 在类模板以外定义其成员函数:
    template <模板参数表>
    类型名 类名<T>::函数名 ( 参数表 )
    
  • 类模板与模板类的区别。

    • 类模板是模板的定义,不是一个实实在在的类,定义中用到通用类型参数。
    • 模板类是实实在在的类定义,是类模板的实例化。类定义中参数被实际类型所代替。
  • 标准模板类

    • 将程序写的尽可能通用
    • 将算法从特定的数据结构中抽象出来,成为通用的
    • C++的模板为泛型的程序设计奠定了关键的基础
    • STL是泛型程序设计的一个范例
      • 容器(container)
      • 迭代器(itetator)
      • 算法(algorithms)
      • 函数对象(function object)
  • 容器是容纳包含一组元素集合的对象

  • 基本容器

    • 向量(vector)
    • 双端队列(deque)
    • 列表(list)
    • 集合(set)
    • 多重集合(multiset)
    • 映射(map)
    • 多重映射(multimap)
  • 容器的接口

    • 通用容器运算符
      • ==,!=,>,>=,<,<=,=
    • 方法(函数)
      • 迭代方法
        • begin(),end(),rbegin(),rend()
      • 访问方法
        • size(),max_size(),swap(),empty()
  • 顺序容器的接口

    • 插入方法
      • push_front(),push_back(),insert(),运算符“=”
    • 删除方法
      • pop() ,erase(),clear()
    • 迭代访问方法
      • 使用迭代器
    • 其他顺序容器访问方法(不修改访问方法)
      • front(),back(),下标[ ]运算符

vector

  • 初始化 vector 容器方法

    1. vector<elementType> v; // 创建一个没有任何元素的空容器
    2. vector<elementType> v(otherVec); //调用拷贝构造函数创建新容器
    3. vector<elementType> v(size); //创建一个大小为size的对象v,并使用默认构造函数初始化该向量
    4. vector<elementType> v(n,elem); //创建一个大小为n的容器,并使用元素elem初始化每一个元素
    5. vector<elementType> v(begin,end); //创建容器v,并使用(begin,end)之间的元素初始化容器
  • 元素的插入

    1. veclist.push_back(elem); //将elem的一个拷贝插入到veclist的末尾
    2. veclist.insert(position,elem); //将elem的一个拷贝插入到指定的position的位置上
    3. veclist.insert(position,n,elem); //将elem的n个拷贝插入到由position指定的位置上
    4. veclist.insert(position,beg,end); //将从迭代器 beg至end-1 之间的元素插入到veclist 的position位置上
  • 元素的删除

    1. veclist.clear(); //清空容器中所有元素
    2. veclist.erase(position); //删除position指定位置的元素
    3. veclist.erase(beg,end); //删除从beg至end-1之间的元素
    4. veclist.pop_back(); //删除最后一个元素
  • 迭代器其他常用函数

    1. veclist.capacity(); //返回当前可插入到容器veclist的最大数(capacity是指容器在必须分配新的存储空间之前可以存放的元素总数)
    2. veclist.empty(); //判断容器是否为空,为空返回true,否则返回false
    3. veclist.size(); //返回容器veclist 中当前元素个数
    4. veclist.max_size(); //返回可以插入到容器中元素的最大个数(max_szie表示STL容器允许的最大元素数,通常,这个数是一个很大的常整数,可以理解为无穷大。这个数目与平台和实现相关)
  • vector使用

#include <iostream>
#include <vector>
using namespace std;
vector<int> v;
int main()
{
    int elem;
    //从标准输入设备输入整形
    //知道输入的不是整形为止
    while(cin>>elem)
    {
        v.push_back(elem);
    }
    for(int j=0;j<v.size();j++)
    {
        cout<<v[j]<<" ";
    }
    cout<<endl;
    /*
    for (vector<int>::size_type ix = 0; ix != ivec.size(); ++ix) 
    {
        cout<<ivec[ix] <<endl; // 用下标方式访问
    }
    */
    for(vector<int>::iterator it=v.begin();it<v.end();it++)
    {
        cout<<*it<<" ";
    }
    cout<<endl;
    for(vector<int>::reverse_iterator it=v.rbegin();it<v.rend();it++)
    {
        cout<<*it<<" ";
    }
    cout<<endl;
}
  • vector 与指针
#include <iostream>
#include <vector>
using namespace std;
int *p1;               //iterator i1;
const int *p2;         //const_iterator i2;
int *const p3;         //const iterator i3;
const int *const p4;   //const const_iterator i4;
void show(vector<int> vi)
{
    vector<int>::iterator it;
    it=vi.begin();
    while(it!=vi.end())
    {
        cout<<*it++<<' ';
    }   
    cout<<endl;
};
int main()
{
    vector<int> vi(3,90);
    show(vi);
    int a[5]={3,4,5,6,7};
    vi.insert(vi.begin(),a,a+5);
    show(vi);
    vi.push_back(100);
    show(vi);
    cout<<"size:"<<vi.size()<<endl;
    vi.assign(5,99);
    show(vi);
    cout<<"size:"<<vi.size()<<endl;
}

List

  • list的使用
    • list的四种构造函数
      1. list<elementType> L; //构造空容器L
      2. list<elementType> L(n, elem); //创建一个大小为size的对象L,并使用elem元素进行初始化
      3. list<elementType> L(list1); //创建容器L,并使用已创建的容器list1进行初始化
      4. list<elementType> L(begin,end); //创建容器L,并使用(begin,end)之间的元素初始化容器
    • list元素的插入
      1. L.push_back(elem); //向容器的末尾插入元素elem的拷贝
      2. L.push_front(elem); //向容器的开端插入元素elem的拷贝
      3. L.insert(position, elem); //向容器的position位置插入元素elem的拷贝
      4. L.insert(position, n, elem); //向容器的position位置上插入元素elem的n个拷贝
      5. L.insert(position, beg, end); //将迭代器beg至 end-1 指向的内容插入到容器的position位置上
      6. L.splice(position, list); //将链表容器list中的元素插入到position位置上,并且清空list容器
      7. L.splice(position, list, pos); //将容器list中的pos位置上的元素插入到position位置上,并将pos位置上的元素从list中移除
      8. L.splice(position, list, beg, end); //将容器list中beg 至 end-1 位置上的元素插入到position位置上,并将这些元素从list中移除
    • 访问容器元素
      1. List<elementType>::iterator i;
        使用begin, end 成员函数来操作

      2. List<elementType>::reverse_iterator ri;
        使用rbegin, rend 成员函数来操作

      3. List<elementType>::const_iterator ci;
        使用begin, end 成员函数来操作,但是只能访问,不能进行修改

    • 元素的删除
      1. L.pop_back(); //删除容器的最后一个元素
      2. L.pop_front(); //删除容器的第一个元素
      3. L.clear(); //删除容器的所有元素
      4. L.erase(position); //删除容器指定位置的元素
      5. L.erase(beg, end); //删除迭代器beg 至 end-1 之间的元素
      6. L.remove(elem); //移除与元素elem相等的元素
#include <iostream>
using namespace std;
#include <list>
int main()
{
    int cpp[5]={3,6,1,7,5};
    int java[8]={6,4,7,8,15,2,3,9};
    int Unix[4]={5,2,6,9};
    list<int> li;
    li.insert(li.begin(),cpp,cpp+5);
    li.insert(li.begin(),java,java+8);
    li.insert(li.begin(),Unix,Unix+4);
    li.sort();
    li.unique();
    li.reverse();
    list<int>::iterator it=li.begin();
    while(it!=li.end())
    {   
        cout<<*it++<<' '; 
    }
    cout<<endl;
}
/*
15 9 8 7 6 5 4 3 2 1 
*/
#include <iostream>
#include <list>
using namespace std;
int main()
{
    list<int> l1;
    int a[5]={3,4,5,6,7};
    list<int> l2(a,a+5);
    cout<<"l1.size="<<l1.size()<<endl;
    cout<<"l2.size="<<l2.size()<<endl;
    list<int>::iterator it;
    for(it=l2.begin();it!=l2.end();it++)
    {
        cout<<*it<<' ';
    }
    cout<<endl;
    it=l2.begin();
    it++;
    l2.erase(it);//删除it对应位置元素
    l2.insert(l2.begin(),100);
    l2.insert(l2.end(),200);
    for(it=l2.begin();it!=l2.end();it++)
    {
        cout<<*it<<' ';
    }
    cout<<endl;
}
l1.size=0
l2.size=5
3 4 5 6 7 
100 3 5 6 7 200 
  • map 使用
#include <iostream>
#include <map>
#include <string>
using namespace std;
int main()
{
    //key(是唯一的) value
    map<int,string> mis;
    //1. 插入map元素
    mis.insert(make_pair(62,"东方不败"));
    mis.insert(make_pair(32,"岳不群"));
    mis.insert(make_pair(36,"林平之"));
    //2. 插入方式,这里的20不是下标
    mis[20]="劳德罗";
    //mis.insert(make_pair(36,"abc"));无法覆盖
    //mis[36]="xyz";//可覆盖
    map<int,string>::iterator it;
    it=mis.begin();
    //元素位置和key相关,和插入顺序无关
    //1.自动顺序的
    while(it!=mis.end())
    {
        cout<<it->first<<":"<<it->second<<endl;
        ++it;
    }
}
/*
20:劳德罗
32:岳不群
36:林平之
62:东方不败
*/
  • 使用映射(map)建立阿拉伯数字0~9和英文单词zero到nine的映射关系,并输入阿拉伯数字(如1),输出英文数字(如one)。
#include <iostream>
#include <map>
#include <string>
using namespace std;
int main()
{
    int num;
    map<int,string> mis;
    mis.insert(make_pair(0,"zero"));
    mis.insert(make_pair(1,"one"));
    mis.insert(make_pair(2,"two"));
    mis.insert(make_pair(3,"three"));
    mis.insert(make_pair(4,"four"));
    mis.insert(make_pair(5,"five"));
    mis.insert(make_pair(6,"six"));
    mis.insert(make_pair(7,"seven"));
    mis.insert(make_pair(8,"eight"));
    mis.insert(make_pair(9,"nine"));
    map<int,string>::iterator it;
    cin>>num;
    for(it=mis.begin();it!=mis.end();it++)
    {
        if(it->first==num)
            cout<<it->second<<endl;
    }
}
  • vector 实现删除指定元素(vector<int> num,从控制台输入int填充vector,删除指定元素)
#include <iostream>
#include <vector>
using namespace std;
vector<int> vi;
int main()
{
    int elem;
    int place;
    while(cin>>elem)
    {
        vi.push_back(elem);
    }
    for(vector<int>::iterator it=vi.begin();it<vi.end();it++)
    {
        cout<<*it<<' ';
    }
    cout<<endl;
    for(vector<int>::iterator it1=vi.begin();it1<vi.end();)
    {
        if(*it1==3)
        {
            it1=vi.erase(it1);
            cout<<*it1<<endl;
        }
        else
            it1++;
    }
    for(vector<int>::iterator it2=vi.begin();it2<vi.end();it2++)
    {
        cout<<*it2<<' ';
    }
    cout<<endl;
}
  • const 的使用拓展
#include <iostream>
#include <vector>
using namespace std;
class Test
{
    int m_t;
    public:
        Test(){}
        void lianxi()const
        {
            cout<<"lianxi const"<<endl;
        }
        void lianxi()
        {
            cout<<"lianxi"<<endl;
        }
};
int main()
{
    const Test t1;
    t1.lianxi();
    Test t2;
    t2.lianxi();
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,026评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,655评论 1 296
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,726评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,204评论 0 213
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,558评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,731评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,944评论 2 314
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,698评论 0 203
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,438评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,633评论 2 247
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,125评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,444评论 3 255
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,137评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,103评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,888评论 0 197
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,772评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,669评论 2 271

推荐阅读更多精彩内容

  • 前言: 详细介绍: List:元素有放入顺序,元素可重复Map:元素按键值对存储,无放入顺序Set:元素无放入顺序...
    YBshone阅读 8,534评论 0 17
  • 模板 模板是C++语言相对较新的一个重要特性。模板使程序员能够快速建立具有类型安全的类库集合和函数集合,它的实现,...
    帅碧阅读 918评论 2 4
  • 原其实也甚没在意 你我的初遇 就像习惯着每天坐地铁 这头到那头 最初日子里更是彼此恪尽职守 陌生人应有的默契 有时...
    为醉醇香也醉风月阅读 168评论 2 2
  • 男人常嗤之以鼻女人间的友谊。说女人间的友谊脆弱不堪,说女人的友谊只在朋友圈,说女人在义气方面只是口头家…… 翻阅一...
    九丑_阅读 465评论 6 6
  • 解构与�重构 分析马斯克深化学习迁移的两步法 1.把知识解构为若干基本原理 马斯克发表在Reddit AMA上的回...
    Eddy君阅读 1,954评论 0 0