亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

匯編高效乘法運(yùn)算的具體使用方法

 更新時(shí)間:2024年03月26日 09:54:08   作者:微軟技術(shù)分享  
在匯編語言中,乘法指令通常是通過mul(無符號(hào)乘法)和imul(有符號(hào)乘法)這兩個(gè)指令實(shí)現(xiàn)的,本文就來詳細(xì)的介紹一下匯編高效乘法運(yùn)算,感興趣的可以了解一下

乘法指令是一種在CPU中實(shí)現(xiàn)的基本算術(shù)操作,用于計(jì)算兩個(gè)數(shù)的乘積。在匯編語言中,乘法指令通常是通過mul(無符號(hào)乘法)imul(有符號(hào)乘法)這兩個(gè)指令實(shí)現(xiàn)的。由于乘法指令在執(zhí)行時(shí)所消耗的時(shí)鐘周期較多,所以編譯器在優(yōu)化代碼時(shí)通常會(huì)嘗試將乘法操作轉(zhuǎn)換為更高效的加法、和移位操作。

  • 對(duì)于較小的數(shù),編譯器可能會(huì)選擇將乘法操作直接轉(zhuǎn)換為加法操作。例如,將表達(dá)式a * b轉(zhuǎn)換為a + a + ... + a(b次相加)的形式。這種方式可以通過循環(huán)展開、代碼向量化等技術(shù)來優(yōu)化。

  • 對(duì)于較大的數(shù),編譯器可能會(huì)使用位移和移位操作來代替乘法。例如,將表達(dá)式a * b轉(zhuǎn)換為a << n + a << m的形式,其中nm為符合條件的位數(shù)。這種方式可以通過位移指令的高效性來加速運(yùn)算。

當(dāng)以上方式均無法進(jìn)行優(yōu)化時(shí),編譯器才會(huì)使用mul/imul指令來執(zhí)行乘法操作。這兩條指令可以對(duì)無符號(hào)數(shù)和有符號(hào)數(shù)進(jìn)行乘法運(yùn)算,即便這兩條指令會(huì)使用更多的時(shí)鐘周期,但乘法指令的計(jì)算效率相對(duì)于其他指令DIV來說仍然較低,因此在編寫高效代碼時(shí),應(yīng)盡可能地避免使用乘法操作,并結(jié)合使用上面提到的技巧進(jìn)行優(yōu)化。

使用IMUL指令完成乘法

要計(jì)算乘法在不考慮執(zhí)行效率的情況下編譯器通常會(huì)直接使用imul指令完成計(jì)算,imul指令在一些情況下可以比其他乘法指令(如mul指令)更快地執(zhí)行乘法運(yùn)算,但性能較低的原因主要是由于imul指令通常用于有符號(hào)數(shù)的乘法運(yùn)算,并且在執(zhí)行時(shí)需要處理符號(hào)位的擴(kuò)展和溢出問題,這轉(zhuǎn)換成了額外的指令和時(shí)鐘周期的消耗。如果對(duì)于無符號(hào)整數(shù)或需要使用寄存器的低位或者高位結(jié)果的情況,使用imul指令可以提供一定的優(yōu)勢(shì)。

計(jì)算乘法時(shí)應(yīng)遵循:

  • 如果乘數(shù)與被乘數(shù)都是8位 則把AL做乘數(shù),結(jié)果放在AX
  • 如果乘數(shù)與被乘數(shù)都是16位 將把AX做乘數(shù),結(jié)果放在EAX
  • 如果乘數(shù)與被乘數(shù)都是32位 將把EAX做乘數(shù),結(jié)果放在EDX:EAX

乘法指令計(jì)算很簡單,只需要累加乘數(shù)即可,如下所示則是一個(gè)簡單的計(jì)算三個(gè)數(shù)相乘的匯編實(shí)現(xiàn);

.data
    x DWORD ?
    y DWORD ?
    z DWORD ?
    szFmt BYTE '計(jì)算結(jié)果: %d',0dh,0ah,0
.code
    main PROC
      mov dword ptr ds:[x],10
      mov dword ptr ds:[y],24
      mov dword ptr ds:[z],18
      
      ; 計(jì)算 x * y * z
      mov eax,dword ptr ds:[x]
      imul eax,dword ptr ds:[y]
      imul eax,dword ptr ds:[z]
      invoke crt_printf,addr szFmt,eax
    main ENDP
END main

使用LEA指令替換乘法

在實(shí)際編程中,我們可以使用LEA指令來替代乘法操作,從而提高代碼的執(zhí)行效率。但讀者需要注意,在使用LEA計(jì)算乘法時(shí)必須要保證乘數(shù)是2的次冪,并且乘數(shù)的范圍必須是2/4/8這三個(gè)區(qū)間才可使用該指令,我們使用匯編來實(shí)現(xiàn)計(jì)算eax*8+2其匯編指令如下。

  • 假設(shè) eax=5 計(jì)算 eax * 8 + 2 的結(jié)果,拆分過程如下:
  • 1.計(jì)算 lea ebx,dword ptr ds:[eax * 8 + 2] 這就相當(dāng)于計(jì)算 ebx = (eax * 8) +2直接可得到結(jié)果。

第一個(gè)案例比較簡單,可直接使用一條lea指令即可完成計(jì)算過程,只要保證被乘數(shù)是2的次冪即可。

.data
  x DWORD ?
  szFmt BYTE '計(jì)算結(jié)果: %d',0dh,0ah,0
.code
  main PROC
    ; 針對(duì)乘法的lea指令優(yōu)化
    mov dword ptr ds:[x],5
    
    mov eax,dword ptr ds:[x]               ; eax = x
    xor ebx,ebx                            ; ebx = 0
    lea ebx,dword ptr ds:[eax * 8 + 2]     ; ebx = eax * 8 + 2
    invoke crt_printf,addr szFmt,ebx
    
    invoke ExitProcess,0
  main ENDP
END main

使用LEA指令拆分計(jì)算

如果我們計(jì)算的乘法超出了2/4/8次冪范圍,則需要對(duì)乘法進(jìn)行拆分,拆分時(shí)也應(yīng)遵循2的次冪原則,拆分后在分開來計(jì)算。

  • 假設(shè) eax=3 計(jì)算 15 * eax 的結(jié)果,拆分過程如下:
  • 1.計(jì)算 lea edx,[eax * 4 + eax] 這就相當(dāng)于計(jì)算 edx = (4 * eax) + eax = 5eax 其中的每個(gè)edx就相當(dāng)于5個(gè)eax
  • 2.計(jì)算 lea edx,[edx * 2 + edx] 這就相當(dāng)于計(jì)算 edx = (5 * eax) * 2 + (5 * eax)
  • 3.計(jì)算 (5eax * 2) = 10eax 接著計(jì)算 (5 * eax) = 5eax 最后得出 10eax + 5eax
  • 4.經(jīng)過該過程可得出 eax * 15 = 45 最終計(jì)算3*15=45得到最終結(jié)果.

這個(gè)計(jì)算過程看似復(fù)雜,但如果將其轉(zhuǎn)化為匯編指令那么只需要兩條即可實(shí)現(xiàn)快速乘法運(yùn)算。

.data
  x DWORD ?
  szFmt BYTE '計(jì)算結(jié)果: %d',0dh,0ah,0
.code
  main PROC
    ; 針對(duì)乘法的lea指令優(yōu)化
    mov dword ptr ds:[x],3
    
    ; 如果使用lea計(jì)算乘法,則乘數(shù)必須是2/4/8
    mov eax,dword ptr ds:[x]               ; eax = 3
    lea edx,dword ptr ds:[eax * 4 + eax]   ; edx = 4eax + eax 得出 5eax,也就是說每一個(gè)edx就代表5個(gè)eax
    lea edx,dword ptr ds:[edx * 2 + edx]   ; edx = (5eax * 2) + 5eax 最終得出 15eax
    invoke crt_printf,addr szFmt,edx       ; edx = eax * 15 計(jì)算后得出 45
    
    invoke ExitProcess,0
  main ENDP
END main

使用LEA指令遞減計(jì)算

如果計(jì)算乘法時(shí)乘數(shù)非2的次冪,這種情況下需要減去特定的值,例如當(dāng)我們計(jì)算eax * 7時(shí),由于7非二的次冪,我們無法通過lea指令進(jìn)行計(jì)算,但我們可以計(jì)算eax * 8計(jì)算出的結(jié)果減去一個(gè)eax同樣可以得到正確的值。

  • 假設(shè) eax=3 計(jì)算 eax * 7 + 10 的結(jié)果,拆分過程如下:
  • 1.計(jì)算 lea edx,dword ptr ds:[eax * 8] 這就相當(dāng)于計(jì)算 edx = (8 * eax)
  • 2.計(jì)算 sub edx,eax 這就相當(dāng)于計(jì)算 edx = (8 * eax) - eax
  • 3.計(jì)算 add edx,10 這就相當(dāng)于計(jì)算 edx = ( (8 * eax) - eax ) + 10
  • 4.經(jīng)過如上計(jì)算,我們就可以計(jì)算出eax * 7 + 10的最終結(jié)果

這個(gè)計(jì)算過程看似復(fù)雜,但其實(shí)在匯編層面并不難構(gòu)建,如下分別實(shí)現(xiàn)計(jì)算兩個(gè)表達(dá)式求值過程。

.data
  x DWORD ?
  szFmt BYTE '計(jì)算結(jié)果: %d',0dh,0ah,0
.code
  main PROC
    ; 針對(duì)乘法的lea指令優(yōu)化
    mov dword ptr ds:[x],3
    
    ; 如果計(jì)算乘法時(shí)乘數(shù)非2的次冪,則此時(shí)需要減

    ; 計(jì)算 edx = eax * 7 + 10
    mov eax,dword ptr ds:[x]               ; eax = 3 => 計(jì)算 eax * 7 + 10
    lea edx,dword ptr ds:[eax * 8]         ; edx = eax * 8
    sub edx,eax                            ; edx = edx - eax
    add edx,10                             ; edx = edx + 10
    invoke crt_printf,addr szFmt,edx       ; edx = eax * 7 + 10
    
    ; 計(jì)算 edx = eax * 3 - 7
    mov eax,dword ptr ds:[x]               ; eax = 3 => 計(jì)算 eax * 3 - 7
    lea edx,dword ptr ds:[eax * 2]         ; edx = eax * 2
    add edx,eax                            ; edx = edx + eax
    sub edx,7                              ; edx = edx - 7
    invoke crt_printf,addr szFmt,edx       ; edx = eax * 3 - 7
    
    invoke ExitProcess,0
  main ENDP
END main

使用SHL計(jì)算無符號(hào)乘法

通過使用邏輯左移同樣可以實(shí)現(xiàn)2的次冪的高速乘法運(yùn)算,但邏輯左移只能用于計(jì)算無符號(hào)乘法,且只能計(jì)算被乘數(shù)是2的次方的算式。

  • 計(jì)算時(shí)我們需要參考次方表,這里我列舉出幾個(gè)常用的次方數(shù)值:

  • 次方表: 1=>2 2=>4 3=>8 4=>16 5=>32 6=>64 7=>128

  • 次方表: 8=>256 9=>512 10=>1024 11=>2048 12=>4096 13=>8192 14=>16384

  • 假設(shè) eax=3 計(jì)算 eax * 8 + 10 的結(jié)果,拆分過程如下:

  • 1.計(jì)算 shl eax,3 這就相當(dāng)于計(jì)算 eax = eax * 2 ^(次方) 3 其公式相當(dāng)于計(jì)算 eax = eax * 8

  • 2.計(jì)算 add eax,10 這就相當(dāng)于計(jì)算 eax = (eax * 8) + 10

  • 3.最終即可得到計(jì)算結(jié)果也就是3*8+10得到34

通過使用邏輯左移,我們可以實(shí)現(xiàn)快速無符號(hào)乘法運(yùn)算,如下代碼是效率最高的一種。

.data
  x DWORD ?
  szFmt BYTE '計(jì)算結(jié)果: %d',0dh,0ah,0
.code
  main PROC
    mov dword ptr ds:[x],3
    
    ; 計(jì)算 eax = eax * 2 ^ 1 相當(dāng)于計(jì)算 eax * 2
    mov eax,dword ptr ds:[x]
    shl eax,1
    invoke crt_printf,addr szFmt,eax
    
    ; 計(jì)算 eax = eax * 2 ^ 2 相當(dāng)于計(jì)算 eax * 4
    mov eax,dword ptr ds:[x]
    shl eax,2
    invoke crt_printf,addr szFmt,eax
    
    ; 計(jì)算 eax = eax * 2 ^ 3 相當(dāng)于計(jì)算 eax * 8
    mov eax,dword ptr ds:[x]
    shl eax,3
    add eax,10
    invoke crt_printf,addr szFmt,eax

    invoke ExitProcess,0
  main ENDP
END main

使用SAL計(jì)算有符號(hào)乘法

通過使用算數(shù)左移同樣可以實(shí)現(xiàn)2的次冪的高速乘法運(yùn)算,與邏輯左移不同,算術(shù)左移只能計(jì)算有符號(hào)乘法,且只能計(jì)算被乘數(shù)是2的次方的算式。

  • 計(jì)算時(shí)我們需要參考次方表,這里我列舉出幾個(gè)常用的次方數(shù)值:

  • 次方表: 1=>2 2=>4 3=>8 4=>16 5=>32 6=>64 7=>128

  • 次方表: 8=>256 9=>512 10=>1024 11=>2048 12=>4096 13=>8192 14=>16384

  • 假設(shè) eax=-5,ebx=3 計(jì)算 (eax * 8) + (ebx * 4) 的結(jié)果,拆分過程如下:

  • 1.計(jì)算 sal eax,3 這就相當(dāng)于計(jì)算 eax = (eax * 2 ^ 3 ) 其公式相當(dāng)于計(jì)算 eax = eax * 8 結(jié)果是一個(gè)有符號(hào)數(shù)

  • 2.計(jì)算 shl ebx,2 這就相當(dāng)于計(jì)算 ebx = (ebx * 2 ^2) 其公式相當(dāng)于計(jì)算 ebx = ebx * 4 結(jié)果是一個(gè)無符號(hào)數(shù)

  • 3.最終將有符號(hào)與無符號(hào)數(shù)通過 add eax,ebx 相加,即可得到(eax * 8) + (ebx * 4)的最終結(jié)果-28

如下是通過算數(shù)左移,實(shí)現(xiàn)2的次冪的高速乘法運(yùn)算,我們可以將算數(shù)運(yùn)算與邏輯運(yùn)算相加通過此方式提高運(yùn)算效率。

.data
  x DWORD ?
  y DWORD ?
  szFmt BYTE '計(jì)算結(jié)果: %d',0dh,0ah,0
.code
  main PROC
    mov dword ptr ds:[x],-5
    mov dword ptr ds:[y],3
    
    ; 計(jì)算 eax = eax * 2 ^ 1 相當(dāng)于計(jì)算 eax * 2
    mov eax,dword ptr ds:[x]
    sal eax,1
    invoke crt_printf,addr szFmt,eax
    
    ; 計(jì)算 eax = eax * 2 ^ 2 相當(dāng)于計(jì)算 eax * 4
    mov eax,dword ptr ds:[x]
    sal eax,2
    invoke crt_printf,addr szFmt,eax

    ; 計(jì)算 eax = (eax * 2 ^ 3 ) + (ebx * 2 ^2) 相當(dāng)于計(jì)算 (eax * 8) + (ebx * 4)
    mov eax,dword ptr ds:[x]
    mov ebx,dword ptr ds:[y]
    sal eax,3                  ; eax * 8 (有符號(hào)乘法)
    shl ebx,2                  ; ebx * 4 (無符號(hào)乘法)
    add eax,ebx                ; eax + ebx
    invoke crt_printf,addr szFmt,eax

    invoke ExitProcess,0
  main ENDP
END main

乘法優(yōu)化的知識(shí)點(diǎn)基本就這些,除了兩個(gè)未知變量的相乘無法優(yōu)化外,其他形式的乘法運(yùn)算均可以進(jìn)行優(yōu)化,如果表達(dá)式中存在一個(gè)常量值,那編譯器則會(huì)匹配各種優(yōu)化策略,最后對(duì)不符合優(yōu)化策略的運(yùn)算進(jìn)行調(diào)整,如果真的無法優(yōu)化,則會(huì)使用原始乘法指令計(jì)算。

到此這篇關(guān)于匯編高效乘法運(yùn)算的具體使用方法的文章就介紹到這了,更多相關(guān)匯編 乘法運(yùn)算內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 匯編跳轉(zhuǎn)指令使用總結(jié)

    匯編跳轉(zhuǎn)指令使用總結(jié)

    這篇文章主要介紹了匯編跳轉(zhuǎn)指令使用總結(jié),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-01-01
  • 匯編 函數(shù)調(diào)用的實(shí)現(xiàn)

    匯編 函數(shù)調(diào)用的實(shí)現(xiàn)

    這篇文章主要介紹了匯編 函數(shù)調(diào)用的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-02-02
  • 使用匯編語言實(shí)現(xiàn)if else 循環(huán)函數(shù)調(diào)用的具體方法

    使用匯編語言實(shí)現(xiàn)if else 循環(huán)函數(shù)調(diào)用的具體方法

    這篇文章主要介紹了使用匯編語言實(shí)現(xiàn)if else 循環(huán)函數(shù)調(diào)用的具體方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-01-01
  • 匯編:Debug的常用命令

    匯編:Debug的常用命令

    DEBUG是專門為匯編語言設(shè)計(jì)的一種調(diào)試工具,它通過步進(jìn),設(shè)置斷點(diǎn)等方式為匯編語言程序員提供了非常有效的調(diào)試手段
    2023-08-08
  • 一位數(shù)乘法的匯編語言實(shí)現(xiàn)方法

    一位數(shù)乘法的匯編語言實(shí)現(xiàn)方法

    這篇文章主要介紹了一位數(shù)乘法的匯編語言實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-02-02
  • 在vs2017中編寫匯編的實(shí)現(xiàn)(圖文)

    在vs2017中編寫匯編的實(shí)現(xiàn)(圖文)

    這篇文章主要介紹了在vs2017中編寫匯編的實(shí)現(xiàn)(圖文),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-03-03
  • 匯編語言MIPS指令分類及尋址模式原理概念

    匯編語言MIPS指令分類及尋址模式原理概念

    這篇文章主要為大家介紹了匯編語言MIPS指令分類及尋址模式的原理及概念,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步
    2021-11-11
  • 匯編語言乘指令 MUL、IMUL的具體使用

    匯編語言乘指令 MUL、IMUL的具體使用

    這篇文章主要介紹了匯編語言乘指令 MUL、IMUL的具體使用,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-02-02
  • 匯編基礎(chǔ)程序編寫教程示例

    匯編基礎(chǔ)程序編寫教程示例

    這篇文章主要為大家介紹了匯編基礎(chǔ),程序編寫教程示例,文中附含詳細(xì)的圖文示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助
    2021-11-11
  • 匯編語言:比較指令、跳轉(zhuǎn)指令、JCC的使用

    匯編語言:比較指令、跳轉(zhuǎn)指令、JCC的使用

    這篇文章主要介紹了匯編語言:比較指令、跳轉(zhuǎn)指令、JCC的使用,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-01-01

最新評(píng)論