首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 计算机考试 > 等级考试 > 复习指导 >

C和C++语言学习总结(二)(2)

2008-12-16 
函数参数传递;类重载、隐藏与覆盖区别;构造和析构的次序


  extern问题

  如果C++程序要调用已经被编译后的C 函数,该怎么办?

  假设某个C 函数的声明如下:

  void foo(int x, int y);

  该函数被C 编译器编译后在库中的名字为_foo,而C++编译器则会产生像_foo_int_int之类的名字用来支持函数重载和类型安全连接.由于编译后的名字不同,C++程序不能直接调用C 函数.C++提供了一个C 连接交换指定符号extern"C"来解决这个问题.例如:

  extern "C"

  {

  void foo(int x, int y);

  … // 其它函数

  }

  或者写成

  extern "C"

  {

  #include "myheader.h"

  … // 其它C 头文件

  }

  这就告诉C++编译译器,函数foo 是个C 连接,应该到库中找名字_foo 而不是找_foo_int_int.C++编译器开发商已经对C 标准库的头文件作了extern"C"处理,所以我们可以用#include 直接引用这些头文件.

  函数参数的缺省值问题

  正确方法:

  void Foo(int x=0, int y=0); // 正确,缺省值出现在函数的声明中

  void Foo(int x,int y)

  {

  ...

  }

  错误方法:

  void Foo(int x=0, int y=0) // 错误,缺省值出现在函数的定义体中

  {

  ...

  }

  正确方法:

  void Foo(int x, int y=0, int z=0);

  错误方法:

  void Foo(int x=0, int y, int z=0);

  宏代码与内联函数区别

  语言支持关系:

  C 宏代码

  C++ 宏代码 内联函数

  宏代码本身不是函数,但使用起来象函数.预处理器用复制宏代码的方式代替函数调用,省去了参数压栈、生成汇编语言的CALL调用、返回参数、执行return 等过程,从而提高了速度.使用宏代码最大的缺点是容易出错,预处理器在复制宏代码时常常产生意想不到的边际效应.

  对于任何内联函数,编译器在符号表里放入函数的声明(包括名字、参数类型、返回值类型).如果编译器没有发现内联函数存在错误,那么该函数的代码也被放入符号表里.在调用一个内联函数时,编译器首先检查调用是否正确(进行类型安全检查,或者进行自动类型转换,当然对所有的函数都一样).如果正确,内联函数的代码就会直接替换函数调用,于是省去了函数调用的开销.这个过程与预处理有显著的不同,因为预处理器不能进行类型安全检查,或者进行自动类型转换.假如内联函数是成员函数,对象的地址(this)会被放在合适的地方,这也是预处理器办不到的.

  内联函数使用方法:

  关键字inline 必须与函数定义体放在一起才能使函数成为内联,仅将inline 放在函数声明前面不起任何作用.

  正确使用方法:

  void Foo(int x, int y);

  inline void Foo(int x, int y) // inline 与函数定义体放在一起

  {

  …

  }

  错误使用方法:

  inline void Foo(int x, int y); // inline 仅与函数声明放在一起

  void Foo(int x, int y)

  {

  …

  }

    3、构造和析构的次序

  构造从类层次的最根处开始,在每一层中,首先调用基类的构造函数,然后调用成员对象的构造函数.析构则严格按照与构造相反的次序执行,该次序是唯一的,否则编译器将无法自动执行析构过程.

  String函数定义

  class String

  {

  public:

  String(const char *str = NULL); // 普通构造函数

  String(const String &other); // 拷贝构造函数

  ~ String(void); // 析构函数

  String & operate =(const String &other); // 赋值函数

  private:

  char *m_data; // 用于保存字符串

  };

  // String 的析构函数

  String::~String(void)

  {

  delete [] m_data;// 由于m_data 是内部数据类型,也可以写成delete m_data;

  }

  // String 的普通构造函数

  String::String(const char *str)

  {

  if(str==NULL)

  {

  m_data = new char[1]; // 若能加NULL 判断则更好

  *m_data = '\0';

  }

  else

  {

  int length = strlen(str);

  m_data = new char[length+1]; // 若能加NULL 判断则更好

  strcpy(m_data, str);

  }

  }

  // 拷贝构造函数

  String::String(const String &other)

  {

  int length = strlen(other.m_data);

  m_data = new char[length+1]; // 若能加NULL 判断则更好

  strcpy(m_data, other.m_data);

  }

  // 赋值函数

  String & String::operate =(const String &other)

  {

  // (1) 检查自赋值

  if(this == &other)

  return *this;

  // (2) 释放原有的内存资源

  delete [] m_data;

  // (3)分配新的内存资源,并复制内容

  int length = strlen(other.m_data);

  m_data = new char[length+1]; // 若能加NULL 判断则更好

  strcpy(m_data, other.m_data);

  // (4)返回本对象的引用

  return *this;

  }




3COME考试频道为您精心整理,希望对您有所帮助,更多信息在http://www.reader8.net/exam/

热点排行