百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

Java修炼终极指南:92. 引入instanceof的记录模式

yuyutoo 2024-10-12 01:47 3 浏览 0 评论


为了引入记录模式,我们需要一个记录,所以这里是一个:

public record Doctor(String name, String specialty)
  implements Staff {}


这个记录实现了Staff接口,就像我们医院的其他员工一样。现在,我们可以通过以下方式通过instanceof以老式风格识别某位医生:

public static String cabinet(Staff staff) {
  if (staff instanceof Doctor) {
    Doctor dr = (Doctor) staff;
    return "Cabinet of " + dr.specialty()
      + ". Doctor: " + dr.name();
  }
  ...
}


但是,正如我们在第2章,问题x-y中所知,JDK引入了可用于instanceof和switch的类型模式。因此,在这种特定情况下,我们可以通过类型模式重写前面的代码,如下所示:

public static String cabinet(Staff staff) {
       
  if (staff instanceof Doctor dr) { // 类型模式匹配
    return "Cabinet of " + dr.specialty()
       + ". Doctor: " + dr.name();
  }
  ...
}


到目前为止没有什么新东西!绑定变量dr可以用来调用记录访问器specialty()和name(),添加检查,计算等。但是,编译器非常清楚地知道Doctor记录是基于两个组件(name和specialty)构建的,所以编译器应该能够解构这个对象,并直接给我们这些组件作为绑定变量,而不是通过dr访问它们。这正是记录模式匹配的全部内容。记录模式匹配作为预览特性在JDK 19(JEP 405)中引入,并在JDK 20(JEP 432)中作为第二个预览特性引入。

记录模式匹配正是按照记录本身的相同声明语法(或像在规范构造函数中一样)将name和specialty声明为绑定变量的语法。这是通过记录模式重写的先前代码:

public static String cabinet(Staff staff) {          
  // 记录模式匹配
  if (staff instanceof Doctor(String name, String specialty)){     
      return "Cabinet of " + name + ". Doctor: " + specialty;
  }
  ...
}


非常简单,不是吗?现在,name和specialty是可以直接使用的绑定变量。我们简单地将这个语法放在类型模式的位置。换句话说,我们将类型模式替换为记录模式。

编译器通过相应的绑定变量公开记录的组件。这是通过在模式匹配中解构记录来实现的,这被称为记录模式。换句话说,解构模式允许我们以非常方便、直观和易于阅读的方式访问对象的组件。

当然,如果你需要dr绑定变量(即记录本身的引用)在name和specialty绑定变量旁边,那么可以像下面这样简单地添加:

public static String cabinet(Staff staff) {          
  if (staff instanceof
    Doctor(String name, String specialty) dr) {
      return "Cabinet of " + specialty + ". Doctor ID: "
        + dr.hashCode() + " (" + name + ")";
  }
  ...
}


这真的很酷!

在记录模式中,编译器有责任初始化name和specialty等绑定变量。为了完成这项工作,编译器调用相应组件的访问器。这意味着,如果你在这些访问器中有一些额外的代码(例如,返回防御性副本,执行验证或应用约束等),那么这个代码将被正确执行。

让我们进一步处理一些嵌套记录。

嵌套记录和记录模式

假设除了Doctor记录外,我们还有以下记录:

public record Resident(String name, Doctor doctor)
  implements Staff {}


每个住院医师都有一个协调员,即医生,所以Resident嵌套了一个Doctor。这次,我们必须相应地嵌套记录模式,如下所示:

public static String cabinet(Staff staff) {   
      
  if (staff instanceof Resident(String rsname,
      Doctor(String drname, String specialty))) {            
    return "Cabinet of " + specialty + ". Doctor: "
                         + drname + ", Resident: " + rsname;
  } 
...
}


住院医师和医生都有name组件。但是,我们不能在这个上下文中两次使用绑定变量name,因为这会导致冲突。这就是为什么我们有rsname和drname。请注意,绑定变量的名称不需要反映组件的名称。这是可能的,因为编译器通过位置而不是通过它们的名称来识别组件。但是,当然,如果可能的话,反映名称可以减少混乱并保持代码的高可读性。如果不需要解构Doctor记录,我们可以这样写:

if (staff instanceof Resident(String name, Doctor dr)) {
  return "Cabinet of " + dr.specialty() + ". Doctor: "
                       + dr.name() + ", Resident: " + name;
}


或者,如果你需要完全解构组件并引用记录,那么可以这样写:

if (staff instanceof Resident(String rsname,
  Doctor(String drname, String specialty) dr) rs) {
    return "Cabinet of " + dr.specialty() + ". Doctor: "
            + dr.name() + ", Resident: " + rs.name();
}


添加更多嵌套记录遵循相同的原则。例如,让我们也添加Patient和Appointment记录:

public record Appointment(LocalDate date, Doctor doctor) {}
public record Patient(
  String name, int npi, Appointment appointment) {}


现在,我们可以写出以下美妙的代码:

public static String reception(Object o) {
  if (o instanceof Patient(var ptname, var npi,
                  Appointment(var date,
                  Doctor (var drname, var specialty)))) {
       
   return "Patient " + ptname + " (NPI: " + npi
          + ") has an appointment at "
          + date + " to the doctor " + drname
          + " (" + specialty + ").";
  }
…
}


这次我们使用了var而不是显式类型。在这种情况下,你可以自由地这样做,因为var非常适合。如果你不熟悉类型推断,那么考虑《Java编码问题》,第一版,第4章类型推断,其中包含详细的解释和最佳实践。当然,也支持使用显式类型。在这里,我们还为引用记录添加了绑定变量:

if (o instanceof Patient(String ptname, int npi,
        Appointment(LocalDate date,
        Doctor (String drname, String specialty) dr) ap) pt) {
       
  return "Patient " + pt.name() + " (NPI: " + pt.npi()
     + ") has an appointment at "
     + ap.date() + " to the doctor " + dr.name() + " ("
     + dr.specialty() + ").";
}


我认为你已经明白了!

相关推荐

史上最全的浏览器兼容性问题和解决方案

微信ID:WEB_wysj(点击关注)◎◎◎◎◎◎◎◎◎一┳═┻︻▄(页底留言开放,欢迎来吐槽)●●●...

平面设计基础知识_平面设计基础知识实验收获与总结
平面设计基础知识_平面设计基础知识实验收获与总结

CSS构造颜色,背景与图像1.使用span更好的控制文本中局部区域的文本:文本;2.使用display属性提供区块转变:display:inline(是内联的...

2025-02-21 16:01 yuyutoo

写作排版简单三步就行-工具篇_作文排版模板

和我们工作中日常word排版内部交流不同,这篇教程介绍的写作排版主要是用于“微信公众号、头条号”网络展示。写作展现的是我的思考,排版是让写作在网格上更好地展现。在写作上花费时间是有累积复利优势的,在排...

写一个2048的游戏_2048小游戏功能实现

1.创建HTML文件1.打开一个文本编辑器,例如Notepad++、SublimeText、VisualStudioCode等。2.将以下HTML代码复制并粘贴到文本编辑器中:html...

今天你穿“短袖”了吗?青岛最高23℃!接下来几天气温更刺激……

  最近的天气暖和得让很多小伙伴们喊“热”!!!  昨天的气温到底升得有多高呢?你家有没有榜上有名?...

CSS不规则卡片,纯CSS制作优惠券样式,CSS实现锯齿样式

之前也有写过CSS优惠券样式《CSS3径向渐变实现优惠券波浪造型》,这次再来温习一遍,并且将更为详细的讲解,从布局到具体样式说明,最后定义CSS变量,自定义主题颜色。布局...

柠檬科技肖勃飞:大数据风控助力信用社会建设

...

你的自我界限够强大吗?_你的自我界限够强大吗英文

我的结果:A、该设立新的界限...

行内元素与块级元素,以及区别_行内元素和块级元素有什么区别?

行内元素与块级元素首先,CSS规范规定,每个元素都有display属性,确定该元素的类型,每个元素都有默认的display值,分别为块级(block)、行内(inline)。块级元素:(以下列举比较常...

让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华
让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华

去年的两会期间,习近平总书记在参加人大会议四川代表团审议时,对治蜀兴川提出了明确要求,指明了前行方向,并带来了“祝四川人民的生活越来越安逸”的美好祝福。又是一年...

2025-02-21 16:00 yuyutoo

今年国家综合性消防救援队伍计划招录消防员15000名

记者24日从应急管理部获悉,国家综合性消防救援队伍2023年消防员招录工作已正式启动。今年共计划招录消防员15000名,其中高校应届毕业生5000名、退役士兵5000名、社会青年5000名。本次招录的...

一起盘点最新 Chrome v133 的5大主流特性 ?

1.CSS的高级attr()方法CSSattr()函数是CSSLevel5中用于检索DOM元素的属性值并将其用于CSS属性值,类似于var()函数替换自定义属性值的方式。...

竞走团体世锦赛5月太仓举行 世界冠军杨家玉担任形象大使

style="text-align:center;"data-mce-style="text-align:...

学物理能做什么?_学物理能做什么 卢昌海

作者:曹则贤中国科学院物理研究所原标题:《物理学:ASourceofPowerforMan》在2006年中央电视台《对话》栏目的某期节目中,主持人问过我一个的问题:“学物理的人,如果日后不...

你不知道的关于这只眯眼兔的6个小秘密
你不知道的关于这只眯眼兔的6个小秘密

在你们忙着给熊本君做表情包的时候,要知道,最先在网络上引起轰动的可是这只脸上只有两条缝的兔子——兔斯基。今年,它更是迎来了自己的10岁生日。①关于德艺双馨“老艺...

2025-02-21 16:00 yuyutoo

取消回复欢迎 发表评论: