背景:老师留了一个作业,对两个数组进行相加,但是总是会出现错误;首先我们需要知道当数组作为参数传递的时候是不能用 sizeof 的,因为当数组传入子函数就变成了普通的数组头;这时候使用 sizeof 只会检测到指针的长度;

我们用模板来传递数组,实际上并非模板可以传递数组,而是引用传递完成了这件事

对应的现象是:值传递的数组会衰减成指针

下面给出利用引用传递的代码(没有使用模板)

 1 #include<iostream>
2 using namespace std;
3
4 int P(int (&a)[5]);   //子函数
5 int main(){
6
7 int a[5]={0};
8
9 int b=P(a);
10
11 cout<<b<<endl;
12
13 return 0;
14 }
15 int P(int (&a)[5]){      //进行了引用传递
16
17 return sizeof(a)/sizeof(a[0]);
18 }

  在这个代码中我们发现一件事,如果要传递这个数组,我们首先要知道数组的位数,但是我们要求的不就是数组的位数吗,这就是上文我们做的,其实也是模板的主要作用:实参演绎,我们规定了两个参数,这两个参数到底谁是5位数组,谁是10位数组,这个工作在模板中被自动的完成了,综合了上述两种技术我们实现了数组的传递

  其次,我们需要知道子函数返回的数组的时候也很麻烦,不仅仅因为返回的是指针,更是因为在子函数中定义的数组会随着函数的结束而释放,最终只留下数组头一个数据可以使用,所以我们用动态内存,去主动的分配一个空间给数组


现在我们把操作和方法放在下面:

模板: 传入一个数组(实现泛型编程,因为要传入的这个数组长度是不确定的)
引用传递: 这里没有操作原来的数,仅仅是为了减少拷贝的操作
new: 给数组创建内存
delete: 配合new,释放内存

  

  

以下代码思路:

先弄两个数组,一起传入到子函数中

由子函数创建新的数组,通过加减法给新的数组赋值,(new)

操作完新的数组之后结束程序(delete)

 1 #include<iostream>
2
3 using namespace std;
4
5 template <typename T1,typename T2> //要在模板参数列表中定义两个模板参数;因为:
6 int* plusArr(T1& a,T2& b); //将来传入的int[10]和int[5]是两个不同的类型
7
8 int main(){
9
10 int a[10]={1,2,3,4,5,6,7,8,9,0};
11 int b[5]={3,4,5,6,7};
12
13 int* sum=plusArr(a,b); //用sum保存这个指针,因为plusArr栈会清掉(包括指针p)
14
15 int i=0;
16 while(sum[i]!='\0'){
17 cout<<sum[i]<<' ';
18 ++i;
19 }
20
21 delete[] sum; //释放内存 在此,delete一定要加中括号[]!!
22 }
23
24 template<typename T1,typename T2>
25 int* plusArr(T1& a,T2& b){
26
27 int m=(int)sizeof(a)/sizeof(a[0]);
28 int n=(int)sizeof(b)/sizeof(b[0]);
29
30 if(m>n){
31 int* p=new int[m+1]; //在堆上分配了动态内存 (并未初始化)
32 for(int i=0;i<n;i++){ //开始赋值
33 p[i]=a[i]+b[i];
34 }
35 for(int i=n;i<m;i++){
36 p[i]=a[i];
37 }
38 p[m]='\0'; //用来标志
39 return p;
40 }
41 else{ //上面的是a长度大,下面b同理
42 int* p=new int[n+1];
43 for(int i=0;i<m;i++){
44 p[i]=a[i]+b[i];
45 }
46 for(int i=n;i<m;i++){
47 p[i]=b[i];
48 }
49 p[n]='\0';
50 return p;
51 }
52
53 }

  这个函数写出来呢,主要是验证一些函数的用法,这些函数虽然在这个程序中显得“过分”了,但是在将来涉及到复用的时候,像这种泛型编程就可以大显身手,顺便复习一下new delete


更加深入的理解:

在人民邮电出版社《C++模板:tamplate》一书中开头就提醒我们模板的定义一定要写常量引用,而且注意const的顺序一定是在后面;

下面我们列出一个模板的定义,并尝试去理解他的形式参数性质:

 1 #include<iostream>
2 int main(){
3 int* b;
4 int* &a=b; //这里对指针a的操作等于直接对指针b进行操作
5
6 int const c;
7 int* const d; //定义一个正常的常量指针d
8
9 int* n;
10 int* const &m=n; //定义一个常量指针的引用,这里n是常量,所有对m的操作最终会作用到n上
11
12 //在模板中我们经常看到如下定义
13 tamplate<typename T>
14 T const& name(T const& a,T const& b){ //这里是引用传递,并且我们不会用引用去更改外面的值
15 ;
16 }
17
18 return 0;
19 }

转载请注明来源

面向对象程序设计(二):C++模板初探的更多相关文章

  1. python基础16 ----面向对象程序设计二

    一.继承与派生 1.继承的定义:继承是一种创建新类的方式,即在类中提取共同的部分创建出一个类,这样的类称为父类,也可称为基类和超类,新建的类称为派生类或子类. 2.单继承:就相当于子类继承了一个父类. ...

  2. 2017-2018-2 20165206 实验二《Java面向对象程序设计》实验报告

    2017-2018-2 20165206 实验二<Java面向对象程序设计>实验报告 一.实验报告封面 课程:Java程序设计 班级:1652班 姓名:韩啸 学号:20165206 指导教 ...

  3. 20165323 实验二 Java面向对象程序设计

    一.实验报告封面 课程:Java程序设计 班级:1653班 姓名:杨金川 学号:20165323 指导教师:娄嘉鹏 实验日期:2018年4月16日 实验时间:13:45 - 15:25 实验序号:二 ...

  4. 20155228 实验二 Java面向对象程序设计

    20155228 实验二 Java面向对象程序设计 实验内容 1. 初步掌握单元测试和TDD 2. 理解并掌握面向对象三要素:封装.继承.多态 3. 初步掌握UML建模 4. 熟悉S.O.L.I.D原 ...

  5. 20155225 实验二《Java面向对象程序设计》实验报告

    20155225 实验二<Java面向对象程序设计>实验报告 一.单元测试 三种代码 知道了伪代码.产品代码.测试代码的关系和用途,并根据老师的例子,按测试代码调试了产品代码. 值得注意的 ...

  6. 20155232 实验二《Java面向对象程序设计》实验报告

    20155232 实验一<Java面向对象程序设计>实验报告 实验内容 1.初步掌握单元测试和TDD 2.理解并掌握面向对象三要素:封装.继承.多态 3.初步掌握UML建模 4.熟悉S.O ...

  7. 20165202 实验二 Java面向对象程序设计

    实验二 Java面向对象程序设计 一.实验报告封面 课程:Java程序设计 班级:1652班 姓名:贾海粟 学号:20165202 指导教师:娄嘉鹏 实验日期:2018年4月16日 实验时间:15:3 ...

  8. [.net 面向对象程序设计进阶] (23) 团队开发利器(二)优秀的版本控制工具SVN(上)

    [.net 面向对象程序设计进阶] (23) 团队开发利器(二)优秀的版本控制工具SVN(上) 本篇导读: 上篇介绍了常用的代码管理工具VSS,看了一下评论,很多同学深恶痛绝,有的甚至因为公司使用VS ...

  9. [.net 面向对象程序设计进阶] (17) 多线程(Multithreading)(二) 利用多线程提高程序性能(中)

    [.net 面向对象程序设计进阶] (17) 多线程(Multithreading)(二) 利用多线程提高程序性能(中) 本节要点: 上节介绍了多线程的基本使用方法和基本应用示例,本节深入介绍.NET ...

  10. [.net 面向对象程序设计进阶] (15) 缓存(Cache)(二) 利用缓存提升程序性能

    [.net 面向对象程序设计进阶] (15) 缓存(Cache)(二) 利用缓存提升程序性能 本节导读: 上节说了缓存是以空间来换取时间的技术,介绍了客户端缓存和两种常用服务器缓布,本节主要介绍一种. ...

随机推荐

  1. 用map来统计数组中各个字符串的数量

    1.背景 想要统计这一个字符串数组中每一个非重复字符串的数量,使用map来保存其key和value.这个需求在实际开发中经常使用到,我以前总是新建一个空数组来记录不重复字符串,并使用计数器计数,效率低 ...

  2. Redis系列10:HyperLogLog实现海量数据基数统计

    Redis系列1:深刻理解高性能Redis的本质 Redis系列2:数据持久化提高可用性 Redis系列3:高可用之主从架构 Redis系列4:高可用之Sentinel(哨兵模式) Redis系列5: ...

  3. Halcon使用MeasurePos来实现检测边缘点

    (1)为了提高性能,测量句柄只需要初始化一次: 参数:测量矩形的中心点行坐标,测量矩形中心的列坐标,测量矩形的角度,测量矩形的宽,测量矩形的高,待处理图像的宽,待处理图像的高,使用的算法,输出测量句柄 ...

  4. K3S 安装及配置

    K3S安装 curl -sfL https://rancher-mirror.oss-cn-beijing.aliyuncs.com/k3s/k3s-install.sh | INSTALL_K3S_ ...

  5. HTTPS - 揭秘 TLS 1.2 协议完整握手过程--此文为转发文,一定要结合wirshark工具看,很清楚

    winshark 筛选条件为:tls and ip.src==xxx 本文通过对一次 TLS 握手过程的数据抓包分析做为切入点,希望能进一步的帮助大家理解 HTTPS 原理. HTTPS 是建立在 S ...

  6. devexpress中searchLookUpEdit赋值不显示

    给searchLookUpEdit进行赋值的时候使用 string str="123"; searchLookUpEdit1.EditValue = str; 一直不显示或者显示为 ...

  7. C温故补缺(十五):栈帧

    栈帧 概念 栈帧:也叫过程活动记录,是编译器用来实现过程/函数调用的一种数据结构,每次函数的调用,都会在调用栈(call stack)上维护一个独立的栈帧(stack frame) 栈帧的内容 函数的 ...

  8. MICCAI 论文投稿须知翻译

    MICCAI 论文投稿须知翻译 以MICCAI 2021 PAPER SUBMISSION AND REBUTTAL GUIDELINES为例,每年投稿须知类似 作者信息和rebuttal 本文件包含 ...

  9. 【Java Web】项目通用返回模块ServerResponse:枚举code状态码、泛型返回值、序列化注解限制数据

    一.枚举类编写ResponseCode package com.boulderaitech.common; /** * 编写枚举类的步骤 * (1)编写所需的变量 * (2)编写枚举类构造方法 * ( ...

  10. 【每日一题】【栈和队列、双端队列】20. 有效的括号/NC52 有效括号序列-211127/220126

    给定一个只包括 '(',')','{','}','[',']' 的字符串 s ,判断字符串是否有效. 有效字符串需满足: 左括号必须用相同类型的右括号闭合.左括号必须以正确的顺序闭合. 来源:力扣(L ...