热门标签 | 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]

推荐阅读
  • 本文详细探讨了JDBC(Java数据库连接)的内部机制,重点分析其作为服务提供者接口(SPI)框架的应用。通过类图和代码示例,展示了JDBC如何注册驱动程序、建立数据库连接以及执行SQL查询的过程。 ... [详细]
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • 本文详细介绍了Akka中的BackoffSupervisor机制,探讨其在处理持久化失败和Actor重启时的应用。通过具体示例,展示了如何配置和使用BackoffSupervisor以实现更细粒度的异常处理。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 本文详细介绍了Java中org.w3c.dom.Text类的splitText()方法,通过多个代码示例展示了其实际应用。该方法用于将文本节点在指定位置拆分为两个节点,并保持在文档树中。 ... [详细]
  • 深入解析 Android IPC 中的 Messenger 机制
    本文详细介绍了 Android 中基于消息传递的进程间通信(IPC)机制——Messenger。通过实例和源码分析,帮助开发者更好地理解和使用这一高效的通信工具。 ... [详细]
  • 本文详细介绍了Java中的注解功能,包括如何定义注解类型、设置注解的应用范围及生命周期,并通过具体示例展示了如何利用反射机制访问注解信息。 ... [详细]
  • Java 架构:深入理解 JDK 动态代理机制
    代理模式是 Java 中常用的设计模式之一,其核心在于代理类与委托类共享相同的接口。代理类主要用于为委托类提供预处理、过滤、转发及后处理等功能,以增强或改变原有功能的行为。 ... [详细]
  • 技术分享:从动态网站提取站点密钥的解决方案
    本文探讨了如何从动态网站中提取站点密钥,特别是针对验证码(reCAPTCHA)的处理方法。通过结合Selenium和requests库,提供了详细的代码示例和优化建议。 ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 主要用了2个类来实现的,话不多说,直接看运行结果,然后在奉上源代码1.Index.javaimportjava.awt.Color;im ... [详细]
  • 在前两篇文章中,我们探讨了 ControllerDescriptor 和 ActionDescriptor 这两个描述对象,分别对应控制器和操作方法。本文将基于 MVC3 源码进一步分析 ParameterDescriptor,即用于描述 Action 方法参数的对象,并详细介绍其工作原理。 ... [详细]
  • 本文探讨了在C++中如何有效地清空输入缓冲区,确保程序只处理最近的输入并丢弃多余的输入。我们将介绍一种不阻塞的方法,并提供一个具体的实现方案。 ... [详细]
  • 本文详细探讨了 org.apache.hadoop.ha.HAServiceTarget 类中的 checkFencingConfigured 方法,包括其功能、应用场景及代码示例。通过实际代码片段,帮助开发者更好地理解和使用该方法。 ... [详细]
  • 在编译BSP包过程中,遇到了一个与 'gets' 函数相关的编译错误。该问题通常发生在较新的编译环境中,由于 'gets' 函数已被弃用并视为安全漏洞。本文将详细介绍如何通过修改源代码和配置文件来解决这一问题。 ... [详细]
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社区 版权所有