cl编译器 不符合四则运算

本文探讨了在C编译器开发过程中遇到的问题,包括四则运算规则的应用、寄存器不足的挑战以及如何通过代码优化解决这些问题。以具体代码为例,展示了编译器如何处理复杂的算术表达式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

最近在写一个c编译器,为什么要写呢——为一个UI系统做runtime


中途发现自己的方法不符合四则运算的规则,于是又改程序,发现程序的复杂性大大增加了,而且还坑爹地发现“x86的寄存器不够用”这句话的悲剧含义。。悲催中看看cl是怎样利用x86的可怜的寄存器的。


结果发现cl编译器也是这个吊样。。(下面黑体部分)

(注:使用其他编译器发现也是这样,这应该是一个标准,在一个语句中,乘法和括号并不是优先级最高的)


#include <stdio.h>
int a=6;
int adda()
{
a++;
return a;
}
void main()
{
int b;
b=0;
a=b+5*3*a+4*adda()+7*a+8*a;
}



cl ass.c /FAs  反编译



; Listing generated by Microsoft (R) Optimizing Compiler Version 16.00.30319.01

TITLE C:\Documents and Settings\Administrator\桌面\ass.c
.686P
.XMM
include listing.inc
.model flat

INCLUDELIB LIBCMT
INCLUDELIB OLDNAMES

PUBLIC _a
_DATA SEGMENT
_a DD 06H
_DATA ENDS
PUBLIC _adda
; Function compile flags: /Odtp
; File c:\documents and settings\administrator\桌面\ass.c
_TEXT SEGMENT
_adda PROC

; 4 : {

push ebp
mov ebp, esp

; 5 : a++;

mov eax, DWORD PTR _a
add eax, 1
mov DWORD PTR _a, eax

; 6 : return a;

mov eax, DWORD PTR _a

; 7 : }

pop ebp
ret 0
_adda ENDP
_TEXT ENDS
PUBLIC _main
; Function compile flags: /Odtp
_TEXT SEGMENT
_b$ = -4 ; size = 4
_main PROC

; 9 : {

push ebp
mov ebp, esp
push ecx
push esi

; 10 : int b;
; 11 : b=0;

mov DWORD PTR _b$[ebp], 0

; 12 : a=b+5*3*a+4*adda()+7*a+8*a;

mov esi, DWORD PTR _a
imul esi, 15 ; 0000000fH
add esi, DWORD PTR _b$[ebp]
call _adda

lea eax, DWORD PTR [esi+eax*4]
mov ecx, DWORD PTR _a
imul ecx, 7
add eax, ecx
mov edx, DWORD PTR _a
lea eax, DWORD PTR [eax+edx*8]
mov DWORD PTR _a, eax

; 13 : }

xor eax, eax
pop esi
mov esp, ebp
pop ebp
ret 0
_main ENDP
_TEXT ENDS
END

### 关于 Microsoft CL 编译器的信息 Microsoft CL 是微软提供的命令行 C 和 C++ 编译器,通常作为 Visual Studio 的一部分提供。以下是关于其基本功能以及如何使用的详细介绍。 #### 安装与获取 要使用 Microsoft CL 编译器,需先安装 Visual Studio 或者单独的构建工具包。可以通过以下链接下载并安装所需的开发环境[^3]: ```plaintext https://visualstudio.microsoft.com/zh-hans/downloads/ ``` 在安装过程中,请确保选择了“C++ 工作负载”,这会自动包含 CL 编译器及其相关组件。 --- #### 基本语法结构 CL 编译器通过命令行调用,支持多种选项来控制编译过程。最基本的调用形式如下所示: ```cmd cl [options] filename[s] ``` 其中 `filename[s]` 表示待编译的一个或多个源文件名,而 `[options]` 则用于指定各种参数,例如优化级别、目标架构等。 --- #### 目标平台定义宏 `_M_IX86` 当针对 x86 处理器进行编译时,CL 编译器会定义 `_M_IX86` 宏,并将其设置为整数值 600。此行为有助于开发者编写条件预处理代码以区分同的硬件架构[^1]。需要注意的是,在其他平台上(如 x64 或 ARM),该宏会被定义。 --- #### 配置与编译流程 类似于许多 Linux 下的传统项目,基于 Windows 平台上的程序也可以遵循类似的构建模式:即准备源码 -> 配置环境变量 (如果必要的话) -> 执行实际编译操作 -> 将最终产物部署到适当位置[^2]。然而具体实现方式可能有所差异,特别是在跨操作系统移植场景下更明显。 对于简单的单文件应用程序而言,可以直接运行上述提到的基础命令完成整个工作流;而对于复杂工程,则推荐借助像 Makefile 或现代构建系统(CMake)这样的工具来进行管理[^4]。 查看完整的帮助文档可以执行下面这条指令: ```cmd cl /? ``` 或者查阅官方在线资源获得更多细节说明。 --- ### 示例代码片段展示 这里给出一段简单例子演示如何利用 cl.exe 来创建可执行二进制文件: 假设有一个名为 hello.cpp 的源文件内容如下: ```cpp #include <iostream> int main() { std::cout << "Hello, world!" << std::endl; return 0; } ``` 可以在终端输入以下命令对其进行编译: ```cmd cl hello.cpp ``` 成功之后就会生成对应的 EXE 文件供后续测试验证之用了! ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值