首頁  >  文章  >  後端開發  >  在 C 中將 64 位元整數轉換為網路位元組順序時如何實現可移植性?

在 C 中將 64 位元整數轉換為網路位元組順序時如何實現可移植性?

Susan Sarandon
Susan Sarandon原創
2024-10-30 04:37:28813瀏覽

How to Achieve Portability When Converting 64-Bit Integers to Network Byte Order in C  ?

C 中64 位元整數的網路位元組順序轉換的可移植性問題

在C 中,需要轉換64 位元整數特定應用程式的“網路位元組順序”,例如實現記憶體快取協定。儘管存在用於轉換 32 位元整數的 htonl() 函數,但用於轉換 32 位元整數的類似函數 (htonll()) 仍然難以捉摸。

標準函數注意事項

不幸的是,目前 C 語言中沒有可移植的標準函數可以執行 64 位元網路位元組順序轉換。這種缺失產生了對自訂實現的需求。

自訂實作

要實作 htonll(),我們必須先確定係統的位元組順序。一個可靠的方法是在運行時計算已知位元組順序(例如42)的整數變數的值:

<code class="cpp">static const int num = 42;
bool isBigEndian = (*reinterpret_cast<const char *>(&num) == num);</code>

接下來,我們可以根據位元組順序執行轉換:

<code class="cpp">uint64_t htonll(uint64_t value)
{
    if (isBigEndian)
    {
        const uint32_t high_part = htonl(static_cast<uint32_t>(value >> 32));
        const uint32_t low_part = htonl(static_cast<uint32_t>(value & 0xFFFFFFFFLL));

        return (static_cast<uint64_t>(low_part) << 32) | high_part;
    }
    else
    {
        return value;
    }
}

用於可移植性的巨集

或者,巨集可以提供一種簡潔的方式來定義htonll() 及其對應項ntohll():

<code class="cpp">#define htonll(x) ((1 == htonl(1)) ? (x) : ((uint64_t)htonl((x) & 0xFFFFFFFF) << 32) | htonl((x) >> 32))
#define ntohll(x) ((1 == ntohl(1)) ? (x) : ((uint64_t)ntohl((x) & 0xFFFFFFFF) << 32) | ntohl((x) >> 32))</code>

這些巨集依賴系統固有的位元組排序行為來決定必要的位元組交換。

編譯器特定的巨集

有些編譯器和作業系統提供特定的巨集來決定位元組順序。例如,在支援它的系統上,我們可以使用:

<code class="cpp">#if __BIG_ENDIAN__
# define htonll(x) (x)
# define ntohll(x) (x)
#else
# define htonll(x) (((uint64_t)htonl((x) & 0xFFFFFFFF) << 32) | htonl((x) >> 32))
# define ntohll(x) (((uint64_t)ntohl((x) & 0xFFFFFFFF) << 32) | ntohl((x) >> 32))
#endif</code>

透過採用這些技術,htonll() 和 ntohll() 的自訂實作可以跨各種系統和編譯器移植。

以上是在 C 中將 64 位元整數轉換為網路位元組順序時如何實現可移植性?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

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