Beginning Scala study note(9) Scala and Java Interoperability

1. Translating Java Classes to Scala Classes

  Example 1:

# a class declaration in Java
public class Book{}

# Scala equivalent of a class declaration
class Book

  Example 2:

# a Java class with a Construtor
public class Book{
    private final int isbn;
    private final String title;

    public Book(int isbn, String title){
        this.isbn = isbn;
        this.title = title;
    }   

    public int getIsbn(){
        return isbn;
    }

    public String getTitle(){
        return title;
    }
}

# Scala equivalent
class Book(val isbn: Int, val title: String)

  Example 3:

# constructor calling superclass in Java
public class NonFiction extends Book{
    public NOnFiction(String title){
        super(title);
    }
}

# Scala equivalent
class NonFiction(title: String) extends Book(title)

  Example 4:

# mutable instance variable in Java
public class Book{
    private String title = "Beginning Scala";

    public String getTitle(){
        return title;
    }

    public void setTitle(String t){
        title = t;
    }
}

# Scala equialent
class Book{
    var title = "Beginning Scala"
}

  Example 5:

# immutable instance variable in Java
public class Book{
    private final int isbn = 999;

    public int getIsbn(){
        return isbn;
    }
}

# Scala equivalent
class Book{
    val isbn = 999
}

  Translating Java imports to Scala imports:

# import in Java
import com.modA.ClassA;
import com.modB.ClassB1;
import com.modB.ClassB2;
import com.modC.*;

# import in Scala
import com.modA.ClassA
import com.modB.{ClassB1,ClassB2} // you can stack multiple imports from the same package in braces
import com.modC._ // underscore in Scala imports is equivalent of * in Java imports

  Example 6:

# Java class with multiple constructors
public class Book {
private Integer isbn;
private String title;
public Book(Integer isbn) {
this.isbn = isbn;
}
public Book(Integer isbn, String title) {
this.isbn = isbn;
this.title = title;
}
public Integer getIsbn() {
return isbn;
}
public void setIsbn(Integer isbn) {
this.isbn = isbn;
}
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}}

# refactoring
class Book(var isbn: Int, var title: String)

  If you create the Book instance with a construtor that takes a single "title" parameter, you will get an error.

scala> val book = new Book("test")
<console>:12: error: not enough arguments for constructor Book: (isbn: Int, title: String)Book.
Unspecified value parameter title.
       val book = new Book("test")

  We need an extra constructor for this case.

scala> class Book(var isbn: Int, var title: String){
     | def this(title: String) = this(0,title)
     | }
defined class Book
scala> val book = new Book("test")
book: Book = [email protected]
scala> book.isbn
res2: Int = 0
scala> book.title
res3: String = test

  You can get and set "isbn" and "title" because of the generated getters and setters that follow the Scala conversion.

2. JavaBeans specification compliant Scala classes

  To have Java-style getters and setters is to annotate the field with scala.beans.BeanProperty. In this way, you can interact with a Java calss or library that accepts only classes that conform to the JavaBean specification.

scala> import scala.beans.BeanProperty
import scala.beans.BeanProperty

scala> class Book(@BeanProperty var isbn: Int, @BeanProperty var title: String)
defined class Book

  After compiling Book.scala with scalac command and disassembling it with javap command:

public class Book {
  public int isbn();
  public void isbn_$eq(int);
  public void setIsbn(int);
  public java.lang.String title();
  public void title_$eq(java.lang.String);
  public void setTitle(java.lang.String);
  public int getIsbn();
  public java.lang.String getTitle();
  public Book(int, java.lang.String);
}

  The methods getTitle,setTitle,getIsbn,setIsbn have all been generated because of the @BeanProperty annotation. Note that use the @BeanProperty annotation on your fields, also making sure you declare each field as a var. If you declare your fields a type val, the setter methods won‘t be generated.

  You can use @BeanProperty annotation on class constructor parameters, even on the fields in a Scala class.

3. Java interfaces and Scala traits

  A java class can‘t extend a Scala trait that has implemented methods.

# A regular Java interface declaration

public interface Book{
    public abstract boolean isBestSeller();
}

# Scala equivalenttrait Book{ def isBestSeller: Boolean}

  Note that in scala, if there is no  = assignment, then the methods denoted with a def keyword or the functions denoted with a val keyword are abstract. That means if there‘s no definition provided with =, then it‘s automatically abstract.

# a concrete Java method
public String someMethod(int arg1, boolean arg2){return "voila"}

# Scala equivalent
def someMethod(arg1: Int, arg2: Boolean) :String = "volia"

# an abstract Java method
abstract int doTheMath(int i)

# Scala equivalent
def doTheMath(i: Int): Int

  Example: you need to be able to user an add method from a Java application:

# a Scala trait
trait Computation{def add(a: Int, b:Int) = a + b}

# a Java application
public class DoTheMath{
    public static void main(String[] args){
        DoTheMath d = new DoTheMath();
    }
}

  Java class DoTheMath cannot implement the trait Computation because Computation is not like a regular Java interface. To be able to use the implemented method add of a Scala trait Computation from Java class DoTheMath, you must wrap the trait Computation in a Scala class.

# Scala class that wraps the trait Computation
class JavaInteroperableComputation extends Computation

# accessing the add method of the Scala trait from Java class
public class DoTheMath extends JavaInteroperableComputation{
    public static void main(String[] args){
        DoTheMath d = new DoTheMath();
       d.add(3,1);
    }
}

  Note that wrap you scala traits with implemented behavior in the Scala class for its Java callers.

4. Java static members and Scala objects

# a singleton in Scala
public class Book{
    private static Book book;
    private Book(){}
    public static synchronized Book getInstance(){
        if(book == null){
            book = new Book();
        }
        return book;
    }
}

# Scala equivalent, object can extend interfaces and traits
object Book{}

  The companion object enables storing of static methods and from this, you have full access to the class‘s members, including private ones. Scala alows you to declare both an object and a class of the same name, placing the static members in the object and the instance members in the class.

# Java class with instance and static methods
public class Book{
    public String getCategory(){
        return "Non-Fiction";
    }

   public static Book createBook(){
        return new Book();
   }
}

# Scala equivalent
class Book{
    def getCategory() = "Non-Fiction"
}

object Book{
    def createBook() = new Book()
}

5.  Handling Exceptions

# a Scala method that throws an Exceotion
class SomeClass{
    def aScalaMethod{ throw new Exception("Exception")}
}

# calling a Scala method from a Java class
publc static void main(String[] args){
    SomeClass s = new SomeClass();
    s.aScalaMethod();
}

# the uncaught exception causes the Java method to fail
[error] (run-main) java.lang.Exception: Exception!
java.lang.Exception: Exception!
at SomeClass.aScalaMethod

  For the Java callers of you Scala methods, add the @throws annotation to your Scala methods so they will know which methods can throw exception and what exception they throw.

# annotating Scala method with @throws
class SomeClass{
    @throws(classOf[Excepion])
    def aScalaMethod{ throw new Exception("Exception")}
}

  If you attempt to call aScalaMethod from a Java class without wrapping it in a try/catch block, or declaring that your Java method throws an exception, the compiler will throw an eeor.

#calling annotated aScalaMethod from Java
SomeClass s = new SomeClass();
try{
    s.aScalaMethod();
}catch(Exception e){
    System.err.println("Caught the exception");
    e.printStackTrace();
}
时间: 2024-08-06 03:33:53

Beginning Scala study note(9) Scala and Java Interoperability的相关文章

Beginning Scala study note(8) Scala Type System

1. Unified Type System Scala has a unified type system, enclosed by the type Any at the top of the hierarchy and the type Nothing at the bottom of the hierarchy. All Scala types inherit from Any. # Using Any, Book extends AnyRef, and x is an Int that

Beginning Scala study note(6) Scala Collections

Scala's object-oriented collections support mutable and immutable type hierarchies. Also support functional higher-order operations such as map, filter, and reduce that let you use expression-oriented programming in collections. Higher-order operatio

Beginning Scala study note(2) Basics of Scala

1. Variables (1) Three ways to define variables: 1) val refers to define an immutable variable; scala> val x = 10 x: Int = 10 scala> x*x res4: Int = 100 scala> res4 + 1 # use result as a value res6: Int = 101 scala> res4 + res6 res7: Int = 201

Beginning Scala study note(3) Object Orientation in Scala

1. The three principles of OOP are encapsulation(封装性), inheritance(继承性) and polymorphism(多态性). example: class Shape{ def area: Double = 0.0 } # supertype # subtypes class Rectangle(val width: Double, val height: Double) extends Shape{ override def ar

Beginning Scala study note(5) Pattern Matching

The basic functional cornerstones of Scala: immutable data types, passing of functions as parameters and pattern matching. 1. Basic Pattern Matching In Scala, your cases can include types, wildcards, sequences, regular expressions, and so forth. scal

Beginning Scala study note(4) Functional Programming in Scala

1. Functional programming treats computation as the evaluation of mathematical and avoids state and mutable data. Scala encourages an expression-oriented programming(EOP) 1) In expression-oriented programming every statement is an expression. A state

Beginning Scala study note(7) Trait

A trait provides code reusability in Scala by encapsulating method and state and then offing possibility of mixing them into classes thus allowing code reuse. #define Trait Gliding scala> trait Gliding{ | def gliding(){ println("gliding")} |

Programming In Scala Reading Note 7

函数和闭包 1 成员方法 java中函数存在的方式,我们给一个类追加一个功能的途径就是给他追加一个方法. 2 本地方法 所谓的本地方法就是存在于一个方法内部的方法. 如果一个类中有一个方法是private的,且只有一个方法使用到他,那么这个方法完全可以被定义为一个本地方法 3 函数式一等公民 方法分为: 1 方法名称 def add 2 方法参数 (a:Int, b:Int) 3 赋值符号 = 4 方法体 a + b 其实,(a:Int, b:Int) => a + b,就构成了一个基本的函数了

Programming In Scala Reading Note 6

Built-in Control Structures 内置控制结构 1 if 控制结构 scala的if跟java的if结构上相同,功能上也差不太多,scala的说法是,if能够返回一个值 val fileName = if (!args.isEmpty) args(0) else "default.txt" 这个语句相对普通,但是这里没有引入一个var,而是使用的val. println(fileName) 等价于 println(if (!args.isEmpty) args(0