您的位置:首页 > 编程语言 > C语言/C++

笔试整理之C++基础试题

2017-02-27 19:03 423 查看
1、i++跟++i

(1)单独执行后,i的值都增加了1

(2)i++不可以做左值,++i可以

<1>左值:C/C++语言中可以放在赋值符号左边的变量,即具有对应的可以由用户访问的存储单元,并且能够由用户去改变其值的量。

<2>具体函数实现:

// 前缀形式:

int& int::operator++() //这里返回的是一个引用形式,就是说函数返回值也可以作为一个左值使用

{//函数本身无参,意味着是在自身空间内增加1的

  *this += 1;  // 增加

  return *this;  // 取回值

}

//后缀形式:

const int int::operator++(int) //函数返回值是一个非左值型的,与前缀形式的差别所在。

{//函数带参,说明有另外的空间开辟

  int oldValue = *this;  // 取回值

  ++(*this);  // 增加

  return oldValue;  // 返回被取回的值

}

2、C++ 的一个常见面试题是让你实现一个 String 类,限于时间,不可能要求具备 std::string 的功能,但至少要求能正确管理资源。

此题转载至:http://coolshell.cn/articles/10478.html

首先选择数据成员,最简单的 String 只有一个 char* 成员变量。

好处是容易实现,坏处是某些操作的复杂度较高(例如 size() 会是线性时间)。

为了面试时写代码不出错,本文设计的 String 只有一个 char* data_成员。

而且规定 invariant 如下:一个 valid 的 string 对象的 data_ 保证不为 NULL,data_ 以 '\0' 结尾,以方便配合 C 语言的 str*() 系列函数。

其次决定支持哪些操作,构造、析构、拷贝构造、赋值这几样是肯定要有的(以前合称 big three,现在叫 copy control)。

如果钻得深一点,C++11的移动构造和移动赋值也可以有。

为了突出重点,本文就不考虑 operator[] 之类的重载了。

这样代码基本上就定型了:
#include <utility> 
#include <string.h> 
  
class String 

 public: 
  String() 
    : data_(new char[1]) 
  { 
    *data_ = '\0'; 
  } 
  
  String(const char* str) 
    : data_(new char[strlen(str) + 1]) 
  { 
    strcpy(data_, str); 
  } 
  
  String(const String& rhs) 
    : data_(new char[rhs.size() + 1]) 
  { 
    strcpy(data_, rhs.c_str()); 
  } 
  /* Delegate constructor in C++11 
  String(const String& rhs) 
    : String(rhs.data_) 
  { 
  } 
  */ 
  
  ~String() 
  { 
    delete[] data_; 
  } 
  
  /* Traditional: 
  String& operator=(const String& rhs) 
  { 
    String tmp(rhs); 
    swap(tmp); 
    return *this; 
  } 
  */ 
  String& operator=(String rhs) // yes, pass-by-value 
  { 
    swap(rhs); 
    return *this; 
  } 
  
  // C++ 11 
  String(String&& rhs) 
    : data_(rhs.data_) 
  { 
    rhs.data_ = nullptr; 
  } 
  
  String& operator=(String&& rhs) 
  { 
    swap(rhs); 
    return *this; 
  } 
  
  // Accessors 
  
  size_t size() const 
  { 
    return strlen(data_); 
  } 
  
  const char* c_str() const 
  { 
    return data_; 
  } 
  
  void swap(String& rhs) 
  { 
    std::swap(data_, rhs.data_); 
  } 
  
 private: 
  char* data_; 
}; 

注意代码的几个要点:
只在构造函数里调用 new char[],只在析构函数里调用 delete[]。
赋值操作符采用了《C++编程规范》推荐的现代写法。
每个函数都只有一两行代码,没有条件判断。
析构函数不必检查 data_ 是否为 NULL。
构造函数 
String(const char* str)
 没有检查 str 的合法性,这是一个永无止境的争论话题。这里在初始化列表里就用到了 str,因此在函数体内用 assert() 是无意义的。

这恐怕是最简洁的 String 实现了。

练习1:增加 operator==、operator<、operator[] 等操作符重载。

练习2:实现一个带 int size_; 成员的版本,以空间换时间。

练习3:受益于右值引用及移动语意,在 C++11 中对 String 实施直接插入排序的性能比C++98/03要高,试编程验证之。(g++的标准库也用到了此技术。)

3.描述在浏览器中敲入一个网址并按下回车后所发生的事情//比如 www.baidu.com

(1)DNS域名解析

首先要查找域名的IP地址,经过DNS查找后,将输入网址域名转换成IP地址。如果url里不包含端口号,则会使用该协议的默认端口号。DNS域名解析的过程是这样的:首先我们知道我们本地的机器上在配置网络时都会填写DNS,这样本机就会把这个url发给这个配置的DNS服务器,如果能够找到相应的url则返回其ip,否则该DNS将继续将该解析请求发送给上级DNS,整个DNS可以看做是一个树状结构,该请求将一直发送到根直到得到结果。

(2)在发送请求前,先要建立一个TCP连接(socket连接)。socket通过ip和端口建立的,现在有了目标ip和端口号。

(3)建立连接后发送请求。浏览器给web服务器发送一个HTTP请求。请求中也包含浏览器存储的该域名的cookies。cookies是与跟踪一个网站状态相匹配的键值。cookies会存储登录用户名,服务器分配的密码和一些用户设置等。Cookies会以文本文档形式存储在客户机里,每次请求时发送给服务器。

(3)服务的永久重定向响应

(4)浏览器跟踪重定向地址

(5)服务器“处理”请求:应答 web服务器收到请求,进行处理。 从它的文档空间中搜索子目录index.html。如果找到该文件,Web服务器把该文件内容传送给相应的Web浏览器;未找到,返回错误信息。为了告知浏览器,Web服务器首先传送一些HTTP头信息,然后传送具体内容(即HTTP体信息),HTTP头信息和HTTP体信息之间用一个空行分开

(6)服务器发回一个HTML响应

(7)释放 TCP 连接:若connection 模式为close,则服务器主动关闭TCP 连接,客户端被动关闭连接,释放TCP 连接;若connection 模式为keepalive,则该连接会保持一段时间,在该时间内可以继续接收请求

(8)浏览器开始显示HTML

(9)浏览器发送获取嵌入在HTML中的对象

参考网址:http://www.mamicode.com/info-detail-1357508.html

4.strcpy和memcpy的区别 ()

strcpy和memcpy都是标准C库函数,它们有下面的特点。

strcpy提供了字符串的复制。即strcpy只用于字符串复制,并且它不仅复制字符串内容之外,还会复制字符串的结束符。

已知strcpy函数的原型是:char* strcpy(char* dest, const char* src);

memcpy提供了一般内存的复制。即memcpy对于需要复制的内容没有限制,因此用途更广。

void *memcpy( void *dest, const void *src, size_t count );

char * strcpy(char * dest, const char * src) // 实现src到dest的复制

{

  if ((src == NULL) || (dest == NULL)) //判断参数src和dest的有效性

  {

      return NULL;

  }

  char *strdest = dest;        //保存目标字符串的首地址

  while ((*strDest++ = *strSrc++)!='\0'); //把src字符串的内容复制到dest下

  return strdest;

}

void *memcpy(void *memTo, const void *memFrom, size_t size)

{

  if((memTo == NULL) || (memFrom == NULL)) //memTo和memFrom必须有效

         return NULL;

  char *tempFrom = (char *)memFrom;             //保存memFrom首地址

  char *tempTo = (char *)memTo;                  //保存memTo首地址      

  while((size-- )> 0)                //循环size次,复制memFrom的值到memTo中

         *tempTo++ = *tempFrom++ ;  

  return memTo;



strcpy和memcpy主要有以下3方面的区别。

1、复制的内容不同。strcpy只能复制字符串,而memcpy可以复制任意内容,例如字符数组、整型、结构体、类等。

2、复制的方法不同。strcpy不需要指定长度,它遇到被复制字符的串结束符"\0"才结束,所以容易溢出。memcpy则是根据其第3个参数决定复制的长度。

3、用途不同。通常在复制字符串时用strcpy,而需要复制其他类型数据时则一般用memcpy
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  左值 C++ 面试 网络