計(jì)算機(jī)專(zhuān)業(yè)畢業(yè)設(shè)計(jì)外文翻譯--面向java開(kāi)發(fā)人員的scala指南類(lèi)操作_第1頁(yè)
已閱讀1頁(yè),還剩21頁(yè)未讀, 繼續(xù)免費(fèi)閱讀

下載本文檔

版權(quán)說(shuō)明:本文檔由用戶(hù)提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)

文檔簡(jiǎn)介

1、<p>  來(lái)自:think in java (3)</p><p><b>  外文原文</b></p><p>  The busy Java developer's guide to Scala: Class action</p><p>  It makes sense for Java? developers to

2、use objects as a first point of reference for understanding Scala. In this second installment of The busy Java developer's guide to Scala series, Ted Neward follows a basic premise of language measurement: that the p

3、ower of a language can be measured in direct relation to its ability to integrate new facilities -- in this case, support for complex numbers. Along the way you'll see some interesting tidbits related to class defini

4、tions and usage in Scala. In </p><p>  Scala's functional programming features are compelling, but they're not the only reason Java developers should be interested in the language. In fact, Scala ble

5、nds functional concepts and object orientation. In order to let the Java-cum-Scala programmer feel more at home, it makes sense to look at Scala's object features and see how they map over to Java linguistically. Bea

6、r in mind that there isn't a direct mapping for some of these features, or in some cases, the "mapping" is more of an analog</p><p>  Scala has class(es), too</p><p>  Rather than

7、embark on a lengthy and abstract discussion of the class features that Scala supports, let's look at a definition for a class that might be used to bring rational number support to the Scala platform (largely swiped

8、from "Scala By Example" -- see Resources):</p><p>  Listing 1. rational.scala</p><p>  While the overall structure of Listing 1 is lexically similar to what you've seen in Java cod

9、e over the last decade, some new elements clearly are at work here. Before picking this definition apart, take a look at the code to exercise the new Rational class:</p><p>  Listing 2. RunRational</p>

10、<p>  What you see in Listing 2 isn't terribly exciting: I create a couple of rational numbers, create two more Rationals as the addition and subtraction of the first two, and echo everything to the console. (

11、Note that Console.println() comes from the Scala core library, living in scala.*, and is implicitly imported into every Scala program, just as java.lang is in Java programming.)</p><p>  How many ways shall

12、I construct thee?</p><p>  Now look again at the first line in the Rational class definition:</p><p>  Listing 3. Scala's default constructor</p><p>  Although you might think y

13、ou're looking at some kind of generics-like syntax in Listing 3, it's actually the default and preferred constructor for the Rational class: n and d are simply the parameters to that constructor.</p><p

14、>  Scala's preference for a single constructor makes a certain kind of sense -- most classes end up having a single constructor or a collection of constructors that all "chain" through a single construct

15、or as a convenience. If you wanted to, you could define more constructors on a Rational like so:</p><p>  Listing 4. A chain of constructors</p><p>  Note that Scala's constructor chain does

16、 the usual Java-constructor-chaining thing by calling into the preferred constructor (the Int,Int version).</p><p>  Details, (implementation) details...</p><p>  When working with rational numb

17、ers, it helps to perform a bit of numerical legerdemain: namely that of finding a common denominator to make certain operations easier. If you want to add 1-over-2 (also known as "one-half") to 2-over-4 (also k

18、nown as "two-fourths"), the Rational class should be smart enough to realize that 2-over-4 is the same as 1-over-2, and convert it accordingly before adding the two together.</p><p>  This is the p

19、urpose of the nested private gcd() function and g value inside of the Rational class. When the constructor is invoked in Scala, the entire body of the class is evaluated, which means g will be initialized with the greate

20、st common denominator of n and d, and then used in turn to set n and d appropriately.</p><p>  Looking back at Listing 1, it's also fairly easy to see that I created an overridden toString method to retu

21、rn the values of Rational, which will be very useful when I start exercising it from the RunRational driver code.</p><p>  Notice the syntax around toString, however: the override keyword in the front of the

22、 definition is required so that Scala can check to make sure that a corresponding definition exists in the base class. This can help prevent subtle bugs created by accidental keyboard slips. (It was this same motivation

23、that led to the creation of the @Override annotation in Java 5.) Notice, as well, that the return type is not specified -- it's obvious from the definition of the method body -- and that the retur</p><p>

24、;  Some core values</p><p>  Next up are the definitions of numer and denom, respectively. The syntax involved, offhand, would lead the Java programmer to believe that numer and denom are public Int fields t

25、hat are initialized to the value of n-over-g and d-over-g, respectively; but this assumption is incorrect.</p><p>  Formally, Scala calls numer and denom methods without parameters, which are used to create

26、a quick-and-easy syntax for defining accessors. The Rational class still has three private fields, n, d, and g, but they are hidden from the world by default private access in the case of n and d, and by explicit private

27、 access in the case of g.</p><p>  The Java programmer in you is probably asking at this point, "Where are the corresponding "setters" for n and d?" No such setters exist. Part of the pow

28、er of Scala is that it encourages developers to create immutable objects by default. Granted, syntax is available to create methods for modifying the internals of Rational, but doing so would ruin the implicit thread-saf

29、e nature of this class. As a result, at least for this example, I'm going to leave Rational as it is.</p><p>  Naturally, that raises the question of how one manipulates a Rational. Like java.lang.String

30、s, you can't take an existing Rational and modify its values, so the only alternative is to create new Rationals out of the values of an existing one, or create it from scratch. This brings into focus the next set of

31、 four methods: the curiously named +, -, *, and / methods.</p><p>  And no, contrary to what it might look like, this isn't operator-overloading.</p><p>  Operator, ring me a number</p>

32、;<p>  Remember that in Scala everything is an object. In the last article, you saw how that principle applies to the idea that functions themselves are objects, which allows Scala programmers to assign functions

33、to variables, pass functions as object parameters, and so on. An equally important principle is that everything is a function; that is to say, in this particular case, there is no distinction between a function named add

34、 and a function named +. In Scala, all operators are functions on a class.</p><p>  In the Rational class, then, four operations have been defined for rational numbers. These are the canonical, mathematical

35、operations add, subtract, multiply, and divide. Each of these is named by its mathematical symbol: +, -, *, and /.</p><p>  Notice, however, that each of these operators works by constructing a new Rational

36、object each time. Again, this is very similar to how java.lang.String works, and it is the default implementation because it yields thread-safe code. (If no shared state -- and internal state of an object shared across t

37、hreads is implicitly shared state -- is modified by a thread, then there is no concern over concurrent access to that state.)</p><p>  What's new with you?</p><p>  The everything is a funct

38、ion rule has two powerful effects:</p><p>  The first, as you've already seen, is that functions can be manipulated and stored as objects themselves. This leads to powerful re-use scenarios like the one

39、explored in the first article in this series.</p><p>  The second effect is that there is no special distinction between the operators that the Scala-language designers might think to provide and the operato

40、rs that Scala programmers think should be provided. For example, let's assume for a moment that it makes sense to provide an "inversion" operator, which will flip the numerator and denominator and return a

41、new Rational (so that Rational(2,5) will return Rational(5,2)). If you decide that the ~ symbol best represents this concept, then you can def</p><p>  Defining this unary "operator" in Scala is sl

42、ightly tricky, but it's purely a syntactic nit:Listing 6. This is how you flip</p><p>  The tricky part is, of course, the fact that you have to prefix the ~ name with "unary_" to tell the Sca

43、la compiler that it is intended to be a unary operator; therefore, the syntax will be "flipped" from the traditional reference-then-method syntax common in most object languages.</p><p>  Note that

44、 this combines with the "everything is an object" rule to create some powerful -- but easy-to-explain -- code opportunities:</p><p>  Listing 7. Add it up</p><p>  Naturally, the Scala

45、 compiler "does the right thing" for the straight integer addition examples, but syntactically it's all the same. This means that you can develop types that are no different from the "built-in" ty

46、pes that come as part of the Scala language.</p><p>  The Scala compiler will even try to infer some meaning out of the "operators" that have some predetermined meaning, such as the += operator. No

47、te how the following code just does what it should, despite the fact that the Rational class doesn't have an explicit definition for +=:</p><p>  Listing 8. Scala infers</p><p>  When printe

48、d, r5 has the value [13 / 12], which is exactly what it should be.</p><p>  Scala under the hood</p><p>  Remember that Scala compiles to Java bytecode, meaning that it runs on the JVM. If you n

49、eed proof, look no further than the fact that the compiler is producing .class files that begin with 0xCAFEBABE, just like javac does. Also note what happens if you fire up the Java bytecode disassembler that comes with

50、the JDK (javap) and point it at the generated Rational class, as shown in Listing 9:Listing 9. Classes compiled from rational.scala</p><p>  The "operators" defined in the Scala class transmogrify

51、 into method calls in the best tradition of Java programming, though they do seem to be based on funny names. Two constructors are defined on the class: one taking an int and one taking a pair of ints. And, if you happen

52、 to be at all concerned that the use of the upper-case Int type is somehow a java.lang.Integer in disguise, note that the Scala compiler is smart enough to transform them into regular Java primitive ints in the class def

53、initi</p><p>  Testing, testing, 1-2-3...</p><p>  It is a well-known meme that good programmers write code, and great programmers write tests; thus far, I have been lax in exercising this rule

54、for my Scala code, so let’s see what happens when you put this Rational class inside of a traditional JUnit test suite, as shown in Listing 10:</p><p>  Listing 10. RationalTest.java</p><p>  As

55、ide from confirming that the Rational class behaves, well, rationally, the above test suite also proves that it is possible to call Scala code from Java code (albeit with a little bit of an impedance mismatch when it com

56、es to the operators). The cool thing about this, of course, is that it lets you try out Scala slowly, by migrating Java classes over to Scala classes without ever having to change the tests that back them.</p><

57、;p>  The only weirdness you might notice in the test code has to do with operator invocation, in this case, the + method on the Rational class. Looking back at the javap output, Scala has obviously translated the + fu

58、nction into the JVM method $plus, but the Java Language Specification does not allow the $ character in identifiers (which is why it's used in nested and anonymous nested class names).</p><p>  In order

59、to invoke those methods, you either have to write the tests in Groovy or JRuby (or some other language that doesn't pose a restriction on the $ character), or you can write a little bit of Reflection code to invoke i

60、t. I go with the latter approach, which isn't all that interesting from a Scala perspective, but the result is included in this article's code bundle, should you be curious. (SeeDownload.)</p><p>  N

61、ote that workarounds like these are only necessary for function names that aren't also legitimate Java identifiers.</p><p>  A "better" Java</p><p>  Back when I was first learning

62、 C++, Bjarne Stroustrup suggested that one way to learn C++ was to see it as "a better C" (see Resources). In some ways, Java developers today might come to see Scala as a "better Java," because it pr

63、ovides a more terse and succinct way of writing traditional Java POJOs. Consider the traditional Person POJO shown in Listing 11:</p><p>  Listing 11. JavaPerson.java (original POJO)</p><p>  No

64、w consider its equivalent written in Scala:</p><p>  Listing 12. person.scala (threadsafe POJO)</p><p>  It isn't a complete drop-in replacement, given that the original Person had some muta

65、ble setters. But considering the original Person also had no synchronization code around those mutable setters, the Scala version is safer to use. Also, if the goal is to truly reduce the number of lines of code in Perso

66、n, you could remove the getFoo property methods entirely because Scala will generate accessor methods around each of the constructor parameters -- firstName() returns a String, lastName() returns</p><p>  Ev

67、en if the need for those mutable setter methods is undeniable, the Scala version is still simpler, as you can see in Listing 13:</p><p>  Listing 13. person.scala (full POJO)</p><p>  As an asid

68、e, notice the introduction of the var keyword on the constructor parameters. Without going into too much detail, var tells the compiler that the value is mutable. As a result, Scala generates both accessor (String firstN

69、ame(void)) and mutator (void firstName_$eq(String)) methods. It then becomes easy to create setFoo property mutator methods that use the generated mutator methods under the hood.</p><p>  Conclusion</p>

70、;<p>  Scala is an attempt to incorporate functional concepts and terseness without losing the richness of the object paradigm. As you've perhaps begun to see in this series, Scala also corrects some of the eg

71、regious (in hindsight) syntactic problems found in the Java language.</p><p>  This second article in the Busy Java developer's guide to Scala series has focused on Scala's object facilities, which l

72、et you start using Scala without having to dive too deeply into the functional pool. Based on what you've learned so far, you can already start using Scala to reduce your programming workload. Among other things, you

73、 can use Scala to produce the very same POJOs needed for other programming environments, such as Spring or Hibernate.</p><p>  Hold on to your diving caps and scuba gear, however, because next month's ar

74、ticle will mark the beginning of our descent into the deep end of the functional pool.</p><p><b>  外文翻譯</b></p><p>  面向Java開(kāi)發(fā)人員的Scala指南:類(lèi)操作</p><p>  Java? 開(kāi)發(fā)人員可以將對(duì)象作為理解S

75、cala的出發(fā)點(diǎn)。本文是面向Java開(kāi)發(fā)人員的 Scala指南系列的第二期,作者Ted Neward遵循對(duì)一種語(yǔ)言進(jìn)行評(píng)價(jià)的基本前提:一種語(yǔ)言的威力可以直接通過(guò)它集成新功能的能力衡量,在本文中就是指對(duì)復(fù)數(shù)的支持。跟隨本文,您將了解在Scala中與類(lèi)的定義和使用有關(guān)的一些有趣特性。</p><p>  在上一期文章中,您只是稍微了解了一些 Scala 語(yǔ)法,這些是運(yùn)行 Scala 程序和了解其簡(jiǎn)單特性的最基本要求。

76、通過(guò)上一篇文章中的Hello World 和 Timer 示例程序,您了解了Scala的Application類(lèi)、方法定義和匿名函數(shù)的語(yǔ)法,還稍微了解Array[] 和一些類(lèi)型推斷方面的知識(shí)。Scala 還提供了很多其他特性,本文將研究 Scala 編程中的一些較復(fù)雜方面。</p><p>  Scala的函數(shù)編程特性非常引人注目,但這并非Java開(kāi)發(fā)人員應(yīng)該對(duì)這門(mén)語(yǔ)言感興趣的惟一原因。實(shí)際上,Scala 融合了

77、函數(shù)概念和面向?qū)ο蟾拍睢榱俗孞ava和Scala程序員感到得心應(yīng)手,可以了解一下Scala 的對(duì)象特性,看看它們是如何在語(yǔ)言方面與 Java對(duì)應(yīng)的。記住,其中的一些特性并不是直接對(duì)應(yīng),或者說(shuō),在某些情況下,“對(duì)應(yīng)” 更像是一種類(lèi)比,而不是直接的對(duì)應(yīng)。不過(guò),遇到重要區(qū)別時(shí),我會(huì)指出來(lái)。</p><p>  Scala和Java一樣使用類(lèi)</p><p>  我們不對(duì)Scala支持的類(lèi)特性作

78、冗長(zhǎng)而抽象的討論,而是著眼于一個(gè)類(lèi)的定義,這個(gè)類(lèi)可用于為Scala平臺(tái)引入對(duì)有理數(shù)的支持(主要借鑒自“Scala By Example”,參見(jiàn)參考資料):清單 1. rational.scala</p><p>  從詞匯上看,清單1的整體結(jié)構(gòu)與Java代碼類(lèi)似,但是,這里顯然還有一些新的元素。在詳細(xì)討論這個(gè)定義之前,先看一段使用這個(gè)新 Rational 類(lèi)的代碼:清單 2. RunRational</

79、p><p>  清單 2 中的內(nèi)容平淡無(wú)奇:先創(chuàng)建兩個(gè)有理數(shù),然后再創(chuàng)建兩個(gè)Rational,作為前面兩個(gè)有理數(shù)的和與差,最后將這幾個(gè)數(shù)回傳到控制臺(tái)上(注意,Console.println()來(lái)自Scala核心庫(kù),位于scala.* 中,它被隱式地導(dǎo)入每個(gè)Scala程序中,就像Java編程中的java.lang一樣)。</p><p>  用多少種方法構(gòu)造類(lèi)?</p><p

80、>  現(xiàn)在,回顧一下Rational類(lèi)定義中的第一行:清單 3.Scala的默認(rèn)構(gòu)造函數(shù)</p><p>  您也許會(huì)認(rèn)為清單 3 中使用了某種類(lèi)似于泛型的語(yǔ)法,這其實(shí)是Rational類(lèi)的默認(rèn)的、首選的構(gòu)造函數(shù):n和d是構(gòu)造函數(shù)的參數(shù)。</p><p>  Scala優(yōu)先使用單個(gè)構(gòu)造函數(shù),這具有一定的意義——大多數(shù)類(lèi)只有一個(gè)構(gòu)造函數(shù),或者通過(guò)一個(gè)構(gòu)造函數(shù)將一組構(gòu)造函數(shù)“鏈接”起來(lái)

81、。如果需要,可以在一個(gè)Rational上定義更多的構(gòu)造函數(shù),例如:清單 4. 構(gòu)造函數(shù)鏈</p><p>  注意,Scala的構(gòu)造函數(shù)鏈通過(guò)調(diào)用首選構(gòu)造函數(shù)(Int,Int版本)實(shí)現(xiàn)Java構(gòu)造函數(shù)鏈的功能。</p><p><b>  實(shí)現(xiàn)細(xì)節(jié)</b></p><p>  在處理有理數(shù)時(shí),采取一點(diǎn)數(shù)值技巧將會(huì)有所幫助:也就是說(shuō),找到公分母

82、,使某些操作變得更容易。如果要將1/2與2/4相加,那 Rational類(lèi)應(yīng)該足夠聰明,能夠認(rèn)識(shí)到2/4和1/2是相等的,并在將這兩個(gè)數(shù)相加之前進(jìn)行相應(yīng)的轉(zhuǎn)換。嵌套的私有g(shù)cd()函數(shù)和 Rational 類(lèi)中的g值可以實(shí)現(xiàn)這樣的功能。在Scala中調(diào)用構(gòu)造函數(shù)時(shí),將對(duì)整個(gè)類(lèi)進(jìn)行計(jì)算,這意味著將g初始化為n和d的最大公分母,然后用它依次設(shè)置n和d。</p><p>  回顧一下清單1就會(huì)發(fā)現(xiàn),我創(chuàng)建了一個(gè)覆蓋的t

83、oString方法來(lái)返回Rational的值,在RunRational驅(qū)動(dòng)程序代碼中使用toString時(shí),這樣做非常有用。</p><p>  然而,請(qǐng)注意toString的語(yǔ)法:定義前面的override關(guān)鍵字是必需的,這樣Scala才能確認(rèn)基類(lèi)中存在相應(yīng)的定義。這有助于預(yù)防因意外的輸入錯(cuò)誤導(dǎo)致難于覺(jué)察的 bug(Java 5中創(chuàng)建@Override注釋的動(dòng)機(jī)也在于此)。還應(yīng)注意,這里沒(méi)有指定返回類(lèi)型 ——

84、從方法體的定義很容易看出——返回值沒(méi)有用return關(guān)鍵字顯式地標(biāo)注,而在Java中則必須這樣做。相反,函數(shù)中的最后一個(gè)值將被隱式地當(dāng)作返回值(但是,如果您更喜歡Java語(yǔ)法,也可以使用return關(guān)鍵字)。</p><p><b>  一些重要值</b></p><p>  接下來(lái)分別是numer和denom的定義。這里涉及的語(yǔ)法可能讓Java程序員認(rèn)為numer和

85、denom是公共的Int字段,它們分別被初始化為n-over-g和d-over-g;但這種想法是不對(duì)的。</p><p>  在形式上,Scala調(diào)用無(wú)參數(shù)的numer和denom方法,這種方法用于創(chuàng)建快捷的語(yǔ)法以定義accessor。Rational類(lèi)仍然有3個(gè)私有字段:n、d和g,但是,其中的n和d被默認(rèn)定義為私有訪(fǎng)問(wèn),而g則被顯式地定義為私有訪(fǎng)問(wèn),它們對(duì)于外部都是隱藏的。</p><p&

86、gt;  此時(shí),Java 程序員可能會(huì)問(wèn):“n和d各自的‘setter’在哪里?”Scala中不存在這樣的 setter。Scala的一個(gè)強(qiáng)大之處就在于,它鼓勵(lì)開(kāi)發(fā)人員以默認(rèn)方式創(chuàng)建不可改變的對(duì)象。但是,也可使用語(yǔ)法創(chuàng)建修改Rational內(nèi)部結(jié)構(gòu)的方法,但是這樣做會(huì)破壞該類(lèi)固有的線(xiàn)程安全性。因此,至少對(duì)于這個(gè)例子而言,我將保持Rational不變。</p><p>  當(dāng)然還有一個(gè)問(wèn)題,如何操縱Rational

87、呢?與java.lang.String一樣,不能直接修改現(xiàn)有的Rational的值,所以惟一的辦法是根據(jù)現(xiàn)有類(lèi)的值創(chuàng)建一個(gè)新的Rational,或者從頭創(chuàng)建。這涉及到4個(gè)名稱(chēng)比較古怪的方法:+、 -、* 和 /。</p><p>  與其外表相反,這并非操作符重載。</p><p><b>  操作符</b></p><p>  記住,在Sca

88、la中一切都是對(duì)象。在上一篇文章中,您看到了函數(shù)本身也是對(duì)象這一原則的應(yīng)用,這使Scala程序員可以將函數(shù)賦予變量,將函數(shù)作為對(duì)象參數(shù)傳遞等等。另一個(gè)同樣重要的原則是,一切都是函數(shù);也就是說(shuō),在此處,命名為add的函數(shù)與命名為 + 的函數(shù)沒(méi)有區(qū)別。在Scala中,所有操作符都是類(lèi)的函數(shù)。只不過(guò)它們的名稱(chēng)比較古怪罷了。</p><p>  在Rational類(lèi)中,為有理數(shù)定義了4種操作。它們是規(guī)范的數(shù)學(xué)操作:加、減

89、、乘、除。每種操作以它的數(shù)學(xué)符號(hào)命名:+、-、 * 和 /。</p><p>  但是請(qǐng)注意,這些操作符每次操作時(shí)都構(gòu)造一個(gè)新的Rational對(duì)象。同樣,這與java.lang.String非常相似,這是默認(rèn)的實(shí)現(xiàn),因?yàn)檫@樣可以產(chǎn)生線(xiàn)程安全的代碼(如果線(xiàn)程沒(méi)有修改共享狀態(tài) —— 默認(rèn)情況下,跨線(xiàn)程共享的對(duì)象的內(nèi)部狀態(tài)也屬于共享狀態(tài) —— 則不會(huì)影響對(duì)那個(gè)狀態(tài)的并發(fā)訪(fǎng)問(wèn))。</p><p>

90、;<b>  有什么變化?</b></p><p>  一切都是函數(shù),這一規(guī)則產(chǎn)生兩個(gè)重要影響:</p><p>  首先,您已經(jīng)看到,函數(shù)可以作為對(duì)象進(jìn)行操縱和存儲(chǔ)。這使函數(shù)具有強(qiáng)大的可重用性,本系列第一篇文章對(duì)此作了探討。</p><p>  第二個(gè)影響是,Scala語(yǔ)言設(shè)計(jì)者提供的操作符與Scala程序員認(rèn)為應(yīng)該提供的操作符之間沒(méi)有特別的

91、差異。例如,假設(shè)提供一個(gè)“求倒數(shù)”操作符,這個(gè)操作符會(huì)將分子和分母調(diào)換,返回一個(gè)新的Rational(即對(duì)于Rational(2,5)將返回Rational(5,2))。如果您認(rèn)為~符號(hào)最適合表示這個(gè)概念,那么可以使用此符號(hào)作為名稱(chēng)定義一個(gè)新方法,該方法將和Java代碼中任何其他操作符一樣,如清單5所示:</p><p><b>  清單 5. 求倒數(shù)</b></p><

92、p>  在 Scala 中定義這種一元操作符”需要一點(diǎn)技巧,但這只是語(yǔ)法上的問(wèn)題而已:清單 6. 如何求倒數(shù)</p><p>  當(dāng)然,需要注意的地方是,必須在名稱(chēng)~之前加上前綴“unary_”,告訴Scala編譯器它屬于一元操作符。因此,該語(yǔ)法將顛覆大多數(shù)對(duì)象語(yǔ)言中常見(jiàn)的傳統(tǒng)reference- then-method語(yǔ)法。</p><p>  這條規(guī)則與 “一切都是對(duì)象” 規(guī)

93、則結(jié)合起來(lái),可以實(shí)現(xiàn)功能強(qiáng)大(但很簡(jiǎn)單)的代碼:清單 7. 求和</p><p>  當(dāng)然,對(duì)于簡(jiǎn)單的整數(shù)加法,Scala編譯器也會(huì)“得到正確的結(jié)果”,它們?cè)谡Z(yǔ)法上是完全一樣的。這意味著您可以開(kāi)發(fā)與Scala語(yǔ)言“內(nèi)置”的類(lèi)型完全相同的類(lèi)型。</p><p>  Scala編譯器甚至?xí)L試推斷具有某種預(yù)定含義的“操作符”的其他含義,例如+= 操作符。注意,雖然Rational類(lèi)并沒(méi)有顯式

94、地定義+=,下面的代碼仍然會(huì)正常運(yùn)行:清單 8. Scala 推斷</p><p>  打印結(jié)果時(shí),r5的值為[13 / 12],結(jié)果是正確的。</p><p><b>  Scala 內(nèi)幕</b></p><p>  記住,Scala將被編譯為Java字節(jié)碼,這意味著它在JVM上運(yùn)行。如果您需要證據(jù),那么只需注意編譯器生成以0xCAFEBA

95、BE開(kāi)頭的.class文件,就像javac一樣。另外請(qǐng)注意,如果啟動(dòng)JDK自帶的Java字節(jié)碼反編譯器(javap),并將它指向生成的Rational類(lèi),將會(huì)出現(xiàn)什么情況,如清單9所示:清單 9. 從 rational.scala 編譯的類(lèi)</p><p>  Scala類(lèi)中定義的“操作符”被轉(zhuǎn)換成傳統(tǒng)Java編程中的方法調(diào)用,不過(guò)它們?nèi)允褂每瓷先ビ行┕殴值拿Q(chēng)。類(lèi)中定義了兩個(gè)構(gòu)造函數(shù):一個(gè)構(gòu)造函數(shù)帶有一個(gè)in

96、t參數(shù),另一個(gè)帶有兩個(gè)int參數(shù)。您可能會(huì)注意到,大寫(xiě)的Int類(lèi)型與java.lang.Integer有點(diǎn)相似,Scala編譯器非常聰明,會(huì)在類(lèi)定義中將它們轉(zhuǎn)換成常規(guī)的Java原語(yǔ)int。</p><p>  測(cè)試 Rational 類(lèi)</p><p>  一種著名的觀(guān)點(diǎn)認(rèn)為,優(yōu)秀的程序員編寫(xiě)代碼,偉大的程序員編寫(xiě)測(cè)試;到目前為止,我還沒(méi)有對(duì)我的Scala代碼嚴(yán)格地實(shí)踐這一規(guī)則,那么現(xiàn)在看

97、看將這個(gè)Rational類(lèi)放入一個(gè)傳統(tǒng)的JUnit測(cè)試套件中會(huì)怎樣,如清單10所示:清單 10. RationalTest.java</p><p>  除了確認(rèn)Rational類(lèi)運(yùn)行正常之外,上面的測(cè)試套件還證明可以從Java代碼中調(diào)用 Scala代碼(盡管在操作符方面有點(diǎn)不匹配)。當(dāng)然,令人高興的是,您可以將Java類(lèi)遷移至Scala類(lèi),同時(shí)不必更改支持這些類(lèi)的測(cè)試,然后慢慢嘗試Scala。</p&g

98、t;<p>  您惟一可能覺(jué)得古怪的地方是操作符調(diào)用,在本例中就是Rational類(lèi)中的 + 方法。回顧一下javap的輸出,Scala顯然已經(jīng)將+函數(shù)轉(zhuǎn)換為JVM方法$plus,但是Java語(yǔ)言規(guī)范并不允許標(biāo)識(shí)符中出現(xiàn)$字符(這正是它被用于嵌套和匿名嵌套類(lèi)名稱(chēng)中的原因)。</p><p>  為了調(diào)用那些方法,需要用Groovy或JRuby(或者其他對(duì)$字符沒(méi)有限制的語(yǔ)言)編寫(xiě)測(cè)試,或者編寫(xiě)Ref

99、lection代碼來(lái)調(diào)用它。我采用后一種方法,從Scala的角度看這不是那么有趣,但是如果您有興趣的話(huà),可以看看本文的代碼中包含的結(jié)果(參見(jiàn)下載)。</p><p>  注意,只有當(dāng)函數(shù)名稱(chēng)不是合法的Java標(biāo)識(shí)符時(shí)才需要用這類(lèi)方法。</p><p>  “更好的” Java</p><p>  我學(xué)習(xí)C++的時(shí)候Bjarne Stroustrup建議,學(xué)習(xí)C++的

100、一種方法是將它看作“更好的C語(yǔ)言”(參見(jiàn) 參考資料)。在某些方面,如今的Java開(kāi)發(fā)人員也可以將Scala看作是“更好的Java”,因?yàn)樗峁┝艘环N編寫(xiě)傳統(tǒng)Java POJO的更簡(jiǎn)潔的方式??紤]清單11中顯示的傳統(tǒng) Person POJO:清單 11. JavaPerson.java(原始 POJO)</p><p>  現(xiàn)在考慮用 Scala 編寫(xiě)的對(duì)等物:清單 12. person.scala(線(xiàn)程安全的

101、 POJO)</p><p>  這不是一個(gè)完全匹配的替換,因?yàn)樵嫉腜erson包含一些可變的setter。但是,由于原始的Person沒(méi)有與這些可變setter相關(guān)的同步代碼,所以Scala版本使用起來(lái)更安全。而且,如果目標(biāo)是減少Person中的代碼行數(shù),那么可以刪除整個(gè)getFoo屬性方法,因?yàn)镾cala將為每個(gè)構(gòu)造函數(shù)參數(shù)生成accessor方法—— firstName()返回一個(gè)String,lastN

102、ame()返回一個(gè)String,age()返回一個(gè)int。</p><p>  即使必須包含這些可變的setter方法,Scala版本仍然更加簡(jiǎn)單,如清單13所示:清單 13. person.scala(完整的 POJO)</p><p>  注意,構(gòu)造函數(shù)參數(shù)引入了var關(guān)鍵字。簡(jiǎn)單來(lái)說(shuō),var告訴編譯器這個(gè)值是可變的。因此,Scala 同時(shí)生成 accessor(String fir

溫馨提示

  • 1. 本站所有資源如無(wú)特殊說(shuō)明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶(hù)所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁(yè)內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒(méi)有圖紙預(yù)覽就沒(méi)有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 眾賞文庫(kù)僅提供信息存儲(chǔ)空間,僅對(duì)用戶(hù)上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶(hù)上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶(hù)因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。

最新文檔

評(píng)論

0/150

提交評(píng)論