我不喜欢汇编中的编码,我只需要在汇编中编写,无论如何这里是我想要做的。 (我正在使用Masm32)
.data
Msg db 31h, 00h, 32h, 00h, 33h ;convert this to string which "123"
.code
start:
invoke MessageBox, 0, addr Msg, 0, 0
call ExitProcess
end start
正如您所看到的,每个字符或字节用空字节分隔,它只能显示一个字节,“1”而不是“123”
如果只有我可以连接每个可读字节,直到它到达字符串的末尾。
mov ebx, offset Msg
mov ecx, ebx
add ebx, 2
invoke lstrcat, ebx, ecx
也许还添加一个循环,我只是不知道更好的编码方式,或者你有一个更好的解决方案,你可以分享。
答案 0 :(得分:0)
为了处理字符串,您需要知道字符串的长度。这是通过使用NULL来终止字符串来完成的。不将字符串长度作为参数的函数将查找00H,并将字符串打印到它,这就是MessageBox正在执行的操作,打印“1”并看到下一个字符是00h,停止打印。
在你的情况下,如果字符串末尾没有NULL终止符,我们可以在汇编时获得字符串长度,其符号为“$”,这是位置计数器的当前值。
Msg db 31h, 00h, 32h, 00h, 33h ;convert this to string which "123"
Msg_Len equ $ - Msg
这意味着,取$的地址并减去Msg的地址,并用该值替换Msg_Len。
include masm32rt.inc
.data
Msg db 31h, 00h, 32h, 00h, 33h ;convert this to string which "123"
Msg_Len equ $ - Msg
.data?
Msg_New db Msg_Len + 1 dup (?)
.code
start:
xor ecx, ecx ; index into source array
xor edx, edx ; index into dest array
lea esi, Msg
lea edi, Msg_New
ConvertIt:
mov al, byte ptr[esi + ecx] ; get byte at pointer + index
cmp al, 0 ; is it a 0?
jne @F ; no, add it to dest array
inc ecx ; yes, increase source index
jmp CheckForEnd ; check to see if at end
@@:
mov byte ptr [edi + edx], al; add byte to dest array
inc ecx ; increase both indexes
inc edx ;
CheckForEnd:
cmp ecx, Msg_Len ; check for end - if ecx == Msg_Len we are done
jne ConvertIt ; if ecx != Msg_Len, continue loop
mov byte ptr [edi + edx], 0 ; don't forget to NULL terminate dest string
invoke MessageBox, HWND_DESKTOP, offset Msg_New, NULL, MB_OK
invoke ExitProcess, 0
end start
这可以用更少的代码完成,并“将”转换为“就地”字符串,但这超出了这里的范围。
<@> @Nik,我们中的一些人喜欢大会,不要使用任何其他东西。