[读书日志]从零开始学习Chisel 第十三篇:Scala的隐式参数与隐式转换(敏捷硬件开发语言Chisel与数字系统设计)
10. 隐式转换与隐式参数
假设编写了一个向量类MyVector
,并且包含一些向量的基本操作。因为向量可以与标量做数乘运算,所以需要一个计算数乘的方法“*”,它应该接收一个类型为基本值类的参数,在向量对象myVec
调用该方法时,可以写成myVec * 2
的形式。在数学中,反过来写2 * myVec
也是可行的,但在程序中不行。操作符左边是调用对象,反过来写就表示Int
对象2是方法的调用者,但是Int类中并没有这种方法。
为了解决上述问题,所有oop语言都会有相应的措施,C++采用友元,Scala采取名为隐式转换的策略,把原来的Int
类对象2转换类型,变成MyVector
类的对象,这样它就能使用数乘方法。隐式转换属于隐式定义的一种,它将程序员事先写好的定义,允许编译器隐式地插入这些定义来解决类型错误。
10.1 隐式定义的规则
隐式定义有如下约束规则:
- 标记规则:只有用关键字
implicit
标记的定义才能被编译器隐式使用,任何函数,变量或单例对象都可以被标记。其中,标记为隐式的变量和单例对象常用作隐式参数。隐式的函数常用于隐式转换。比如x + y
由于x类型错误不能通过编译,编译器会将代码改成convert(x) + y
,其中convert
是某种可用的隐式转换。如果convert
能把x改成某种支持+
方法的对象,那么这段代码就可以通过类型检查; - 作用域规则:Scala编译器只会考虑当前作用域内的隐式定义,隐式定义在当前作用域必须是单个标记符,编译器不会展开成
A.convert(x) + y
的形式,有一个例外是编译器会在与隐式转换相关的源类型和目标类型的伴生对象中查找隐式定义,因此常在伴生对象中定义隐式转换,而不需要显式导入; - 每次一个规则:编译器只会插入一个隐式定义,不会出现
convert1(convert2(x)) + y
这种嵌套的形式; - 显式优先规则:如果显式定义能通过类型检查,就不必进行隐式转换。
Scala只会在三个地方使用隐式定义:转换到一个预期的类型,转换某个选择接收端,隐式参数。
10.2 隐式地转换到期望类型
Scala编译器对于类型检查比较严格,比如一个浮点数赋值给整数变量,人们可能希望通过截断小数部分完成赋值,但是Scala默认情况下不允许这种丢失精度的转换。可以定义一个隐式转换来完成:
scala> import scala.language.implicitConversionsscala> implicit def doubleToInt(x: Double) : Int = x.toInt
def doubleToInt(x: Double): Intscala> val i: Int = 1.5
val i: Int = 1scala> val i: Int = doubleToInt(2.33)
val i: Int = 2
10.3 隐式地转换接收端
接收端就是指调用方法或字段的对象,调用对象在非法的情况下,被隐式转换成了合法的对象,这是隐式转换最常用的地方。
scala> class MyInt(val i: Int)
// defined class MyIntscala> 1.i
-- [E008] Not Found Error: ---------------------------------------------------------------------------------------------
1 |1.i|^^^|value i is not a member of Int
1 error foundscala> implicit def intToMy(x: Int): MyInt = new MyInt(x)
def intToMy(x: Int): MyIntscala> 1.i
val res0: Int = 1
在定义隐式转换前,Int类没有i字段,所以调用会报错,定义隐式转换后,用一个Int对象构造了一个新的MyInt
对象,而MyInt
对象就有字段i。1.i被编译器隐式展开成intToMy(1).i
。
隐式转换因此经常用于模拟新的语法,比如Chisel中大量使用了隐式定义。
映射的键-值对语法键->值其实是一个对偶(键,值),也是隐式转换起作用。
10.4 隐式类
隐式类是一个以关键字implicit
开头的类,用于简化富包装类的编写。它不能是样例类,并且主构造方法有且仅有一个参数。此外,隐式类只能位于某个单例对象、类或特质中,不能单独出现在顶层。隐式类的特点是让编译器在相同层次下自动生成一个与类名相同的隐式转换,该转换接受一个与隐式类的主构造方法相同的参数,并用这个参数构造一个隐式类的实例对象来返回。
10.5 隐式参数
函数的最后一个参数列表可以用关键字implicit
声明为隐式的,这样整个参数列表的参数都是隐式参数。注意,是整个参数列表,即使括号中有多个参数,也只需要在开头写一个implicit
,每个参数都是隐式的,不存在部分隐式部分显式。
当调用函数时,若缺省了隐式参数列表,则编译器尝试插入相应的隐式定义。也可以显式给出参数,但是必须全部缺省或者全部写出,不能只写出一部分。
要让编译器隐式插入参数,就必须实现定义好符合预期类型的隐式变量,隐式单例对象和隐式函数,这些隐式定义也必须用implicit
修饰。隐式变量,单例对象,函数在当前作用域的引用必须满足单标识符原则,即不同层次之间需要用import
解决。
隐式参数的类型应该是稀有或特定的,类型名称最好能表明该参数的作用,而不是使用Int,String等已经约定好的名称。
class PreferredPrompt(val preference: String)
class PreferredDrink(val preference: String)object Greeter {def greet(name: String)(implicit prompt: PreferredPrompt, drink: PreferredDrink) = {println("Welcome, " + name + ". The system is ready.")print("But while you work, ")println("why not enjoy a cup of " + drink.preference + "?")println(prompt.preference)}
}object JoesPrefs {implicit val prompt: PreferredPrompt = new PreferredPrompt("Yes, master> ")implicit val drink: PreferredDrink = new PreferredDrink("tea")
}
保存后执行编译指令,并进入解释器,执行命令:
jia@J-MateBookEGo:~/scala_test$ scalac test.scala
jia@J-MateBookEGo:~/scala_test$ scala
Jan 13, 2025 1:09:02 PM org.jline.utils.Log logr
Welcome to Scala 3.6.2 (21.0.5, Java OpenJDK 64-Bit Server VM).
Type in expressions for evaluation. Or try :help.scala> :load test.scala
// defined class PreferredPrompt
// defined class PreferredDrink
// defined object Greeter
// defined object JoesPrefsscala> Greeter.greet("Joe")
-- [E172] Type Error: --------------------------------------------------------------------------------------------------
1 |Greeter.greet("Joe")| ^| No given instance of type PreferredPrompt was found for parameter prompt of method greet in object Greeter|| The following import might fix the problem:|| import JoesPrefs.prompt|
1 error foundscala> import JoesPrefs._scala> Greeter.greet("Joe")
Welcome, Joe. The system is ready.
But while you work, why not enjoy a cup of tea?
Yes, master>scala> Greeter.greet("Joe")(prompt, drink)
Welcome, Joe. The system is ready.
But while you work, why not enjoy a cup of tea?
Yes, master>
10.6 含有隐式参数的主构造方法
普通的函数可以含有隐式参数,类的主构造方法也可以包含隐式参数,辅助构造方法不允许出现隐式参数。假设类A仅有一个参数列表,并且该列表是隐式的,那么A的实际定义形式是A()(implicit 参数)
,也就是比字面上的代码多了一对空括号,无论用new
实例化类A,还是被其他类继承,若调用主构造方法时显式地给出隐式参数,就必须写出这对空括号,若隐式参数是编译器自动插入,则空括号可有可无。
scala> class A(implicit val x: Int)
// defined class Ascala> val a = new A(1)
-- Error: --------------------------------------------------------------------------------------------------------------
1 |val a = new A(1)| ^| too many arguments for constructor A in class A: ()(implicit x: Int): A
1 error foundscala> val a = new A()(1)
val a: A = A@6caeba36scala> implicit val ORZ = 233
-- Error: --------------------------------------------------------------------------------------------------------------
1 |implicit val ORZ = 233| ^| type of implicit definition needs to be given explicitly
1 error foundscala> implicit val ORZ: Int = 233
val ORZ: Int = 233scala> val b = new A
val b: A = A@6a7aa675scala> b.x
val res0: Int = 233scala> val c = new A()
val c: A = A@1534bdc6scala> c.x
val res1: Int = 233
相关文章:
[读书日志]从零开始学习Chisel 第十三篇:Scala的隐式参数与隐式转换(敏捷硬件开发语言Chisel与数字系统设计)
10. 隐式转换与隐式参数 假设编写了一个向量类MyVector,并且包含一些向量的基本操作。因为向量可以与标量做数乘运算,所以需要一个计算数乘的方法“*”,它应该接收一个类型为基本值类的参数,在向量对象myVec调用该方法时…...

CMake学习笔记(1)
1. CMake概述 CMake 是一个项目构建工具,并且是跨平台的。关于项目构建我们所熟知的还有Makefile(通过 make 命令进行项目的构建),大多是IDE软件都集成了make,比如:VS 的 nmake、linux 下的 GNU make、Qt …...

cursor+deepseek构建自己的AI编程助手
文章目录 准备工作在Cursor中添加deepseek 准备工作 下载安装Cursor (默认安装在C盘) 注册deepseek获取API key 在Cursor中添加deepseek 1、打开cursor,选择设置 选择Model,添加deepseek-chat 注意这里去掉其他的勾选项&…...
Kotlin实现DataBinding结合ViewModel的时候,提示找不到Unresolved reference: BR解决方案
在用Kotlin语言实现DataBinding结合ViewModel的代码的时候,如下所示: class UserModel(private val userName: String, private val userAge: Int) : BaseObservable() {get:Bindablevar name: String userNameset (value) {field valuenotifyPropert…...
java项目启动时,执行某方法
1. J2EE项目 在Servlet类中重写init()方法,这个方法会在Servlet实例化时调用,即项目启动时调用。 import javax.servlet.ServletException; import javax.servlet.http.HttpServlet;public class MyServlet extends HttpServlet {Overridepublic void …...

详解如何自定义 Android Dex VMP 保护壳
版权归作者所有,如有转发,请注明文章出处:https://cyrus-studio.github.io/blog/ 前言 Android Dex VMP(Virtual Machine Protection,虚拟机保护)壳是一种常见的应用保护技术,主要用于保护 And…...

Grails应用http.server.requests指标数据采集问题排查及解决
问题 遇到的问题:同一个应用,Spring Boot(Java)和Grails(Groovy)混合编程,常规的Spring Controller,可通过Micromete Pushgateway, 采集到http.server.requests指标数据,注意下面的指标名称是点号&#…...

开源临床试验软件OpenClinica的安装
本文是为帮网友 A萤火虫 解决安装问题做的记录; 简介 什么是 OpenClinica ? OpenClinica 是世界上第一个商业开源临床试验软件,主要用于电子数据捕获(EDC)和临床数据管理(CDM)。它的设计旨在优…...

网络安全 | 网络安全法规:GDPR、CCPA与中国网络安全法
网络安全 | 网络安全法规:GDPR、CCPA与中国网络安全法 一、前言二、欧盟《通用数据保护条例》(GDPR)2.1 背景2.2 主要内容2.3 特点2.4 实施效果与影响 三、美国《加利福尼亚州消费者隐私法案》(CCPA)3.1 背景3.2 主要内…...
深入学习 Python 爬虫:从基础到实战
深入学习 Python 爬虫:从基础到实战 前言 Python 爬虫是一个强大的工具,可以帮助你从互联网上抓取各种数据。无论你是数据分析师、机器学习工程师,还是对网络数据感兴趣的开发者,爬虫都是一个非常实用的技能。在本文中ÿ…...

element plus 使用 upload 组件达到上传数量限制时隐藏上传按钮
最近在重构项目,使用了 element plus UI框架,有个功能是实现图片上传,且限制只能上传一张图片,结果,发现,可以限制只上传一张图片,但是上传按钮还在,如图: 解决办法&…...
音频DSP的发展历史
音频数字信号处理(DSP)的发展历史是电子技术、计算机科学和音频工程共同进步的结果。这个领域的进展不仅改变了音乐制作、音频后期制作和通信的方式,也影响了音频设备的设计和功能。以下是对音频DSP发展历史的概述: 早期概念和理论…...

2025低代码与人工智能AI新篇
在当今数字化浪潮汹涌澎湃的时代,低代码开发与人工智能(AI)犹如两颗璀璨的星辰,正逐渐交汇融合,为企业解锁前所未有的智能业务解决方案。今天,咱们就深入探讨一下低代码平台是如何集成 AI 技术,…...
【HarmonyOS Next NAPI 深度探索1】Node.js 和 CC++ 原生扩展简介
【HarmonyOS Next NAPI 深度探索1】Node.js 和 CC 原生扩展简介 如果你用过 Node.js,应该知道它强大的地方在于能处理各种场景,速度还很快。但你有没有想过,Node.js 的速度秘密是什么?今天我们来聊聊其中一个幕后英雄——原生扩展…...

redis的学习(四)
13. 渐进式遍历 通过渐进式遍历能够获取当前所有的key,又不会讲当前的服务器卡死。不是一个命令将所有的key获取,而是每执行一次命令,只获取到其中的一部分。所以想要获取到所有的key就需要多次遍历,即化整为零的思想。 渐进式遍历…...
C# winform 多线程 UI更新数据 报错:无法访问已释放的对象。
System.ObjectDisposedException HResult0x80131622 Message无法访问已释放的对象。 ObjectDisposed_ObjectName_Name SourceSystem.Windows.Forms StackTrace: at System.Windows.Forms.Control.MarshaledInvoke(Control caller, Delegate method, Object[] args, …...

error: linker `link.exe` not found
开始学习rust,安装好rust的环境,开始从hello world开始,结果用在win10环境下,使用vs code或cmd窗口编译rust报错: PS E:\study_codes\rust-demo\chart01> rustc hello.rs error: linker link.exe not found| note:…...
Vue.js组件开发-如何使用moment.js
在Vue.js组件开发中,需要处理日期和时间,moment.js 是一个非常有用的库。moment.js 提供了丰富的API来解析、验证、操作和显示日期和时间。 步骤: 1. 安装moment.js 首先,需要通过npm或yarn安装moment.js。在项目根目录下运行以…...

Linux第二课:LinuxC高级 学习记录day01
0、大纲 0.1、Linux 软件安装,用户管理,进程管理,shell 命令,硬链接和软连接,解压和压缩,功能性语句,结构性语句,分文件,make工具,shell脚本 0.2、C高级 …...
《DOM NodeList》
《DOM NodeList》 介绍 DOM(文档对象模型)是HTML和XML文档的编程接口,它允许开发者在JavaScript等编程语言中操作文档的结构、样式和内容。在DOM中,NodeList是一个重要的接口,它表示一个包含节点(如元素、…...
零门槛NAS搭建:WinNAS如何让普通电脑秒变私有云?
一、核心优势:专为Windows用户设计的极简NAS WinNAS由深圳耘想存储科技开发,是一款收费低廉但功能全面的Windows NAS工具,主打“无学习成本部署” 。与其他NAS软件相比,其优势在于: 无需硬件改造:将任意W…...

地震勘探——干扰波识别、井中地震时距曲线特点
目录 干扰波识别反射波地震勘探的干扰波 井中地震时距曲线特点 干扰波识别 有效波:可以用来解决所提出的地质任务的波;干扰波:所有妨碍辨认、追踪有效波的其他波。 地震勘探中,有效波和干扰波是相对的。例如,在反射波…...
DeepSeek 赋能智慧能源:微电网优化调度的智能革新路径
目录 一、智慧能源微电网优化调度概述1.1 智慧能源微电网概念1.2 优化调度的重要性1.3 目前面临的挑战 二、DeepSeek 技术探秘2.1 DeepSeek 技术原理2.2 DeepSeek 独特优势2.3 DeepSeek 在 AI 领域地位 三、DeepSeek 在微电网优化调度中的应用剖析3.1 数据处理与分析3.2 预测与…...
QMC5883L的驱动
简介 本篇文章的代码已经上传到了github上面,开源代码 作为一个电子罗盘模块,我们可以通过I2C从中获取偏航角yaw,相对于六轴陀螺仪的yaw,qmc5883l几乎不会零飘并且成本较低。 参考资料 QMC5883L磁场传感器驱动 QMC5883L磁力计…...

《用户共鸣指数(E)驱动品牌大模型种草:如何抢占大模型搜索结果情感高地》
在注意力分散、内容高度同质化的时代,情感连接已成为品牌破圈的关键通道。我们在服务大量品牌客户的过程中发现,消费者对内容的“有感”程度,正日益成为影响品牌传播效率与转化率的核心变量。在生成式AI驱动的内容生成与推荐环境中࿰…...
五年级数学知识边界总结思考-下册
目录 一、背景二、过程1.观察物体小学五年级下册“观察物体”知识点详解:由来、作用与意义**一、知识点核心内容****二、知识点的由来:从生活实践到数学抽象****三、知识的作用:解决实际问题的工具****四、学习的意义:培养核心素养…...

P3 QT项目----记事本(3.8)
3.8 记事本项目总结 项目源码 1.main.cpp #include "widget.h" #include <QApplication> int main(int argc, char *argv[]) {QApplication a(argc, argv);Widget w;w.show();return a.exec(); } 2.widget.cpp #include "widget.h" #include &q…...
Java 加密常用的各种算法及其选择
在数字化时代,数据安全至关重要,Java 作为广泛应用的编程语言,提供了丰富的加密算法来保障数据的保密性、完整性和真实性。了解这些常用加密算法及其适用场景,有助于开发者在不同的业务需求中做出正确的选择。 一、对称加密算法…...
vue3 定时器-定义全局方法 vue+ts
1.创建ts文件 路径:src/utils/timer.ts 完整代码: import { onUnmounted } from vuetype TimerCallback (...args: any[]) > voidexport function useGlobalTimer() {const timers: Map<number, NodeJS.Timeout> new Map()// 创建定时器con…...
【C语言练习】080. 使用C语言实现简单的数据库操作
080. 使用C语言实现简单的数据库操作 080. 使用C语言实现简单的数据库操作使用原生APIODBC接口第三方库ORM框架文件模拟1. 安装SQLite2. 示例代码:使用SQLite创建数据库、表和插入数据3. 编译和运行4. 示例运行输出:5. 注意事项6. 总结080. 使用C语言实现简单的数据库操作 在…...