热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Scala核心概念:隐式转换详解

本文深入探讨了Scala中的隐式转换机制,包括其在类扩展、隐式解析规则以及隐式参数和上下文绑定等方面的应用。通过具体示例,详细解释了如何利用隐式转换增强类的功能。

类扩展与隐式转换

在Scala中,可以通过隐式转换来扩展类的功能,例如对String类进行扩展以提供更多的操作方法。例如,下面的代码展示了如何检查一个字符串是否包含数字:

"+9519760513".exists(_.isDigit)

尽管java.lang.String本身没有exists方法,但Scala的标准库通过在Predef对象中定义了一个隐式转换,将String隐式转换为StringOps,从而提供了这一方法。

object Predef {   implicit def augmentString(x: String): StringOps = new StringOps(x) }

隐式解析规则

标记规则

只有被标记为implicit的定义才可用作隐式转换。

例如,intWrapper方法将Int类型包装成RichInt类型:

object Predef {   implicit def intWrapper(x: Int) = new scala.runtime.RichInt(x) }

另一个例子是在Predef中定义的any2stringadd类,它允许任何类型的对象与字符串进行拼接:

object Predef {   implicit final class any2stringadd[A](private val self: A) extends AnyVal {     def +(other: String): String = String.valueOf(self) + other   } }

作用域规则

插入的隐式转换必须在作用域内作为一个单一标识符存在,或者与转换的源类型或目标类型相关联。

例如,定义了两个案例类YardMile

case class Yard(val amount: Int) case class Mile(val amount: Int)

可以在object Mile中定义一个隐式转换mile2yard,将Mile转换为Yard

object Mile {   implicit def mile2yard(mile: Mile) = new Yard(10 * mile.amount) }

同样,这个转换也可以在object Yard中定义:

object Yard {   implicit def mile2yard(mile: Mile) = new Yard(10 * mile.amount) }

在需要将Mile转换为Yard时,通常会遇到以下两种情况:

  • 传递参数时,但类型不匹配;

  • 赋值表达式中,但类型不匹配。

例如:

def accept(yard: Yard) = println(yard.amount + " yards") accept(Mile(10)) val yard: Yard = Mile(10)

其他规则

  • 一次一条规则:每次只尝试一个隐式转换。

  • 显式优先规则:如果代码本身类型检查通过,则不会尝试隐式转换。

  • 无歧义规则:只有当没有其他可能的转换时,才会插入隐式转换。

隐式转换的尝试位置

  • 转换为预期类型

    • 传递参数时,但类型不匹配;

    • 赋值表达式中,但类型不匹配。

  • 接收者的选择转换

    • 调用方法,但方法不存在;

    • 调用方法,方法存在但参数类型不匹配。

  • 隐式参数

隐式参数

隐式参数允许方法在调用时自动提供某些参数,这些参数通常由编译器根据上下文自动推断。例如:

import scala.math.Ordering case class Pair[T](first: T, second: T) {   def smaller(implicit order: Ordering[T]) = order.min(first, second) }

TInt时:

Pair(1, 2).smaller

实际上编译器调用的是:

Pair(1, 2).smaller(Ordering.Int)

其中Ordering.Int定义在Ordering的伴生对象中:

object Ordering {   trait IntOrdering extends Ordering[Int] {     def compare(x: Int, y: Int) =       if (x 

因此:

implicitly[Ordering[Int]] == Ordering.Int  // true

其中,implicitly是定义在Predef中的一个工具函数,用于提取隐式值:

@inline def implicitly[T](implicit e: T) = e

对于自定义类型,也可以类似地定义隐式参数:

import scala.math.Ordering case class Point(x: Int, y: Int) object Point {   implicit object OrderingPoint extends Ordering[Point] {     def compare(lhs: Point, rhs: Point): Int = (lhs.x + lhs.y) - (rhs.x + rhs.y)   } } Pair(Point(0, 0), Point(1, 1)).smaller

等价于:

Pair(Point(0, 0), Point(1, 1)).smaller(Point.OrderingPoint)

因此:

implicitly[Ordering[Point]] == Point.OrderingPoint

上下文绑定

上下文绑定(Context Bound)是一种简化的语法,用于声明一个类型参数必须有一个特定类型的隐式值。例如:

import scala.math.Ordering case class Pair[T : Ordering](first: T, second: T) {   def smaller(implicit order: Ordering[T]) = order.min(first, second) }

可以进一步简化为:

import scala.math.Ordering case class Pair[T : Ordering](first: T, second: T) {   def smaller = implicitly[Ordering[T]].min(first, second) }

甚至可以更简洁地表示为:

import scala.math.Ordering case class Pair[T : Ordering](first: T, second: T) {   def smaller = Ordering[T].min(first, second) }

这里,Ordering[T]实际上是调用了object Orderingapply方法,从而便捷地找到了Ordering[T]的隐式值:

object Ordering {   def apply[T](implicit ord: Ordering[T]) = ord }

因此,Ordering[T].min等价于implicitly[Ordering[T]].min

视图绑定

视图绑定(View Bound)允许一个类型参数可以隐式转换为另一个类型。例如:

import scala.math.Ordered case class Pair[T](first: T, second: T) {   def smaller(implicit order: T => Ordered[T]) = {     if (order(first) 

在这里,order既是隐式参数,也是一个隐式转换函数。例如:

Pair(1, 2).smaller

等价于:

Pair(1, 2).smaller(Predef.intWrapper _)

这种设计使得隐式参数order变得多余,形成了一个常见的模式——视图绑定:

import scala.math.Ordered case class Pair[T <% Ordered[T]](first: T, second: T) {   def smaller = if (first 

需要注意的是,T <% Ordered[T]表示T可以隐式转换为Ordered[T],而T <: Ordered[T]表示TOrdered[T]的子类型。

上限约束

上限约束(Upper Bound)用于限制类型参数必须是某个特定类型的子类型。例如:

import scala.math.Ordered case class Pair[T <: Comparable[T]](first: T, second: T) {   def smaller = if (first.compareTo(second) <0) first else second }

在这个例子中:

Pair("1", "2").smaller  // OK, String is subtype of Comparable[String] Pair(1, 2).smaller      // Compile Error, Int is not subtype of Comparable[Int]

推荐阅读
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 本文深入探讨了 Java 中的 Serializable 接口,解释了其实现机制、用途及注意事项,帮助开发者更好地理解和使用序列化功能。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • XNA 3.0 游戏编程:从 XML 文件加载数据
    本文介绍如何在 XNA 3.0 游戏项目中从 XML 文件加载数据。我们将探讨如何将 XML 数据序列化为二进制文件,并通过内容管道加载到游戏中。此外,还会涉及自定义类型读取器和写入器的实现。 ... [详细]
  • 本文详细介绍了Java中org.neo4j.helpers.collection.Iterators.single()方法的功能、使用场景及代码示例,帮助开发者更好地理解和应用该方法。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文详细介绍了 GWT 中 PopupPanel 类的 onKeyDownPreview 方法,提供了多个代码示例及应用场景,帮助开发者更好地理解和使用该方法。 ... [详细]
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • 本文详细介绍了Java中org.eclipse.ui.forms.widgets.ExpandableComposite类的addExpansionListener()方法,并提供了多个实际代码示例,帮助开发者更好地理解和使用该方法。这些示例来源于多个知名开源项目,具有很高的参考价值。 ... [详细]
  • 本文详细介绍了Akka中的BackoffSupervisor机制,探讨其在处理持久化失败和Actor重启时的应用。通过具体示例,展示了如何配置和使用BackoffSupervisor以实现更细粒度的异常处理。 ... [详细]
  • DNN Community 和 Professional 版本的主要差异
    本文详细解析了 DotNetNuke (DNN) 的两种主要版本:Community 和 Professional。通过对比两者的功能和附加组件,帮助用户选择最适合其需求的版本。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • 本文详细介绍了 Apache Jena 库中的 Txn.executeWrite 方法,通过多个实际代码示例展示了其在不同场景下的应用,帮助开发者更好地理解和使用该方法。 ... [详细]
  • libsodium 1.0.15 发布:引入重大不兼容更新
    最新发布的 libsodium 1.0.15 版本带来了若干不兼容的变更,其中包括默认密码散列算法的更改和其他重要调整。 ... [详细]
  • 本文详细介绍了 Java 中 org.eclipse.jface.viewers.ViewerCell 类的 getBackground() 方法,并提供了多个实际代码示例,帮助开发者更好地理解和应用该方法。 ... [详细]
author-avatar
乐乐
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有