首頁 >後端開發 >C++ >C 編譯器可以最佳化掉「新」運算子呼叫嗎?

C 編譯器可以最佳化掉「新」運算子呼叫嗎?

DDD
DDD原創
2024-12-02 08:49:09980瀏覽

Can a C   Compiler Optimize Away `new` Operator Calls?

編譯器可以最佳化堆記憶體分配嗎?

考慮以下使用 new 運算子分配記憶體的簡單 C 程式碼:

int main()
{
    int* mem = new int[100];
    return 0;
}

編譯器能否最佳化新的調用,從而避免動態記憶體分配?

編譯器行為

不同的編譯器在這種情況下的行為不同。 g 和 Visual Studio 2015 不會最佳化新調用,而 clang 會最佳化,正如在啟用完全最佳化的測試中觀察到的那樣。

編譯器最佳化原理

根據 N3664 :澄清記憶體分配,後來成為 C 14 的一部分,允許編譯器圍繞記憶體分配進行最佳化。這種最佳化是基於新調用沒有可觀察到的副作用的假設。

As-If 規則

但是,草案中的 as-if 規則C 標準第 1.9 節要求一致的實現來模擬抽象機的可觀察行為。從 new 中拋出異常會產生明顯的副作用,因此可以認為編譯器不允許最佳化 new 呼叫。

實作細節

On另一方面,可以說何時從 new 拋出異常的決定是實作細節。 Clang 可能會確定分配不會導致異常,因此在不違反 as-if 規則的情況下忽略新呼叫。

非拋出分配

使用new 的非拋出版本 new (std::nothrot) int[100] 仍允許 clang 最佳化分配。這是因為 clang 可能能夠證明不存在可能導致可觀察行為的全域替換運算子 new。

積極優化

在clang 的早期版本中,甚至進行了更積極的優化,如以下程式碼所示:

#include <cstddef>

extern void* operator new(std::size_t n);

template<typename T>
T* create() { return new T(); }

int main() {
    auto result = 0;
    for (auto i = 0; i < 1000000; ++i) {
        result += (create<int>() != nullptr);
    }
    return result;
}

此程式碼已最佳化to:

main:                                   # @main
    movl    00000, %eax          # imm = 0xF4240
    ret

實際上,整個循環都被最佳化掉了。更高版本的 clang 不會執行如此激進的優化。

以上是C 編譯器可以最佳化掉「新」運算子呼叫嗎?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn