• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            平凡的世界

            神鷹忽展翅,頭頂青天飛
            隨筆 - 10, 文章 - 0, 評(píng)論 - 34, 引用 - 0
            數(shù)據(jù)加載中……

            [原][譯] Tips for Better Coding Style --- 關(guān)于更好的編程風(fēng)格的建議

            [C/C++] - Tips for Better Coding Style

            關(guān)于更好的編程風(fēng)格的建議 (v1.5)

            Translated By Phoenix(phoenix8848@gmail.com)

                In this entry, I show you 4 tips that address frequently asked questions from C++ programmers of all levels of expertise. It's surprising to discover how many experienced programmers are still unaware of the deprecation of the .h notation of standard header files, the proper usage of namespaces, and the rules regarding binding of references to temporary objects, for example. These issues and others will be discussed here.
                在這篇文章里我將談?wù)劯鞣N層次的C++程序員經(jīng)常問及的四個(gè)問題。例如我很驚訝地發(fā)現(xiàn)還有很多程序員沒有意識(shí)到標(biāo)準(zhǔn)頭文件擴(kuò)展名.h的爭(zhēng)議,命名空間的恰當(dāng)用法以及引用臨時(shí)對(duì)象的規(guī)則。這些問題及其它將在這里進(jìn)行討論。

                First, we start by explaining the difference between the deprecated “xxx.h” header names and the modern, standard-compliant “xxx” header-naming notation. Next, we explore a few dark corners of C++ which due to compilers' limitations and the somewhat recondite nature of the associated language rules tend(原文為“rulestend”) to confuse many programmers, e.g., the notion of comma-separated expressions and the rules of binding references to rvalues. Finally, we will learn how to invoke a function prior to a program's startup.

                首先我們從解釋受非議的“XXX.h”頭文件名與現(xiàn)代、符合標(biāo)準(zhǔn)的“<XXX>”頭文件名記號(hào)之間的區(qū)別開始。接下來我們探索C++不為人知的角落,由于編譯器的局限性和關(guān)聯(lián)語言規(guī)則某些隱蔽的自然特性迷惑了許多程序員,比如逗號(hào)分隔表達(dá)式的意義與引用型變量的規(guī)則。最后我們將學(xué)習(xí)如何在程序啟動(dòng)前啟動(dòng)一個(gè)函數(shù)。

                Tip 1: “iostream.h” or “iostream”?

                語題1:“iostream.h” or “iostream”?

                Many C++ programmers still use “iostream.h” instead of the newer, standard compliant “iostream” library. What are the differences between the two? First, the .h notation of standard header files was deprecated more than five years ago. Using deprecated features in new code is never a good idea. In terms of functionality, “iostream” contains a set of templatized I/O classes which support both narrow and wide characters, as opposed to “iostream.h” which only supports char-oriented streams. Third, the C++ standard specification of iostream's interface was changed in many subtle aspects. Consequently, the interfaces and implementation of “iostream” differ from those of “iostream.h”. Finally, “iostream” components are declared in namespace std whereas “iostream.h” components are global.

                 很多C++程序員還在使用“iostream.h”代替新的符合標(biāo)準(zhǔn)的“iostream”庫(kù)。兩者有什么區(qū)別呢?首先,標(biāo)準(zhǔn)頭文件“.h”擴(kuò)展名在五年前就倍受爭(zhēng)議。在新代碼中使用有爭(zhēng)議的(過時(shí)的)特性永遠(yuǎn)都不是一個(gè)好主意。從本質(zhì)上看,“iostream”包括一系列支持窄字符與寬字符的模板化(templatized) I/O輸入輸出類,而相反地,“iostream.h”只支持字符流。第三,iostream接口的標(biāo)準(zhǔn)C++規(guī)范在許多細(xì)節(jié)方面進(jìn)行了變動(dòng)。因此,“iostream”的接口與實(shí)現(xiàn)同那些“iostream.h”是有區(qū)別的。最后,“iostream”是在std命名空間中定義的而“iostream.h”則是全局的。

                Because of these substantial differences, you cannot mix the two libraries in one program. As a rule, use “iostream” unless you're dealing with legacy code that is only compatible with “iostream.h”.

                由于這些本質(zhì)方面的不同,不能在同一程序中混合使用這兩種庫(kù)。作為一個(gè)規(guī)則,應(yīng)盡量使用“iostream”,除非你要處理的是只能與“iostream.h”兼容的遺留代碼。(感謝autumnm1981

                Tip 2: Binding a Reference to an R-Value

                話題2:將引用與右值綁定

                (R-Value:右值,與“左值”相對(duì)。例如x=3中,“x”是一個(gè)“左值”,“3”是一個(gè)右值。從本質(zhì)上講,左值是一個(gè)內(nèi)存的地址,右值是一個(gè)實(shí)際的二進(jìn)制值。)

                R-Values and L-Values are a fundamental concept of C++ programming. In essence, an R-Value is an expression that cannot appear on the left-hand side of an assignment expression. By contrast, an L-Value refers to an object (in its wider sense), or a chunk of memory, to which you can write a value. References can be bound to both R-Values and L-Values. However, due to the language's restrictions regarding R-Values, you have to be aware of the restrictions on binding references to R-Values, too.

                右值和左值是C++編程的一個(gè)基本概念。本質(zhì)上來講右值是一個(gè)不可能出現(xiàn)在等號(hào)左邊的表達(dá)式。相反,左值引用一個(gè)對(duì)象(廣義范圍上的),或者一塊可讀寫的內(nèi)存。引用既可以指向右值也可以指向左值。然而,由于語言在處理右值上的限制,你也得在將引用指向右值是慎重考慮。

                Binding a reference to an R-Value is allowed as long as the reference is bound to a const type. The rationale behind this rule is straightforward: you can't change an R-Value, and only a reference to const ensures that the program doesn't modify an R-Value through its reference. In the following example, the function f() takes a reference to const int:

                將引用與右值綁定像引用常量(這里const type應(yīng)為"常量",感謝新奧爾良土鱉 )一樣也是被允許的。這條原則背后的原理是很顯而易見的:你無法改變右值,因?yàn)閷?duì)常量的引用確保程序不會(huì)通過這個(gè)接口改變右值。下面的例子,f()函數(shù)包含一個(gè)對(duì)整型變量的引用。

            1 void f(const int & i);
            2 
            3 int main()
            4 {
            5 f(2); /* OK */
            6 }


                The program passes the R-Value 2 as an argument to f(). At runtime, C++ creates a temporary object of type int with the value 2 and binds it to the reference i. The temporary and its reference exist from the moment f() is invoked until it returns; they are destroyed immediately afterwards. Note that had we declared the reference i without the const qualifier, the function f() could have modified its argument, thereby causing undefined behavior. For this reason, you may only bind references to const objects.

                這段代碼將右值“2”做為函數(shù)f()的一個(gè)參數(shù)。代碼運(yùn)行時(shí),C++將創(chuàng)建一個(gè)值為2的臨時(shí)整型變量并將其與引用類型i綁定。這個(gè)臨時(shí)對(duì)象與它的接口將在 f()運(yùn)行期間一直存到直到函數(shù)f返回。函數(shù)f返回后它們立即被釋放。注意我們沒有將i聲明為常量類型,但是函數(shù)f仍有可能修改它的這個(gè)參數(shù),這將引起異常。因此最好是將引用與常量類型綁定。

                The same rule applies to user-defined objects. You may bind a reference to a temporary object only if it's const:

                同樣的規(guī)則適用于自定義類型。只有一個(gè)臨時(shí)對(duì)象為常量時(shí)才可以與引用類型綁定。

            struct A{};

            1 void f(const A& a);
            2 
            3 int main()
            4 {
            5     f(A()); /* OK, binding a temporary A to a const reference*/
            6 }


                Tip 3: Comma-Separated Expressions

                話題3:逗號(hào)表達(dá)式

            Comma-separated expressions were inherited from C. It's likely that you use such expressions in for- and while-loops rather often. Yet, the language rules in this regard are far from being intuitive. First, let's see what a comma separated expression is.

                逗號(hào)表達(dá)式是從C語言沿襲下來的。它就像你經(jīng)常使用的for循環(huán)與while-loop循環(huán)一樣。但是這里面的語法規(guī)則遠(yuǎn)不像看起來的那樣。首先,讓我們看看什么是逗號(hào)表達(dá)式。

                An expression may consist of one or more sub-expressions separated by commas. For example:

                一個(gè)表達(dá)式可以被逗號(hào)分隔為一個(gè)或若干個(gè)子表達(dá)式。例如:

            1     if(++x, --y, cin.good()) /*three expressions*/


            The if condition contains three expressions separated by commas. C++ ensures that each of the expressions is evaluated and its side effects take place. However, the value of an entire comma-separated expression is only the result of the rightmost expression. Therefore, the if condition above evaluates as true only if cin.good() returns true. Here's another example of a comma expression:

                這條if語句被逗號(hào)分事為三個(gè)表達(dá)式。從C++的角度每個(gè)表達(dá)式都是合法的但是副作用產(chǎn)生了。整個(gè)逗號(hào)表達(dá)式的值是由最右邊的表達(dá)式?jīng)Q定的。于是只有con.good()的返回值是true時(shí)整個(gè)表達(dá)式的值才是true。這里有另一個(gè)關(guān)于逗號(hào)表達(dá)式的例子。

            1 int j=10;
            2 int i=0;
            3 
            4 while++i, --j)
            5 {
            6     /*..repeat as long as j is not 0*/
            7 }


                Tip 4: Calling a Function Before Program's Startup

                在程序啟動(dòng)前調(diào)用函數(shù)

            Certain applications need to invoke startup functions that run before the main program starts. For example, polling, billing, and logger functions must be invoked before the actual program begins. The easiest way to achieve this is by calling these functions from a constructor of a global object. Because global objects are conceptually constructed before the program's outset, these functions will run before main() starts. For example:

                有些應(yīng)用需要在主程序啟運(yùn)前運(yùn)行啟動(dòng)函數(shù)。例如投票、支付和登錄函數(shù)必須在實(shí)際種程序啟動(dòng)前運(yùn)行。一個(gè)最簡(jiǎn)單的實(shí)現(xiàn)方法就是在一個(gè)全局對(duì)象的構(gòu)造函數(shù)里調(diào)用這些函數(shù)。因?yàn)槿謱?duì)象在程序的最開頭被隱式的創(chuàng)建,這些函數(shù)就可以在main()函數(shù)之前得到運(yùn)行。例如:

             1 class Logger
             2 {
             3 public:
             4     Logger()
             5     {
             6         activate_log();
             7     }
             8 };
             9 
            10 Logger log; /*global instance*/
            11 
            12 int main()
            13 {
            14     record * prec=read_log();
            15     //.. application code
            16 }


                The global object log is constructed before main() starts. During its construction, log invokes the function activate_log(). Thus, when main() starts, it can read data from the log file.

                全局對(duì)象log在main()函數(shù)啟動(dòng)之前被創(chuàng)建。在它的構(gòu)造函數(shù)里,log調(diào)用了active_log()函數(shù)。于是,當(dāng)main()函數(shù)啟動(dòng)時(shí),它可以從日志文件中讀取數(shù)據(jù)。


            posted on 2008-09-15 23:13 西門有悔 閱讀(2465) 評(píng)論(12)  編輯 收藏 引用

            評(píng)論

            # re: [原][譯]關(guān)于更好的編程風(fēng)格的建議  回復(fù)  更多評(píng)論   

            很好的東西呀!
            Tip1里講的東西我以前還真沒意識(shí)到,一直用.h呢,呵呵。
            2008-09-16 00:14 | abettor

            # re: [原][譯]關(guān)于更好的編程風(fēng)格的建議  回復(fù)  更多評(píng)論   

            @abettor

            謝謝。

            我是第一次試著翻譯一些自己認(rèn)為比較好的文檔,一方面是為了增加自己的知識(shí),另一方面也是為與大家一起分享。

            有什么翻譯得不好的地方歡迎討論。我的email:phoenix8848@gmail.com
            2008-09-16 01:42 | 西門有悔

            # re: [原][譯]關(guān)于更好的編程風(fēng)格的建議  回復(fù)  更多評(píng)論   

            As a rule, use “iostream” unless you're dealing with legacy code that is only compatible with “iostream.h”.
            結(jié)論為除非在處理與“iostream”保持兼容的歷史遺留代碼時(shí)否則最好使用“iostream.h”。

            這里翻譯有誤,應(yīng)該譯為:應(yīng)盡量使用“iostream”,除非你要處理的是只能與“iostream.h”兼容的遺留代碼。
            2008-09-16 09:06 | autumnm1981

            # re: [原][譯]關(guān)于更好的編程風(fēng)格的建議  回復(fù)  更多評(píng)論   

            const應(yīng)該翻譯成常量
            靜態(tài)是static
            2008-09-16 10:02 | 新奧爾良土鱉

            # re: [原][譯]關(guān)于更好的編程風(fēng)格的建議  回復(fù)  更多評(píng)論   

            為什么都喜歡留著英文?一跳一跳的,覺得不方便閱讀啊
            2008-09-16 11:52 | 肥仔

            # re: [原][譯]關(guān)于更好的編程風(fēng)格的建議[未登錄]  回復(fù)  更多評(píng)論   

            因?yàn)橛袝r(shí)候翻譯可能不正確
            2008-09-16 13:05 | 陳梓瀚(vczh)

            # re: [原][譯]關(guān)于更好的編程風(fēng)格的建議  回復(fù)  更多評(píng)論   

            @autumnm1981

            對(duì)啊。非常感謝!我馬上更新
            2008-09-16 15:52 | 西門有悔

            # re: [原][譯]關(guān)于更好的編程風(fēng)格的建議  回復(fù)  更多評(píng)論   

            @新奧爾良土鱉

            寒啊。竟然犯這樣低級(jí)的錯(cuò)誤。也反映了平時(shí)我以靜態(tài)變量與常量沒有在意。

            謝謝。馬上更新。
            2008-09-16 15:53 | 西門有悔

            # re: [原][譯]關(guān)于更好的編程風(fēng)格的建議  回復(fù)  更多評(píng)論   

            @陳梓瀚(vczh)

            對(duì)啊。昨天晚上沒有睡覺,半睡半醒狀態(tài)下翻的。說實(shí)話很多英文的文檔看起來還是很費(fèi)勁。雖然俺也過了CET6,不過是五年前的事了。
            2008-09-16 15:55 | 西門有悔

            # re: [原][譯] Tips for Better Coding Style --- 關(guān)于更好的編程風(fēng)格的建議[未登錄]  回復(fù)  更多評(píng)論   

            睡覺是人生的終極目標(biāo)……
            2008-09-17 13:10 | 陳梓瀚(vczh)

            # re: [原][譯] Tips for Better Coding Style --- 關(guān)于更好的編程風(fēng)格的建議  回復(fù)  更多評(píng)論   

            Thx to your great sharing!
            From:Kven
            2008-09-18 00:40 | Kven

            # re: [原][譯] Tips for Better Coding Style --- 關(guān)于更好的編程風(fēng)格的建議  回復(fù)  更多評(píng)論   

            The same rule applies to user-defined objects. You may bind a reference to a temporary object only if it's const:

            同樣的規(guī)則適用于自定義類型。只有一個(gè)臨時(shí)對(duì)象為常量時(shí)才可以與引用類型綁定。


            這里的"only if it's const"感覺應(yīng)該是修飾"reference",即翻譯為“只有當(dāng)引用為const時(shí)才可以將其綁定到一個(gè)臨時(shí)對(duì)象”,《C++編程思想》上好像也提到了這點(diǎn),不知道是不是我理解有誤。
            2009-07-22 10:01 | k.j

            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            日日躁夜夜躁狠狠久久AV| 亚洲精品高清一二区久久| 精品国产一区二区三区久久久狼| 国产精品久久久久久久久| 99久久99久久精品国产| 漂亮人妻被中出中文字幕久久| 色综合久久久久无码专区| 99久久精品免费国产大片| 亚洲综合久久夜AV | 国产精品久久精品| 中文精品99久久国产| 国产精品VIDEOSSEX久久发布| 精品久久久久久无码不卡| 伊人丁香狠狠色综合久久| 精品国产99久久久久久麻豆| 久久久久这里只有精品| 99久久无码一区人妻a黑| 狠狠综合久久综合88亚洲| 久久精品国产99久久丝袜| 人妻无码αv中文字幕久久琪琪布| 久久露脸国产精品| 狠狠色婷婷综合天天久久丁香| 久久妇女高潮几次MBA| 中文字幕精品无码久久久久久3D日动漫| 99久久精品国内| 人妻无码αv中文字幕久久| A级毛片无码久久精品免费| 国内精品久久久久影院老司| 久久精品这里只有精99品| 很黄很污的网站久久mimi色| 成人精品一区二区久久 | 91精品观看91久久久久久| 国产偷久久久精品专区| 久久天天躁狠狠躁夜夜躁2014| 亚洲精品美女久久久久99小说| 久久精品亚洲欧美日韩久久| 久久久精品久久久久久| 亚洲国产日韩欧美综合久久| 久久人妻少妇嫩草AV蜜桃| 久久人人爽人人澡人人高潮AV| 国产精品美女久久久久av爽|