いずれにしても、コンパイラがこれらの2つのコードを実際に生成するかどうかは私には分かります。 srcとdstはインクリメントされていますが、決して使用されていないので、コンパイラは生成されたマシンコード内でそれらを変数として永続的に保存しようとしないと知っていると思いました。
32ビットリリースモード(/ O2)のVS 2010 C++ SP1ビルドでwindows7を使用すると、上記の両方のインカネーションのディスアセンブルコードが取得されました。関数自体が入力を直接参照したり、インライン展開されたりするのを防ぐため、関数ごとにdllを作成しました。私はASMのプロローグとエピローグを省略しました。
while (*dst++ = *src++)
6EBB1003 8B 55 08 mov edx,dword ptr [src]
6EBB1006 8B 45 0C mov eax,dword ptr [dst]
6EBB1009 2B D0 sub edx,eax //prepare edx so that edx + eax always points to src
6EBB100B EB 03 jmp docopy+10h (6EBB1010h)
6EBB100D 8D 49 00 lea ecx,[ecx] //looks like align padding, never hit this line
6EBB1010 8A 0C 02 mov cl,byte ptr [edx+eax] //ptr [edx+ eax] points to char in src :loop begin
6EBB1013 88 08 mov byte ptr [eax],cl //copy char to dst
6EBB1015 40 inc eax //inc src ptr
6EBB1016 84 C9 test cl,cl // check for 0 (null terminator)
6EBB1018 75 F6 jne docopy+10h (6EBB1010h) //if not goto :loop begin
;
上記のコードには、基本的に1つのループがあり、nullと1つのメモリコピーのみがチェックされます。
今すぐ私のミスのバージョンを見てみましょう:私のバージョンで
while (*dst = *src)
6EBB1003 8B 55 08 mov edx,dword ptr [src]
6EBB1006 8A 0A mov cl,byte ptr [edx]
6EBB1008 8B 45 0C mov eax,dword ptr [dst]
6EBB100B 88 08 mov byte ptr [eax],cl //copy 0th char to dst
6EBB100D 84 C9 test cl,cl //check for 0
6EBB100F 74 0D je docopy+1Eh (6EBB101Eh) // return if we encounter null terminator
6EBB1011 2B D0 sub edx,eax
6EBB1013 8A 4C 02 01 mov cl,byte ptr [edx+eax+1] //get +1th char :loop begin
{
src++;
dst++;
6EBB1017 40 inc eax
6EBB1018 88 08 mov byte ptr [eax],cl //copy above char to dst
6EBB101A 84 C9 test cl,cl //check for 0
6EBB101C 75 F5 jne docopy+13h (6EBB1013h) // if not goto :loop begin
}
、私はそれ最初にコピー先への0番目のcharことがわかり、その後、ヌルをチェックした後、最終的にはそれがチェックするループに入りますもう一度nullになります。したがって、ループはほとんど同じままですが、ループの前の0番目の文字を処理します。これは、もちろん、最初の場合と比較して準最適になるでしょう。
コンパイラが最初の例と同じ(またはほぼ同じ)コードを作成できない理由が分かっているのでしょうか。これはmsコンパイラ特有の問題か、おそらく私のコンパイラ/リンカ設定ですか?
ここでは完全なコードですが、2つのファイル(1つの機能がもう一方の機能を置き換えます)。
// in first dll project
__declspec(dllexport) void docopy(const char* src, char* dst)
{
while (*dst++ = *src++);
}
__declspec(dllexport) void docopy(const char* src, char* dst)
{
while (*dst = *src)
{
++src;
++dst;
}
}
//seprate main.cpp file calls docopy
void docopy(const char* src, char* dst);
char* source ="source";
char destination[100];
int main()
{
docopy(source, destination);
}
あなたが始めたCコード全体を投稿してください。 srcとdstの宣言のために異なるかもしれませんが、私は知ることができません。削除したアセンブラプリアンブルは同じですか?それがあれば貼り付ける必要はありません。 – gbulmer
どちらの場合も、エピローグとプロローグを除いて、コード全体です。 – skimon
これは、多くの読者が式内の「=」が「==」の入力ミスとして認識するため、コーディングのコーディングスタイルが悪いことです。 –