8. C++11新特性又哪些
自动类型推导auto,智能指指针(share_ptr,unique_ptr等),for循环简化,线程相关的(std::thread/std::mutex),空指针nullptr,lambda表达式,等等
9. share_ptr是线程安全的吗
share_ptr里包含引用计数和数据指针,引用计数是原子操作,线程安全的,但是改变数据指针的指向,后导致引用技术加减,并不是线程安全的。设想当一个share_ptr的引用计数为1,这时两个线程同时对它赋值,改变其数据指向,则会引起引用计数减小2次,引起crash。
10. 线程间同步有哪些方法
主要有锁/临界区(lock/mutex),条件变量(condition variable)和future。
- std::unique_lock<std::mutex>或 std::lock_guard<std::mutex>可以保证一次只有一个线程访问临界区
- std::condition_variable 可以通过wait方法阻塞线程,另一个线程可以通过Notify方法进行通知
- std::future配合std::async使用,可以异步开启线程,并存储返回结果,另一线程通过future的wait或get方法可以阻塞线程,直至结果返回。
参考资料:
11. 虚函数多态的实现机制
虚函数多态主要指的是基类指针(引用)调用虚函数,若基类指针(引用)指向的是派生类,则自动调用派生类的虚函数。其实现机制是通过虚函数表 和虚表指针。在含有虚函数的类编译时,会自动生成相应的虚函数表和虚表指针,虚函数表里对于派生类重写的虚函数,会替换成重写后的虚函数地址。这样派生类赋值给基类时,实际上是基类之类指向的是派生类的虚表指针和虚函数表,调用的就是派生类的虚函数了。
12. 子类构造函数中调用虚函数会怎么样
子类在构造时会先调用父类构造,再调用子类构造,所以在子类中调用虚函数,如果子类重写了,会调用重写后的子类函数,没有重写的话,仍会调用父类函数。
- 注意:如果在父类构造中调用虚函数,则子类继承后总会调用父类的虚函数,如果虚函数为纯虚,会产生链接错误。
另外尽量不要在构造中调用虚函数,参见《effective C++》条款9--绝不在构造和析构过程中调用virtual函数
13. 介绍下C++ 11中的移动语义
移动语义即std::move关键字的使用,配合移动构造,可以将临时变量直接进行构造,节省一次赋值操作和临时变量的析构,提高了效率。
14. std::vector中插入元素的push_back与embrace_back有什么区别
embrace_back是C++11中新的接口,由于C++11有了移动语义,对于vector中插入元素,也用移动语义重写了,对于临时元素可以移动构造,节省了空间,提高了效率