scala : 类型与类
scala类型系统:1) 类型与类
在Java里,一直到jdk1.5之前,我们说一个对象的类型(type),都与它的class
是一一映射的,通过获取它们的class
对象,比如 String.class
, int.class
, obj.getClass()
等,就可以判断它们的类型(type)是不是一致的。
而到了jdk1.5之后,因为引入了泛型的概念,类型系统变得复杂了,并且因为jvm选择了在运行时采用类型擦拭的做法(兼容性考虑),类型已经不能单纯的用class
来区分了,比如 List<String>
和List<Integer>
的class
都是 Class<List>
,然而两者类型(type)却是不同的。泛型类型的信息要通过反射的技巧来获取,同时java里增加了Type
接口来表达更泛的类型,这样对于 List<String>
这样由类型构造器和类型参数组成的类型,可以通过 Type
来描述;它和 List<Integer>
类型的对应的Type
对象是完全不同的。
在Scala里,类型系统又比java复杂很多,泛型从一开始就存在,还支持高阶的概念(后续会讲述)。所以它没有直接用Java里的Type接口,而是自己提供了一个scala.reflect.runtime.universe.Type
(2.10后)
在scala里获取类型信息是比较便捷的:
scala> import scala.reflect.runtime.universe._
scala> class A
scala> typeOf[A]
res44: reflect.runtime.universe.Type = A
同样scala里获取类(Class)信息也很便捷,类似:
scala> classOf[A]
res52: Class[A] = class A
另外,因为java的Object里提供了getClass方法,对于对象来说,可以直接调用这个方法
scala> val a = new A
scala> a.getClass
res53: Class[_ <: A] = class A
scala> trait T
scala> classOf[T]
res50: Class[T] = interface T
scala> typeOf[T]
res51: reflect.runtime.universe.Type = T
注意,typeOf
和 classOf
方法接收的都是类型符号(symbol),并不是对象实例
scala> object O
scala> classOf[O] // 这里O是一个单例对象
<console>:14: error: not found: type O
对于实例,要获取他的 Class 信息,只有通过 getClass 方法
scala> O.getClass
res60: Class[_ <: O.type] = class O$
注意到了,上面的 单例对象O
对应的class是 O$ 而不是 O,你通过 :javap O
也能看到这个单例反编译后是一个名为O$的java class
而这个单例的类型更有趣了:O.type 看上去像是这个单例内部的一个成员,用这个成员的值表示其类型;实际上.type
之前的都可以看做是一种类型路径,这种特殊的类型也叫单例类型,它是面向对象实例的,每个实例都可以通过.type
方式表达它的单例类型,这个后续我们再说。
再举一个例子:
scala> class A { class B } // 嵌套类
scala> val a1 = new A
scala> val a2 = new A
scala> val b1 = new a1.B
scala> val b2 = new a2.B
对于内部类B的实例,它们的class都是相同的: A$B
scala> b1.getClass
res8: Class[_ <: a1.B] = class A$B
scala> b1.getClass == b2.getClass
res7: Boolean = true
而它们的类型却是不同的:
scala> typeOf[a1.B] == typeOf[a2.B]
res10: Boolean = false
这是因为内部类型依赖外部实例(路径依赖类型),外部实例不同,它们也不同。但还可以对这种类型再抽象
scala> typeOf[a1.B] <:< typeOf[A#B]
res11: Boolean = true
scala> typeOf[a2.B] <:< typeOf[A#B]
res12: Boolean = true
这里A#B
涉及到类型投影的概念,以后再讲。
简单的说,类(class)与类型(type)是两个不一样的概念(在java里因为早期一直使用class表达type,并且现在也延续这样的习惯);类型(type)比类(class)更”具体”,任何数据都有类型。类是面向对象系统里对同一类数据的抽象,在没有泛型之前,类型系统不存在高阶概念,直接与类一一映射,而泛型出现之后,就不在一一映射了。比如定义class List[T] {}
, 可以有List[Int]
和 List[String]
等具体类型,它们的类是同一个List
,但类型则根据不同的构造参数类型而不同。
类型一致的对象它们的类也是一致的,反过来,类一致的,其类型不一定一致。
scala> classOf[List[Int]] == classOf[List[String]]
res16: Boolean = true
scala> typeOf[List[Int]] == typeOf[List[String]]
res17: Boolean = false
在jvm里,类的实例数据都是引用形式,而类型没有这个约束,基础类型int
,byte
,char
等就是非引用的。(虽然可以通过int.class
来获取Class对象,但并不能找到有定义class int
的地方,这只是早期java为了统一用class来承载其类型信息的方式)
小结,类型是所有编程语言都有的概念,一切数据都有类型。类更多存在于面向对象语言,非面向对象语言也有“结构体”等与之相似的概念;类是对数据的抽象,而类型则是对数据的”分类”,类型比类更“具体”,更“细”一些。
scala : 类型与类的更多相关文章
- Scala类型系统——高级类类型(higher-kinded types)
高级类类型就是使用其他类型构造成为一个新的类型,因此也称为 类型构造器(type constructors).它的语法和高阶函数(higher-order functions)相似,高阶函数就是将其它 ...
- Scala 深入浅出实战经典 第63讲:Scala中隐式类代码实战详解
王家林亲授<DT大数据梦工厂>大数据实战视频 Scala 深入浅出实战经典(1-87讲)完整视频.PPT.代码下载:百度云盘:http://pan.baidu.com/s/1c0noOt6 ...
- Programming In Scala笔记-第十一章、Scala中的类继承关系
本章主要从整体层面了解Scala中的类层级关系. 一.Scala的类层级 在Java中Object类是所有类的最终父类,其他所有类都直接或间接的继承了Object类.在Scala中所有类的最终父类为A ...
- Scala学习(五)---Scala中的类
Scala中的类 摘要: 在本篇中,你将会学习如何用Scala实现类.如果你了解Java或C++中的类,你不会觉得这有多难,并且你会很享受Scala更加精简的表示法带来的便利.本篇的要点包括: 1. ...
- Scala类型检查与转换
Scala类型检查与转换 isInstanceOf:检查某个对象是否属于某个给定的类. asInstanceOf:将引用转换为子类的引用. classOf:如果想测试p指向的是一个Employee对象 ...
- Scala(三):类
类:Class 1.简单类和无参方法 2.带getter和setter属性 3.只带getter属性 4.对象私有字段 5.Bean属性 6.辅助构造器 7.主构造器 8.嵌套类 1.简单类和无参方法 ...
- Scala中的类学习
Scala中的类学习 从java了解类的情况下,了解Scala的类并不难.Scala类中的字段自动带getter和setter方法,用@BeanProperty注解生成javaBean对象的getXX ...
- Scala基础:类和构造器
类 package com.zy.scala.cls /** * 在 Scala 中,类并不用声明为 public 类型的. * Scala 源文件中可以包含多个类,所有这些类都具有共有可见性. */ ...
- Scala具体解释---------类
Scala中的类 摘要: 在本篇中.你将会学习怎样用Scala实现类. 假设你了解Java或C++中的类,你不会认为这有多难.而且你会非常享受Scala更加精简的表示法带来的便利.本篇的要点包含: 1 ...
随机推荐
- xss练习平台及writeup
今天玩了一天的xss. 分享几个xss game https://xss.haozi.me/#/0x00 http://47.94.13.75/test/ writeup:http://www.cn ...
- selenium + python自动化测试unittest框架学习(三)webdriver对页面其他控件操作(三)
1.对话框,下拉框 (1)对话框的有两种,一种是iframe格式的,需要switch_to_iframe()进行定位,现在大部分的对话框是div格式的,这种格式的可以通过层级定位来定位元素,先定位对话 ...
- unordered_map 遇到 vector subscript out of range 的错误提示
错误类型 当调用unordered_map的函数的时候,会出现如下问题: 使用linux运行则会提示 float exeption(core dump) 原因 遇到vector subscript o ...
- Linux内存管理-高端内存(一)
高端内存是指物理地址大于 896M 的内存.对于这样的内存,无法在“内核直接映射空间”进行映射. 为什么? 因为“内核直接映射空间”最多只能从 3G 到 4G,只能直接映射 1G 物理内存,对于大于 ...
- xcode10不兼容问题解决方法,framework编译脚本
XCode10报错:Build/Intermediates.noindex/XCBuildData/build.db": disk I/O error 更改-scheme 为-target ...
- Oracle高级函数篇之递归查询start with connect by prior简单用法
路飞:" 把原来CSDN的博客转移到博客园咯!" 前段时间,自己负责的任务中刚好涉及到了组织关系的业务需求,自己用了oracle递归查询.下面简单来举个例子.在工作中我们经常会遇到 ...
- iOS获取设备ip地址(OC版)
#import <SystemConfiguration/CaptiveNetwork.h> #import <ifaddrs.h> #import <arpa/inet ...
- 一个简单的Linux启动jar包的shell脚本
背景: 项目设备端需要运行jar包程序与服务端进行socket连接并发送数据,每次启动进程时,都需要在Linux终端输入启动jar包的命令,比较繁琐,随之尝试将启动jar包的命令写入shell脚本文件 ...
- 偏前端-vue.js学习之路初级(一)概念
首先--不推荐新手直接使用 vue-cli,尤其是在你还不熟悉基于 Node.js 的构建工具时. 新建一个html,引入一下js: <!-- 开发环境版本,包含了有帮助的命令行警告 -- ...
- CDH升级 5.7.5 --> 5.13.3(tar包方式)
博客园首发,转载请注明出处:https://www.cnblogs.com/tzxxh/p/9123231.html 一.准备 1.关闭cdh中的服务 hdfs.yarn等所有服务:关闭 cm-ser ...