国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

C++ 面試題:用 unique_ptr 作為返回值可以嗎?

開發(fā)
使用unique_ptr作為函數(shù)返回值的核心優(yōu)勢在于明確所有權(quán)轉(zhuǎn)移與自動化資源釋放。

首先給出答案:使用 std::unique_ptr 作為函數(shù)返回值不僅是合法的,而且是一種推薦做法,尤其在需要明確轉(zhuǎn)移對象所有權(quán)時?!?/p>

寫個簡單的測試代碼:

#include <iostream>
#include <functional>

classA
{
public:
        voidfunc()
        {
                std::cout << "A::Func" << std::endl;
        }
};

std::unique_ptr<A> CreateA()
{
        std::unique_ptr<A> pA = std::make_unique<A>();

        return pA;//正確
        //return std::move(pA);//也正確 兩種方法都可以
}

intmain()
{        
        auto p = CreateA();
        p->func();

        getchar();
        return0;
}

return pA; 和 return std::move(pA); 兩種寫法都可以! 

一、unique_ptr作為返回值的核心優(yōu)勢

1. 明確所有權(quán)轉(zhuǎn)移語義

unique_ptr的“唯一所有權(quán)”特性,天然適合用于表示資源的轉(zhuǎn)移。  

當(dāng)函數(shù)返回unique_ptr時,相當(dāng)于向調(diào)用者傳遞一個明確的信息:“這個對象的所有權(quán)現(xiàn)在屬于你,你負(fù)責(zé)管理它的生命周期?!薄?/p>

代碼示例:  

std::unique_ptr<DatabaseConnection> createConnection() {
    return std::make_unique<DatabaseConnection>("user", "password");
}

int main() {
    auto conn = createConnection(); // 所有權(quán)轉(zhuǎn)移至main函數(shù)
    conn->query("SELECT * FROM table"); 
} // conn自動釋放,連接關(guān)閉

對比分析:  若返回裸指針或 shared_ptr,調(diào)用者可能誤解是否需要釋放資源或共享所有權(quán),而unique_ptr徹底避免了這類歧義?!?/p>

2. 工廠模式的天然搭檔

工廠函數(shù)的核心任務(wù)是創(chuàng)建對象并移交控制權(quán)。unique_ptr與工廠模式完美契合?!?/p>

擴(kuò)展案例:多態(tài)對象的創(chuàng)建  

class Animal {
public:
    virtual ~Animal() = default;
    virtualvoidspeak()const= 0;
};

classDog : public Animal {
public:
    voidspeak()constoverride{ std::cout << "Woof!"; }
};

classCat : public Animal {
public:
    voidspeak()constoverride{ std::cout << "Meow!"; }
};

// 工廠函數(shù)返回基類的unique_ptr
std::unique_ptr<Animal> createAnimal(const std::string& type){
    if (type == "dog") return std::make_unique<Dog>();
    if (type == "cat") return std::make_unique<Cat>();
    throw std::invalid_argument("Unknown animal type");
}

關(guān)鍵點:  

  • 基類必須有虛析構(gòu)函數(shù),確保正確釋放派生類資源。
  • 調(diào)用者無需關(guān)心具體類型,通過基類接口操作對象。

3. 異常安全的強(qiáng)保證

在可能拋出異常的代碼中,unique_ptr 能確保資源不被泄漏?!?/p>

場景分析:  

// 錯誤示例:裸指針在異常時泄漏
voidunsafeProcess(){
    int* data = newint[1024];
    process(data);  // 若拋出異常,內(nèi)存可能泄漏!
    delete[] data;
}

// 正確示例:unique_ptr 自動釋放
voidsafeProcess(){
    auto data = std::make_unique<int[]>(1024);
    process(data.get());  // 即使異常,內(nèi)存仍釋放
}

若使用裸指針,在process() 拋出異常時,內(nèi)存將泄漏;而unique_ptr在任何執(zhí)行路徑下都能正確釋放資源?!?/p>

4. 幾乎零額外開銷的性能優(yōu)勢

盡管unique_ptr提供了自動化管理,但其性能與裸指針幾乎無異(這點和 shared_ptr 指針有很大區(qū)別)。 

編譯器優(yōu)化機(jī)制:

(1) 返回值優(yōu)化(RVO)

RVO 是C++編譯器的一種優(yōu)化技術(shù),旨在消除函數(shù)返回對象時的不必要拷貝或移動操作。其核心思想是:直接在調(diào)用者的內(nèi)存空間中構(gòu)造返回的對象,而非在函數(shù)內(nèi)部構(gòu)造后再拷貝或移動到調(diào)用者處。 

① RVO的工作機(jī)制 

  • 傳統(tǒng)流程(無RVO): 函數(shù)內(nèi)部構(gòu)造對象 → 將對象拷貝/移動到調(diào)用者的接收位置 → 銷毀函數(shù)內(nèi)的臨時對象。 此過程可能觸發(fā)拷貝構(gòu)造函數(shù)或移動構(gòu)造函數(shù)?!?/li>
  • RVO優(yōu)化后流程: 編譯器直接在調(diào)用者預(yù)留的內(nèi)存空間中構(gòu)造對象,跳過了臨時對象的創(chuàng)建和傳遞。 這意味著沒有拷貝或移動操作發(fā)生,對象的構(gòu)造和析構(gòu)僅發(fā)生一次。 

② RVO的觸發(fā)條件 

  • 返回的必須是局部對象(非全局或靜態(tài)對象)?!?/li>
  • 返回的表達(dá)式類型與函數(shù)返回類型嚴(yán)格匹配。 
  • 返回的表達(dá)式是純右值(prvalue)(例如直接返回構(gòu)造函數(shù)調(diào)用或 make_unique 的結(jié)果)。 

代碼示例: 

// 觸發(fā)RVO的情況
std::unique_ptr<int> create() {
    return std::make_unique<int>(42);  // 直接返回prvalue
}

(2) 命名返回值優(yōu)化( NRVO)

NRVO 是C++編譯器對返回具名局部對象時的一種優(yōu)化技術(shù)。與RVO(返回值優(yōu)化)不同,NRVO針對的是函數(shù)內(nèi)部已命名且非臨時的局部變量作為返回值的情況。 

① NRVO基本定義 

NRVO:當(dāng)函數(shù)返回一個在函數(shù)內(nèi)部定義并命名的局部對象時,編譯器嘗試直接在調(diào)用者的內(nèi)存空間中構(gòu)造該對象,避免額外的拷貝或移動?!?/p>

② NRVO與RVO的區(qū)別: 

  • RVO優(yōu)化的是返回純右值(prvalue)(例如return A();)?!?/li>
  • NRVO優(yōu)化的是返回具名局部變量(例如A a; return a;)?!?/li>

(3) 移動語義

當(dāng)RVO和NRVO不適用時,C++11的移動語義會將資源所有權(quán)轉(zhuǎn)移而非復(fù)制?!?/p>

// RVO場景:返回臨時對象(prvalue)
std::unique_ptr<int> rvoExample() {
    return std::make_unique<int>(42); // RVO生效
}

// NRVO場景:返回具名局部變量
std::unique_ptr<int> nrvoExample() {
    auto ptr = std::make_unique<int>(42); 
    return ptr; // 可能觸發(fā)NRVO或移動語義
}

注意:NRVO編譯器支持程度不同,優(yōu)先依賴 RVO,NRVO 不是 C++ 標(biāo)準(zhǔn)強(qiáng)制要求的! 

性能測試對比:在10萬次對象創(chuàng)建測試中,unique_ptr返回與裸指針直接 new 的性能差異小于1%。

二、unique_ptr 作為返回值的實踐細(xì)節(jié)

1. 返回局部對象的正確方式

無需std::move:  

std::unique_ptr<MyClass> createObject() {
    auto obj = std::make_unique<MyClass>();
    obj->initialize();
    return obj; // 正確!編譯器自動應(yīng)用移動語義
}
return std::move(obj); // 不必要!可能抑制RVO優(yōu)化

2. 處理繼承與多態(tài)

基類聲明虛析構(gòu)函數(shù):  

class Base {
public:
    virtual ~Base() = default; // 必須聲明為虛函數(shù)!
};

class Derived : public Base { /*...*/ };

std::unique_ptr<Base> createDerived() {
    return std::make_unique<Derived>();
}

技術(shù)細(xì)節(jié):  

  • 若基類析構(gòu)函數(shù)非虛,通過基類指針刪除派生對象是未定義行為。
  • make_unique 在構(gòu)造時即確定具體類型,確保正確析構(gòu)。

三、unique_ptr與STL容器的交互

容器中的 unique_ptr 不能被復(fù)制,只能通過移動或引用來操作?!?/p>

插入元素: 

#include <memory>
#include <vector>

int main() {
    std::vector<std::unique_ptr<int>> vec;

    // 正確:通過移動語義插入
    auto ptr = std::make_unique<int>(42);
    vec.push_back(std::move(ptr));  // ptr所有權(quán)轉(zhuǎn)移至vec,ptr變?yōu)閚ullptr

    // 直接構(gòu)造并插入(C++11起)
    vec.emplace_back(std::make_unique<int>(100)); // 無拷貝或移動,直接構(gòu)造在容器內(nèi)
}

訪問元素:通過迭代器或索引訪問容器內(nèi)的 unique_ptr,但需注意不能轉(zhuǎn)移所有權(quán)。

// 訪問但不轉(zhuǎn)移所有權(quán)
if (!vec.empty()) {
    std::cout << *vec[0] << std::endl;  // 解引用訪問對象值
    auto& ref = vec.front();            // 獲取引用,仍由容器管理所有權(quán)
}

刪除元素:當(dāng)從容器中移除元素時,unique_ptr會自動釋放其管理的對象?!?/p>

vec.pop_back();  // 移除最后一個元素,其管理的對象被銷毀

vec.erase(vec.begin());  // 刪除首個元素,對象立即釋放

四、優(yōu)秀實踐與常見陷阱

1. 必須避免的錯誤

陷阱1:返回局部變量的地址  

std::unique_ptr<int> invalidReturn() {
    int x = 42;
    return std::unique_ptr<int>(&x); // 錯誤!x是棧對象
} // x被銷毀,導(dǎo)致懸垂指針

陷阱2:所有權(quán)不明導(dǎo)致重復(fù)釋放  

auto ptr = std::make_unique<int>(10);
int* raw = ptr.get();
delete raw; // 錯誤!unique_ptr仍擁有所有權(quán)

2. 設(shè)計原則

  • 單一所有權(quán)原則:每個資源有且僅有一個unique_ptr擁有所有權(quán)。  
  • 優(yōu)先使用make_unique:比直接new更安全(異常安全)和高效。  
  • 接口明確性:函數(shù)返回unique_ptr即宣告所有權(quán)轉(zhuǎn)移,調(diào)用者必須接收或顯式忽略?!?/li>

五、結(jié)論

使用unique_ptr作為函數(shù)返回值的核心優(yōu)勢在于明確所有權(quán)轉(zhuǎn)移與自動化資源釋放。 

開發(fā)者應(yīng)遵循以下準(zhǔn)則: 

  • 優(yōu)先依賴編譯器優(yōu)化:避免不必要的std::move,信任RVO/NRVO機(jī)制。 
  • 工廠函數(shù)首選:用于創(chuàng)建動態(tài)對象,傳遞清晰的所有權(quán)語義。 
  • 避免跨作用域濫用:僅在單一作用域內(nèi)管理資源,復(fù)雜場景結(jié)合shared_ptr使用。
  • 結(jié)合自定義刪除器:擴(kuò)展 unique_ptr 至非內(nèi)存資源管理?!?/li>
責(zé)任編輯:趙寧寧 來源: CppPlayer
相關(guān)推薦

2025-05-28 08:50:00

C++循環(huán)引用節(jié)點

2025-05-22 10:10:00

C++循環(huán)引用開發(fā)

2025-09-15 02:00:00

2025-06-24 10:00:00

智能指針代碼unique_ptr

2021-10-27 11:00:30

C++語言面試

2025-05-23 08:15:00

C++constexpr字面類型

2025-06-09 07:55:00

C++引用語言

2025-05-26 03:20:00

2024-01-24 11:35:28

C++多返回值開發(fā)

2025-05-20 10:00:00

C++命名空間別名代碼

2011-03-29 14:31:41

CC++

2025-04-30 10:10:00

在 C++C++11Lambda

2025-05-27 10:15:00

void*函數(shù)開發(fā)

2025-05-06 08:20:00

互斥鎖C++編程

2009-09-07 03:07:11

C# Main方法

2025-06-05 08:05:00

vectorC++對象存儲

2025-05-20 08:10:00

函數(shù)函數(shù)類型函數(shù)指針類型

2020-06-04 14:40:40

面試題Vue前端

2021-12-19 23:58:51

Golang語言返回值

2023-11-13 07:37:36

JS面試題線程
點贊
收藏

51CTO技術(shù)棧公眾號

欧美精品videosex极品1| 一女被多男玩喷潮视频| 精品久久对白| 亚洲欧美日韩国产中文| 成年人视频在线免费观看| 国产精品免费看片| 日本免费在线视频观看| 久久久久久久久国产一区| 精选一区二区三区四区五区| 成人免费91| 久久女同精品一区二区| 国产精品99久久99久久久二8| 中文字幕在线第一页| 青草综合视频| 色综合色综合网色综合| 国产日韩另类视频一区| 日韩欧美亚洲国产另类 | 亚洲国产一区二区三区青草影视| 水蜜桃色314在线观看| 国产一区二区在线免费观看| 日韩视频精品| 美腿丝袜亚洲色图| 影音先锋男人的网站| 国产精品77777| 欧美一区二区三区爽大粗免费| 国产一区欧美二区| 一二三四视频社区在线| 久久品道一品道久久精品| www.精品在线| 一区二区三区中文在线观看| 色影视在线视频资源站| 在线观看欧美精品| 欧美xxx黑人xxx水蜜桃| 亚洲精品视频网上网址在线观看| zzzwww在线看片免费| 中文字幕精品视频| 国产精品欧美大片| 国产精品直播网红| 国产偷自视频区视频一区二区| 久久99九九| 国产福利精品一区| 蜜桃特黄a∨片免费观看| 亚洲不卡av一区二区三区| 国产一区精品| 精品一区二区电影| 成人av婷婷| 91精品国产一区二区三区动漫| 久久国产精品亚洲77777| 免费在线黄网站| 一区二区三区中文免费| 欧美videosex性欧美黑吊| 久久视频在线看| 久久国产电影| 午夜啪啪福利视频| 亚洲午夜精品网| 女厕盗摄一区二区三区| 91av在线视频观看| 美女国产一区| 日日噜噜夜夜狠狠| 在线电影国产精品| 视频国产精品| 久中文字幕一区| 欧美国产精品一区二区| 国产精品扒开做爽爽爽的视频| 最近2019中文免费高清视频观看www99 | 国产一区二区三区无遮挡| 国产精品一品视频| 福利av痴女| 亚洲电影免费观看高清完整版在线观看| 日韩影片在线观看| 国产精品国产精品国产专区蜜臀ah| 国产麻豆午夜三级精品| 绯色av一区二区| 一区二区三区久久精品| 在线一区免费| 成人免费毛片播放| 欧美xxxx在线观看| 日韩伦理视频| 自慰无码一区二区三区| 欧美日韩国产另类一区| 精品欧美午夜寂寞影院| 一区二区免费在线视频| 午夜国产精品一区| 亚洲视频一起| 中文字幕欧美日韩一区二区| 欧美日韩久久久久| 日韩深夜福利网站| 久久综合狠狠综合久久综青草| 欧美激情一区在线观看| 午夜不卡影院| 国产美女精品久久久| 亚洲人被黑人高潮完整版| 少妇一区视频| 久久综合色一本| 欧美色图在线视频| 91欧美日韩在线| 免费的av在线| 亚洲第一黄色网| 亚洲黄色影院| 中文字幕免费在线| 91精品国产精品| 久久一区二区三区四区| 欧美电影免费看| 亚欧洲精品在线视频免费观看| 色综合天天综合色综合av | 日本一二三区视频免费高清| 久久久精品国产| 国产精品资源在线看| 国产黄色在线免费观看| 亚洲a在线播放| 亚洲国产成人高清精品| 欧美美女在线直播| 天美星空大象mv在线观看视频| 亚洲午夜精品视频| 日韩精品国产欧美| 日韩精品黄色| 精品国产一区二区三区四区vr | 亚洲综合图片| 91大神在线播放精品| 91免费视频网址| 亚洲精品一区| 亚洲午夜激情| 精品国产三级a在线观看| 日韩一区二区免费看| 最新中文字幕在线观看| 国产精品嫩草视频| 一区二区三区四区乱视频| 国产精品xxxav免费视频| 手机看片一级片| 91精品国产网站| 亚洲欧美综合另类在线卡通| 老司机aⅴ在线精品导航| 黄网站免费入口| 国产精品色午夜在线观看| 欧美性猛交xxxx久久久| 欧美人成在线| 午夜dj在线观看高清视频完整版| 欧美大香线蕉线伊人久久国产精品| 7777精品伊人久久久大香线蕉超级流畅| 99国产精品久久久久久久| 国产乱妇乱子在线播视频播放网站| 亚洲欧洲一区二区| 色狠狠久久aa北条麻妃| 欧美韩国日本一区| 国际精品欧美精品| 国产福利在线视频| 亚洲高清不卡一区| 日韩一级黄色av| 国产精品初高中害羞小美女文| 国产成人一区| yiren22综合网成人| 日韩高清av电影| 亚洲男人天堂久| 久久综合久久鬼色| 久久av免费看| 尤物在线视频| 欧美日韩中文字幕在线播放| 欧美黑人国产人伦爽爽爽| 亚洲永久精品大片| 亚洲免费一区二区| 日韩中文在线播放| 16—17女人毛片毛片| 官网99热精品| 日韩精品高清在线观看| 国产欧美精品一区二区三区四区| 精品国产乱码久久久| 动漫一区在线| 每日在线更新av| 成人两性免费视频| 亚洲精品suv精品一区二区| 国产亚洲欧美一区在线观看| 亚洲区综合中文字幕日日| 丁香花在线高清完整版视频| 免费观看日韩毛片| 亚洲在线观看视频| 亚洲一区二区国产| 亚洲自拍偷拍综合| 麻豆精品国产传媒mv男同| 91久久精品无嫩草影院 | 亚洲午夜在线观看| 久久久亚洲国产| 欧美日韩国产一级二级| 国产乱色国产精品免费视频| 亚洲+变态+欧美+另类+精品| 黄色av电影在线观看| 密臀av一区二区三区| 国产伦精品一区二区三区四区免费| 日韩av最新在线观看| 一区av在线播放| 国产一二三精品| 久久裸体网站| 国外成人福利视频| 免费黄色电影在线观看| 草草草在线视频| 日韩久久久久久久久久久久久| 国外视频精品毛片| 亚洲美女视频网站| 在线免费观看日韩欧美| 国产精品久久看| 免费xxxx性欧美18vr|