Scala入門指南與建議

最近在學習使用Scala語言作項目,感受這門語言實在是太優美了!做爲一個本科數學、研究生機器學習專業的混合人才(哈哈),這門語言真的是知足了普通計算機編程(告訴計算機怎麼作)和函數式編程(告訴計算機作什麼)的全部幻想。學了半個多月,根本停不下來!先來作個總結:java

  1. 語法簡潔,每每比Java少至少一半的代碼量。好比:
    • 支持自動類型判斷,能夠省去不少類型標誌。 e.g.  val x = 2
    • 用伴生對象來生成類,省去new的麻煩。e.g. val cat = Cat("Hello Ketty")
    • 不用return,直接把一個塊(可使if...else...塊,for循環塊等)的值返回。例如一行代碼定義函數:def add(x: Int, y: Int): Int = x + y
    • 用()來統一函數的參數傳遞與帶參類的構造。對類來講,這種寫法實際上是語法糖,由於中間有自動的轉換機制,使得簡潔的代碼和底層實現能夠分離。
    • 程序易讀。對比C/C++、Python、Java,Scala是最符合人類理解的程序語言。
  2. 有幾乎徹底的函數式風格支持。
    • 函數和值同樣,是第一等公民。函數也是值,值也能夠做爲函數。
    • 支持高階函數、Curry化、lambda運算等函數運算概念。
    • 函數式風格要求函數儘可能無反作用,這樣一方面適合作單元測試來驗證程序的正確性,另外很適合作並行計算!
  3. 能夠知足大多數OOP編程需求。
    • 這裏就不展開了。
  4. Scala兼具科學計算語言(matlab、R、Python等)的易讀性與靜態語言(C/C++、Java等)的高效率。

正是因爲以上優勢,我以爲Scala頗有潛力成爲下一門普及的編程語言。node

另外,關於Scala學習曲線,個人建議是:react

  1. 從網上各類博客大概瞭解Scala的特性,包括上面提到的幾條。
  2. 看書:
    • 《Programming in Scala》和《Scala for the Impatient》交替看。前一本是Scala語言建立者寫的,很是通俗易懂,包含了Scala的大多數「是什麼」和「爲何」;後一本則是總結性的,把Scala的大多少「是什麼」告訴你。
    • 《Scala in Depth》適合中等水平的Scala程序員閱讀。(計劃下一步看完)
  3. 作項目 / 開發程序。

最後,附上《Scala for the Impatient》(中文版《快學Scala》)的全部key notes 以及我作的部分課後題參考答案地址。若是有須要相關電子書的,能夠給我發email。git

 


 

《Scala for the Impatient》key notes程序員

Chapter 1. The Basics
  • Using the Scala interpreter
  • Defining variables with val and var
  • Numeric types
  • Using operators and functions
  • Navigating Scaladoc
 
Chapter 2. Control Structures and Functions
  • An if expression has a value
  • A block has a value — the value of its last expression
  • The Scala for loop is like an "enhanced" Java for loop
  • Semicolons are (mostly) optional
  • The Void type is Unit
  • Avoid using return in a function
  • Beware of missing = in a function definition
  • Exceptions work just like in Java or C++, but you use a "pattern matching" syntax for catch 
  • Scala has no checked exceptions
 
Chapter 3. Working with Arrays
  • Use an Array if the length is fixed, and an ArrayBuffer if the length can vary
  • Don’t use new when supplying initial values
  • Use () to access elements
  • Use for (elem <- arr) to traverse the elements
  • Use for (elem <- arr if …) … yield … to transform into a new array
  • Scala and Java arrays are interoperable; with ArrayBuffer, use scala.collection.JavaConversions
 
Chapter 4. Maps and Tuples
  • Scala has a pleasant syntax for creating, querying, and traversing maps
  • You need to choose between mutable and immutable maps
  • By default, you get a hash map, but you can also get a tree map
  • You can easily convert between Scala and Java maps
  • Tuples are useful for aggregating values
 
Chapter 5. Classes
  • Fields in classes automatically come with getters and setters
  • You can replace a field with a custom getter / setter without changing the client of a class — that is the "uniform access principle"
  • Use the @BeanProperty annotation to generate the JavaBeans getXxx / setXxx methods
  • Every class has a primary constructor that is "interwoven" with the class definition. Its parameters turn into the fields of the class. The primary constructor excuses all statements in the body of the class
  • Auxiliary constructors are optional. That are called this.
 
Chapter 6. Objects
  • Use Objects for singletons and utility methods
  • A class can have a companion object with the same name.
  • Objects can extend classes or traits
  • The apply method of an object is usually used for constructing new instance of the companion class
  • To avoid the main method, use an object that extends that App trait
  • You can implement enumerations by extending the Enumeration object
 
Chapter 7. Packages and imports
  • Packages nest just like inner classes
  • Package paths are not absolute
  • A chain x.y.z in a package clause leaves the intermediate packages x and x.y invisible
  • Package statements without braces at the top of the file extend to the entire file
  • A package object can hold functions and variables
  • Import statements can import packages, classes, and objects
  • Import statements can be anywhere
  • Import statements can rename and hide menbers
  • java.lang, scala, and Predef are always imported
 
Chapter 8. Inheritance
  • The extends and final keywords are as in Java.
  • You must use override when you override a method
  • Only the primary constructors can all the primary superclass constructor
  • You can override fields
 
Chapter 9. Files and Regular Expressions
  • Source.fromFile(…).getLines.toArray yields all lines of a file
  • Source.fromFile(…).mkString yields the file contents as a string
  • To convert a string into a number, use the toInt or toDouble method
  • Use the Java PrintWriter to write text files
  • "regex".r is a Regex object
  • Use """…""" if your regular expression contains backslashes or quotes
  • If a regex pattern has groups, you can extract their contents using the syntax for (regex(var_1,…,var_n) <- string )
 
Chapter 10. Traits
  • A class can implement any number of traits
  • Traits can require that implementing classes have certain fields, methods, or superclasses
  • Unlike Java Interface, a Scala trait can provide implementations of methods and fields
  • When you layer multiple traits, the order matters — the trait whose methods execute first goes to the back
 
Chapter 11. Operators
  • Identifiers contain either alphanumeric or operator characters
  • Unary and binary operators are method calls
  • Operator precedence depends on the first character, associativity on the last
  • The apply and update methods are called when evaluating expr(args)
  • Extractors extract tuples or sequences of values from an input
 
Chapter 12. Higher-Order Functions
  • Functions are "first-class citizens" in Scala, just like numbers
  • You can create anonymous functions, usually to give them to other functions
  • A function argument specifies behavior that should be executed later
  • Many collection methods take function parameters, applying a function to the values of the collections
  • There are syntax shortcuts that allow you to express function parameters in a way that is short and easy to read
  • You can create functions that operate on the blocks of code and look much like the build-in control statement
 
Chapter 13. Collections
  • All collections extend the Iterable trait
  • The three major categories of collections are sequences, sets, and maps.
  • Scala has mutable and immutable versions of most collections
  • A Scala list is either empty, or it has a head and a tail which is again a list
  • Sets are unordered collections
  • Use a LinkedHashSet to retain the insertion order or a SortedSet to iterate in sorted order
  • + adds an elements to an unordered collection; +: and :+ prepend or append to a sequence; ++ concatenates two collections; - and -- remove elements
  • The Iterable and Seq traits have dozens of useful methods for common operations. Check them out before writing tedious loops.
  • Mapping, folding, and zipping are useful techniques for applying a function or operation to the elements of a collection
 
Chapter 14. Pattern Matching and Case Classes
  • The match expression is a better switch, without fall-through
  • If no pattern matches, a MatchError is thrown. Use the case _ pattern to avoid that
  • A pattern can include an arbitrary condition, called a guard
  • You can match on the type of an expression; prefer this over isInstanceOf / asInstanceOf
  • You can match patterns of arrays, tuples, and case classes, and bind parts of the pattern to variables
  • In a for expression, non matches are silently skipped
  • A case class is a class for which the compiler automatically produce the methods that are needed for pattern matching
  • The common superclass in a case class hierarchy should be sealed
  • Use the Option type for values that may or may not be present — it is safer than using null
 
Chapter 15. Annotations
  • You can annotate classes, methods, fields, local variables, parameters, expressions, type parameters and types.
  • With expressions and types, the annotation follows the annotated item
  • Annotation have the form @Annotation, @Annotation(value), or @Annotation(name1= value1, …)
  • @volatile, @transient, @strictfp, and @native generate the equivalent Java modifiers
  • Use @throws to generate Java-compatible throws specifications
  • The @tailrec annotation lets you verify that a recursive function uses tail call optimization
  • The assert function takes advantage of the @elidable annotation. You can optionally remove assertions from your Scala program
  • Use the @deprecated annotation to mark deprecated features.
 
Chapter 16. XML Processiong
  • XML literals <like> this </like> are of type NodeSeq
  • You can embed Scala code inside XML literals
  • The child property of a Node yield the child nodes
  • The attributes property of a Node yields a MetaData object containing the node attributes.
  • The \ and \\ operators carry out XPath-Like matches
  • You can match mode patterns with XML literals in case clauses
  • Use the RuleTransformer with RewriteRule instance to transform descendants of a node
  • The XML object interfaces with Java XML methods for loading and saving
  • The ConstructingParser is an alternate parser that preserves comments and CDATA sections
 
Chapter 17. Type parameters
  • Classes, traits, methods, and functions can have type parameters
  • Place the type parameters after the name, enclosed in square brackets
  • Type bounds have the form T <: UpperBound, T >: LowerBound, T <% ViewBound, T : ContextBound
  • You can restrict a method with a type constraint such as (implicit ev: T <: < UpperBound)
  • Use +T (covariance) to indicate that a generic type’s subtype relationship is in the same direction as the parameter T, or -T (contravariance) to indicate the reverse direction
  • Covariance is appropriate for parameters that denote outputs, such as elements in an immutable collection.
  • Contravariance is appropriate for parameters that denote inputs, such as function arguments.
 
Chapter 18. Advanced types
  • Singleton types are useful for method chaining and methods with object parameters
  • A type projection includes inner class instances for all objects of an outer class.
  • A type alias gives a short name for a type.
  • Structural types are equivalent to "duck typing"
  • Existential types provide the formalism for wildcard parameters of generic types.
  • Use a self type declaration to indicate that a trait requires another type.
  • The "cake pattern" uses self types to implement a dependency injection mechanism
  • An abstract type must be made concrete in a subclass
  • A higher-kinded type has a type parameter that is itself a parameterized type.
 
Chapter 19. Parsing
  • Alternatives, concatenation, options, and repetitions in a grammar turn into |, ~, opt, and rep in Scala combinator parsers.
  • With RegexParsers, literal strings and regular expressions match tokens.
  • Use ^^ to process parse results.
  • Use pattern matching in a function supplied to ^^ to take apart ~ result.
  • Use ~> and <~ to discard tokens that are no longer needed after matching.
  • The respell combinator handles the common case of repeated items with a seperator
  • A token-based parser is useful for paring languages with reserved words and operations. Be prepared to define your own lexer.
  • Parsers are functions that consume a reader and yield a parse result: success, failure, or error.
  • The Failure result provides the details for error reporting.
  • You may want to add failure clauses to your grammar to improve the quality of error message.
  • Thanks to operator symbols, implicit conversions, and pattern matching, the parser combinator library makes parser writing easy for anyone who understands context-free grammars. Even if you don’t feel the urge to write your own parsers, you may find this an interesting case study for an effective domain-specific language.
 
Chapter 20. Actors
  • Extend the Actor class and provide an act method for each other.
  • To send a message to an actor, use actor ! message.
  • Message sending is asynchronous: "send and forget."
  • To receive messages, an actor calls receive or react, usually in a loop.
  • The argument to receive / react is a block of case clauses (technically, a partial function)
  • Actors should never share state. Always send data using messages.
  • Don’t invoke methods on actors. Communicating by sending messages.
  • Avoid synchronous messaging — that is, unlink sending a message and waiting for a reply.
  • Actors can share threads by using react instead of receive, provided the control flow of the message handler is simple.
  • It is OK to let actors crash, provided you have other actors that monitor their demise. Use linking to set up monitoring relationships.
 
Chapter 21. Implicits
  • Implicit conversions are used to convert between types.
  • You must import implicit conversions so that they are in scope as single identifiers.
  • An implicit parameter list requests objects of a given type. They can be obtained from implicit objects that are defined as single identifiers in scope, or from the companion object of the desired type.
  • If an implicit parameter is a single-argument function, it is also used as an implicit conversion.
  • A context bound of a type parameter requires the existence of an implicit object of the given type.
  • If it is possible to locate an implicit object, this can serve as evidence that a type conversion is valid.
 
Chapter 22. Delimited Continuations
  • A continuation lets you go back to a previous point in a program.
  • You can capture a continuation in a shift block.
  • A contination function extends until the end of the enclosing reset block.
  • A continuation is the "reset of the computation" from the expression containing the shift to the end of the enclosing reset, with the shift replaced by a "hole"
  • When you call a continuation with an argument, the "hole" is set to the argument.
  • Code containing shift expressions is rewritten in "continuation-passing stype", or CPS, up to the enclosing reset.
  • A method containing a shift without a reset must be annotated with a CPS annotation.
  • Continuation can be used to turn a recursive visit of a tree structure into an iteration.
  • Continuations can undo the "inversion of control" in a web or GUI application.

 


部分《快學Scala》課後題答案(還在更新中):github

https://github.com/fengfu-chris/Scala-for-the-Impatient-Exercisesweb

相關文章
相關標籤/搜索