(window.webpackJsonp=window.webpackJsonp||[]).push([[383],{819:function(e,t,r){"use strict";r.r(t);var a=r(56),n=Object(a.a)({},(function(){var e=this,t=e.$createElement,r=e._self._c||t;return r("ContentSlotsDistributor",{attrs:{"slot-key":e.$parent.slotKey}},[r("h1",{attrs:{id:"项目阅读器和项目编写器"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#项目阅读器和项目编写器"}},[e._v("#")]),e._v(" 项目阅读器和项目编写器")]),e._v(" "),r("h2",{attrs:{id:"条目阅读器和条目编写器"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#条目阅读器和条目编写器"}},[e._v("#")]),e._v(" 条目阅读器和条目编写器")]),e._v(" "),r("p",[e._v("XMLJavaBoth")]),e._v(" "),r("p",[e._v("所有批处理都可以用最简单的形式描述为读取大量数据,执行某种类型的计算或转换,并将结果写出来。 Spring Batch 提供了三个关键接口来帮助执行大容量读写:"),r("code",[e._v("ItemReader")]),e._v("、"),r("code",[e._v("ItemProcessor")]),e._v("和"),r("code",[e._v("ItemWriter")]),e._v("。")]),e._v(" "),r("h3",{attrs:{id:"itemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#itemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("ItemReader")])]),e._v(" "),r("p",[e._v("虽然是一个简单的概念,但"),r("code",[e._v("ItemReader")]),e._v("是从许多不同类型的输入提供数据的手段。最常见的例子包括:")]),e._v(" "),r("ul",[r("li",[r("p",[e._v("平面文件:平面文件项读取器从平面文件中读取数据行,该文件通常用文件中固定位置定义的数据字段或用某些特殊字符(例如逗号)分隔的数据字段来描述记录。")])]),e._v(" "),r("li",[r("p",[e._v("XML:XML"),r("code",[e._v("ItemReaders")]),e._v("独立于用于解析、映射和验证对象的技术来处理 XML。输入数据允许根据 XSD 模式验证 XML 文件。")])]),e._v(" "),r("li",[r("p",[e._v("数据库:访问数据库资源以返回结果集,这些结果集可以映射到对象以进行处理。默认的 SQL"),r("code",[e._v("ItemReader")]),e._v("实现调用"),r("code",[e._v("RowMapper")]),e._v("以返回对象,如果需要重新启动,则跟踪当前行,存储基本统计信息,并提供一些事务增强,稍后将对此进行说明。")])])]),e._v(" "),r("p",[e._v("还有更多的可能性,但我们将重点放在本章的基本可能性上。在"),r("RouterLink",{attrs:{to:"/spring-batch/appendix.html#listOfReadersAndWriters"}},[e._v("Appendix A")]),e._v("中可以找到所有可用"),r("code",[e._v("ItemReader")]),e._v("实现的完整列表。")],1),e._v(" "),r("p",[r("code",[e._v("ItemReader")]),e._v("是用于通用输入操作的基本接口,如以下接口定义所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public interface ItemReader {\n\n T read() throws Exception, UnexpectedInputException, ParseException, NonTransientResourceException;\n\n}\n")])])]),r("p",[r("code",[e._v("read")]),e._v("方法定义了"),r("code",[e._v("ItemReader")]),e._v("中最基本的契约。调用它将返回一个项,如果没有更多项,则返回"),r("code",[e._v("null")]),e._v("。项目可以表示文件中的行、数据库中的行或 XML 文件中的元素。通常预期这些被映射到一个可用的域对象(例如"),r("code",[e._v("Trade")]),e._v(","),r("code",[e._v("Foo")]),e._v(",或其他),但是在契约中没有这样做的要求。")]),e._v(" "),r("p",[e._v("预计"),r("code",[e._v("ItemReader")]),e._v("接口的实现方式仅是前向的。但是,如果底层资源是事务性的(例如 JMS 队列),那么在回滚场景中,调用"),r("code",[e._v("read")]),e._v("可能会在随后的调用中返回相同的逻辑项。还值得注意的是,缺少由"),r("code",[e._v("ItemReader")]),e._v("处理的项并不会导致抛出异常。例如,配置了返回 0 结果的查询的数据库"),r("code",[e._v("ItemReader")]),e._v("在"),r("code",[e._v("read")]),e._v("的第一次调用时返回"),r("code",[e._v("null")]),e._v("。")]),e._v(" "),r("h3",{attrs:{id:"itemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#itemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("ItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("ItemWriter")]),e._v("在功能上类似于"),r("code",[e._v("ItemReader")]),e._v(",但具有反向操作。资源仍然需要定位、打开和关闭,但它们的不同之处在于"),r("code",[e._v("ItemWriter")]),e._v("写出,而不是读入。在数据库或队列的情况下,这些操作可以是插入、更新或发送。输出的序列化的格式是特定于每个批处理作业的。")]),e._v(" "),r("p",[e._v("与"),r("code",[e._v("ItemReader")]),e._v("一样,"),r("code",[e._v("ItemWriter")]),e._v("是一个相当通用的接口,如下面的接口定义所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public interface ItemWriter {\n\n void write(List items) throws Exception;\n\n}\n")])])]),r("p",[e._v("与"),r("code",[e._v("read")]),e._v("上的"),r("code",[e._v("ItemReader")]),e._v("一样,"),r("code",[e._v("write")]),e._v("提供了"),r("code",[e._v("ItemWriter")]),e._v("的基本契约。它尝试写出传入的项目列表,只要它是打开的。由于通常期望将项目“批处理”到一个块中,然后输出,因此接口接受一个项目列表,而不是一个项目本身。在写出列表之后,可以在从写方法返回之前执行任何必要的刷新。例如,如果对 Hibernate DAO 进行写操作,则可以对每个项进行多个 write 调用。然后,写入器可以在返回之前调用 Hibernate 会话上的"),r("code",[e._v("flush")]),e._v("。")]),e._v(" "),r("h3",{attrs:{id:"itemstream"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#itemstream"}},[e._v("#")]),e._v(" "),r("code",[e._v("ItemStream")])]),e._v(" "),r("p",[r("code",[e._v("ItemReaders")]),e._v("和"),r("code",[e._v("ItemWriters")]),e._v("都很好地服务于它们各自的目的,但是它们之间有一个共同的关注点,那就是需要另一个接口。通常,作为批处理作业范围的一部分,读取器和编写器需要被打开、关闭,并且需要一种机制来保持状态。"),r("code",[e._v("ItemStream")]),e._v("接口实现了这一目的,如下例所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public interface ItemStream {\n\n void open(ExecutionContext executionContext) throws ItemStreamException;\n\n void update(ExecutionContext executionContext) throws ItemStreamException;\n\n void close() throws ItemStreamException;\n}\n")])])]),r("p",[e._v("在描述每个方法之前,我们应该提到"),r("code",[e._v("ExecutionContext")]),e._v("。如果"),r("code",[e._v("ItemReader")]),e._v("的客户端也实现"),r("code",[e._v("ItemStream")]),e._v(",则在调用"),r("code",[e._v("read")]),e._v("之前,应该调用"),r("code",[e._v("open")]),e._v(",以便打开任何资源,例如文件或获得连接。类似的限制适用于实现"),r("code",[e._v("ItemStream")]),e._v("的"),r("code",[e._v("ItemWriter")]),e._v("。正如在第 2 章中提到的,如果在"),r("code",[e._v("ExecutionContext")]),e._v("中找到了预期的数据,则可以使用它在其初始状态以外的位置启动"),r("code",[e._v("ItemReader")]),e._v("或"),r("code",[e._v("ItemWriter")]),e._v("。相反,调用"),r("code",[e._v("close")]),e._v("是为了确保在打开期间分配的任何资源都被安全地释放。调用"),r("code",[e._v("update")]),e._v("主要是为了确保当前持有的任何状态都被加载到所提供的"),r("code",[e._v("ExecutionContext")]),e._v("中。在提交之前调用此方法,以确保在提交之前将当前状态持久化到数据库中。")]),e._v(" "),r("p",[e._v("在"),r("code",[e._v("ItemStream")]),e._v("的客户端是"),r("code",[e._v("Step")]),e._v("(来自 Spring 批处理核心)的特殊情况下,将为每个分步执行创建一个"),r("code",[e._v("ExecutionContext")]),e._v(",以允许用户存储特定执行的状态,期望在再次启动相同的"),r("code",[e._v("JobInstance")]),e._v("时返回。对于那些熟悉 Quartz 的人,其语义非常类似于 Quartz"),r("code",[e._v("JobDataMap")]),e._v("。")]),e._v(" "),r("h3",{attrs:{id:"委托模式并与步骤一起注册"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#委托模式并与步骤一起注册"}},[e._v("#")]),e._v(" 委托模式并与步骤一起注册")]),e._v(" "),r("p",[e._v("请注意,"),r("code",[e._v("CompositeItemWriter")]),e._v("是委托模式的一个示例,这在 Spring 批处理中很常见。委托本身可能实现回调接口,例如"),r("code",[e._v("StepListener")]),e._v("。如果它们确实存在,并且如果它们是作为"),r("code",[e._v("Job")]),e._v("中的"),r("code",[e._v("Step")]),e._v("的一部分与 Spring 批处理核心一起使用的,那么几乎肯定需要用"),r("code",[e._v("Step")]),e._v("手动注册它们。直接连接到"),r("code",[e._v("Step")]),e._v("的读取器、编写器或处理器如果实现"),r("code",[e._v("ItemStream")]),e._v("或"),r("code",[e._v("StepListener")]),e._v("接口,就会自动注册。但是,由于委托不为"),r("code",[e._v("Step")]),e._v("所知,因此需要将它们作为侦听器或流注入(或者在适当的情况下将两者都注入)。")]),e._v(" "),r("p",[e._v("下面的示例展示了如何将委托作为流注入到 XML 中:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n \n \n \n \n \n\n\n\n \n\n\n\n')])])]),r("p",[e._v("下面的示例展示了如何将委托作为流注入到 XML 中:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic Job ioSampleJob() {\n\treturn this.jobBuilderFactory.get("ioSampleJob")\n\t\t\t\t.start(step1())\n\t\t\t\t.build();\n}\n\n@Bean\npublic Step step1() {\n\treturn this.stepBuilderFactory.get("step1")\n\t\t\t\t.chunk(2)\n\t\t\t\t.reader(fooReader())\n\t\t\t\t.processor(fooProcessor())\n\t\t\t\t.writer(compositeItemWriter())\n\t\t\t\t.stream(barWriter())\n\t\t\t\t.build();\n}\n\n@Bean\npublic CustomCompositeItemWriter compositeItemWriter() {\n\n\tCustomCompositeItemWriter writer = new CustomCompositeItemWriter();\n\n\twriter.setDelegate(barWriter());\n\n\treturn writer;\n}\n\n@Bean\npublic BarWriter barWriter() {\n\treturn new BarWriter();\n}\n')])])]),r("h3",{attrs:{id:"平面文件"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#平面文件"}},[e._v("#")]),e._v(" 平面文件")]),e._v(" "),r("p",[e._v("交换大容量数据的最常见机制之一一直是平面文件。与 XML 不同的是,XML 有一个一致的标准来定义它是如何结构化的(XSD),任何读取平面文件的人都必须提前确切地了解文件是如何结构化的。一般来说,所有的平面文件都分为两种类型:定长和定长。分隔符文件是那些字段被分隔符(如逗号)分隔的文件。固定长度文件的字段是固定长度的。")]),e._v(" "),r("h4",{attrs:{id:"thefieldset"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#thefieldset"}},[e._v("#")]),e._v(" the"),r("code",[e._v("FieldSet")])]),e._v(" "),r("p",[e._v("在处理 Spring 批处理中的平面文件时,无论它是用于输入还是输出,最重要的类之一是"),r("code",[e._v("FieldSet")]),e._v("。许多体系结构和库包含帮助你从文件中读取的抽象,但它们通常返回"),r("code",[e._v("String")]),e._v("或"),r("code",[e._v("String")]),e._v("对象的数组。这真的只会让你走到一半。"),r("code",[e._v("FieldSet")]),e._v("是 Spring 批处理的抽象,用于从文件资源中绑定字段。它允许开发人员以与处理数据库输入大致相同的方式处理文件输入。a"),r("code",[e._v("FieldSet")]),e._v("在概念上类似于 jdbc"),r("code",[e._v("ResultSet")]),e._v("。"),r("code",[e._v("FieldSet")]),e._v("只需要一个参数:一个"),r("code",[e._v("String")]),e._v("令牌数组。还可以选择地配置字段的名称,以便可以按照"),r("code",[e._v("ResultSet")]),e._v("之后的模式通过索引或名称访问字段,如以下示例所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('String[] tokens = new String[]{"foo", "1", "true"};\nFieldSet fs = new DefaultFieldSet(tokens);\nString name = fs.readString(0);\nint value = fs.readInt(1);\nboolean booleanValue = fs.readBoolean(2);\n')])])]),r("p",[e._v("在"),r("code",[e._v("FieldSet")]),e._v("接口上还有许多选项,例如"),r("code",[e._v("Date")]),e._v("、long、"),r("code",[e._v("BigDecimal")]),e._v(",等等。"),r("code",[e._v("FieldSet")]),e._v("的最大优点是它提供了对平面文件输入的一致解析。在处理由格式异常引起的错误或进行简单的数据转换时,它可以是一致的,而不是以潜在的意外方式对每个批处理作业进行不同的解析。")]),e._v(" "),r("h4",{attrs:{id:"flatfileitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#flatfileitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("FlatFileItemReader")])]),e._v(" "),r("p",[e._v("平面文件是最多包含二维(表格)数据的任何类型的文件。 Spring 批处理框架中的平面文件的读取是由一个名为"),r("code",[e._v("FlatFileItemReader")]),e._v("的类提供的,该类为平面文件的读取和解析提供了基本功能。"),r("code",[e._v("FlatFileItemReader")]),e._v("的两个最重要的必需依赖项是"),r("code",[e._v("Resource")]),e._v("和"),r("code",[e._v("LineMapper")]),e._v("。"),r("code",[e._v("LineMapper")]),e._v("接口将在下一节中进行更多的探讨。资源属性表示 Spring 核心"),r("code",[e._v("Resource")]),e._v("。说明如何创建这种类型的 bean 的文档可以在"),r("a",{attrs:{href:"https://docs.spring.io/spring/docs/current/spring-framework-reference/core.html#resources",target:"_blank",rel:"noopener noreferrer"}},[e._v("Spring Framework, Chapter 5. Resources"),r("OutboundLink")],1),e._v("中找到。因此,除了展示下面的简单示例之外,本指南不涉及创建"),r("code",[e._v("Resource")]),e._v("对象的细节:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('Resource resource = new FileSystemResource("resources/trades.csv");\n')])])]),r("p",[e._v("在复杂的批处理环境中,目录结构通常由 Enterprise 应用程序集成基础设施管理,在该基础设施中,外部接口的下拉区被建立,用于将文件从 FTP 位置移动到批处理位置,反之亦然。文件移动实用程序超出了 Spring 批处理体系结构的范围,但是批处理作业流将文件移动实用程序作为步骤包含在作业流中并不少见。批处理架构只需要知道如何定位要处理的文件。 Spring 批处理开始从该起点将数据送入管道的过程。然而,"),r("a",{attrs:{href:"https://projects.spring.io/spring-integration/",target:"_blank",rel:"noopener noreferrer"}},[e._v("Spring Integration"),r("OutboundLink")],1),e._v("提供了许多这类服务。")]),e._v(" "),r("p",[r("code",[e._v("FlatFileItemReader")]),e._v("中的其他属性允许你进一步指定如何解释数据,如下表所示:")]),e._v(" "),r("table",[r("thead",[r("tr",[r("th",[e._v("Property")]),e._v(" "),r("th",[e._v("Type")]),e._v(" "),r("th",[e._v("说明")])])]),e._v(" "),r("tbody",[r("tr",[r("td",[e._v("comments")]),e._v(" "),r("td",[e._v("String[]")]),e._v(" "),r("td",[e._v("指定表示注释行的行前缀。")])]),e._v(" "),r("tr",[r("td",[e._v("encoding")]),e._v(" "),r("td",[e._v("String")]),e._v(" "),r("td",[e._v("指定要使用的文本编码。默认值是"),r("code",[e._v("Charset.defaultCharset()")]),e._v("。")])]),e._v(" "),r("tr",[r("td",[e._v("lineMapper")]),e._v(" "),r("td",[r("code",[e._v("LineMapper")])]),e._v(" "),r("td",[e._v("将表示项的"),r("code",[e._v("String")]),e._v("转换为"),r("code",[e._v("Object")]),e._v("。")])]),e._v(" "),r("tr",[r("td",[e._v("linesToSkip")]),e._v(" "),r("td",[e._v("int")]),e._v(" "),r("td",[e._v("文件顶部要忽略的行数。")])]),e._v(" "),r("tr",[r("td",[e._v("recordSeparatorPolicy")]),e._v(" "),r("td",[e._v("RecordSeparatorPolicy")]),e._v(" "),r("td",[e._v("用于确定行尾的位置"),r("br"),e._v(",并执行类似于在引号字符串中的行尾上继续的操作。")])]),e._v(" "),r("tr",[r("td",[e._v("resource")]),e._v(" "),r("td",[r("code",[e._v("Resource")])]),e._v(" "),r("td",[e._v("可供阅读的资源。")])]),e._v(" "),r("tr",[r("td",[e._v("skippedLinesCallback")]),e._v(" "),r("td",[e._v("LineCallbackHandler")]),e._v(" "),r("td",[e._v("传递"),r("br"),e._v("中要跳过的文件行的原始行内容的接口。如果"),r("code",[e._v("linesToSkip")]),e._v("被设置为 2,那么这个接口被"),r("br"),e._v("调用了两次。")])]),e._v(" "),r("tr",[r("td",[e._v("strict")]),e._v(" "),r("td",[e._v("boolean")]),e._v(" "),r("td",[e._v("在严格模式下,如果输入资源不存在"),r("br"),e._v(",读取器将在"),r("code",[e._v("ExecutionContext")]),e._v("上抛出异常。否则,它会记录问题并继续处理。")])])])]),e._v(" "),r("h5",{attrs:{id:"linemapper"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#linemapper"}},[e._v("#")]),e._v(" "),r("code",[e._v("LineMapper")])]),e._v(" "),r("p",[e._v("与"),r("code",[e._v("RowMapper")]),e._v("一样,它接受一个低层次的构造,例如"),r("code",[e._v("ResultSet")]),e._v("并返回一个"),r("code",[e._v("Object")]),e._v(",平面文件处理需要相同的构造来将"),r("code",[e._v("String")]),e._v("行转换为"),r("code",[e._v("Object")]),e._v(",如以下接口定义所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public interface LineMapper {\n\n T mapLine(String line, int lineNumber) throws Exception;\n\n}\n")])])]),r("p",[e._v("基本的约定是,给定当前行和与其相关联的行号,映射器应该返回一个结果域对象。这类似于"),r("code",[e._v("RowMapper")]),e._v(",因为每一行都与其行号关联,就像"),r("code",[e._v("ResultSet")]),e._v("中的每一行都与其行号关联一样。这允许将行号绑定到结果域对象,以进行身份比较或进行更有信息量的日志记录。然而,与"),r("code",[e._v("RowMapper")]),e._v("不同的是,"),r("code",[e._v("LineMapper")]),e._v("给出的是一条未加工的线,正如上面讨论的那样,这条线只能让你达到一半。该行必须标记为"),r("code",[e._v("FieldSet")]),e._v(",然后可以映射到对象,如本文档后面所述。")]),e._v(" "),r("h5",{attrs:{id:"linetokenizer"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#linetokenizer"}},[e._v("#")]),e._v(" "),r("code",[e._v("LineTokenizer")])]),e._v(" "),r("p",[e._v("将一行输入转换为"),r("code",[e._v("FieldSet")]),e._v("的抽象是必要的,因为可能有许多格式的平面文件数据需要转换为"),r("code",[e._v("FieldSet")]),e._v("。在 Spring 批处理中,这个接口是"),r("code",[e._v("LineTokenizer")]),e._v(":")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public interface LineTokenizer {\n\n FieldSet tokenize(String line);\n\n}\n")])])]),r("p",[e._v("a"),r("code",[e._v("LineTokenizer")]),e._v("的契约是这样的,给定一条输入线(理论上"),r("code",[e._v("String")]),e._v("可以包含多条线),返回一个代表该线的"),r("code",[e._v("FieldSet")]),e._v("。然后可以将这个"),r("code",[e._v("FieldSet")]),e._v("传递给"),r("code",[e._v("FieldSetMapper")]),e._v("。 Spring 批处理包含以下"),r("code",[e._v("LineTokenizer")]),e._v("实现:")]),e._v(" "),r("ul",[r("li",[r("p",[r("code",[e._v("DelimitedLineTokenizer")]),e._v(":用于记录中的字段用分隔符分隔的文件。最常见的分隔符是逗号,但也经常使用管道或分号。")])]),e._v(" "),r("li",[r("p",[r("code",[e._v("FixedLengthTokenizer")]),e._v(":用于记录中的字段都是“固定宽度”的文件。必须为每个记录类型定义每个字段的宽度。")])]),e._v(" "),r("li",[r("p",[r("code",[e._v("PatternMatchingCompositeLineTokenizer")]),e._v(":通过检查模式,确定在特定行上应该使用记号符列表中的哪一个"),r("code",[e._v("LineTokenizer")]),e._v("。")])])]),e._v(" "),r("h5",{attrs:{id:"fieldsetmapper"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#fieldsetmapper"}},[e._v("#")]),e._v(" "),r("code",[e._v("FieldSetMapper")])]),e._v(" "),r("p",[r("code",[e._v("FieldSetMapper")]),e._v("接口定义了一个方法"),r("code",[e._v("mapFieldSet")]),e._v(",它接受一个"),r("code",[e._v("FieldSet")]),e._v("对象并将其内容映射到一个对象。该对象可以是自定义 DTO、域对象或数组,具体取决于作业的需要。"),r("code",[e._v("FieldSetMapper")]),e._v("与"),r("code",[e._v("LineTokenizer")]),e._v("结合使用,以将资源中的一行数据转换为所需类型的对象,如以下接口定义所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public interface FieldSetMapper {\n\n T mapFieldSet(FieldSet fieldSet) throws BindException;\n\n}\n")])])]),r("p",[e._v("使用的模式与"),r("code",[e._v("JdbcTemplate")]),e._v("使用的"),r("code",[e._v("RowMapper")]),e._v("相同。")]),e._v(" "),r("h5",{attrs:{id:"defaultlinemapper"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#defaultlinemapper"}},[e._v("#")]),e._v(" "),r("code",[e._v("DefaultLineMapper")])]),e._v(" "),r("p",[e._v("既然已经定义了在平面文件中读取的基本接口,那么显然需要三个基本步骤:")]),e._v(" "),r("ol",[r("li",[r("p",[e._v("从文件中读出一行。")])]),e._v(" "),r("li",[r("p",[e._v("将"),r("code",[e._v("String")]),e._v("行传递到"),r("code",[e._v("LineTokenizer#tokenize()")]),e._v("方法中,以检索"),r("code",[e._v("FieldSet")]),e._v("。")])]),e._v(" "),r("li",[r("p",[e._v("将从标记化返回的"),r("code",[e._v("FieldSet")]),e._v("传递到"),r("code",[e._v("FieldSetMapper")]),e._v(",从"),r("code",[e._v("ItemReader#read()")]),e._v("方法返回结果。")])])]),e._v(" "),r("p",[e._v("上面描述的两个接口代表两个独立的任务:将一行转换为"),r("code",[e._v("FieldSet")]),e._v(",并将"),r("code",[e._v("FieldSet")]),e._v("映射到域对象。由于"),r("code",[e._v("LineTokenizer")]),e._v("的输入与"),r("code",[e._v("LineMapper")]),e._v("(一行)的输入匹配,并且"),r("code",[e._v("FieldSetMapper")]),e._v("的输出与"),r("code",[e._v("LineMapper")]),e._v("的输出匹配,因此提供了一个同时使用"),r("code",[e._v("LineTokenizer")]),e._v("和"),r("code",[e._v("FieldSetMapper")]),e._v("的默认实现。下面的类定义中显示的"),r("code",[e._v("DefaultLineMapper")]),e._v("表示大多数用户需要的行为:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public class DefaultLineMapper implements LineMapper<>, InitializingBean {\n\n private LineTokenizer tokenizer;\n\n private FieldSetMapper fieldSetMapper;\n\n public T mapLine(String line, int lineNumber) throws Exception {\n return fieldSetMapper.mapFieldSet(tokenizer.tokenize(line));\n }\n\n public void setLineTokenizer(LineTokenizer tokenizer) {\n this.tokenizer = tokenizer;\n }\n\n public void setFieldSetMapper(FieldSetMapper fieldSetMapper) {\n this.fieldSetMapper = fieldSetMapper;\n }\n}\n")])])]),r("p",[e._v("上述功能是在默认实现中提供的,而不是内置在阅读器本身中(就像框架的以前版本中所做的那样),以允许用户在控制解析过程中具有更大的灵活性,尤其是在需要访问原始行的情况下。")]),e._v(" "),r("h5",{attrs:{id:"简单分隔的文件读取示例"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#简单分隔的文件读取示例"}},[e._v("#")]),e._v(" 简单分隔的文件读取示例")]),e._v(" "),r("p",[e._v("下面的示例演示了如何在实际的域场景中读取平面文件。这个特定的批处理作业从以下文件中读取足球运动员:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('ID,lastName,firstName,position,birthYear,debutYear\n"AbduKa00,Abdul-Jabbar,Karim,rb,1974,1996",\n"AbduRa00,Abdullah,Rabih,rb,1975,1999",\n"AberWa00,Abercrombie,Walter,rb,1959,1982",\n"AbraDa00,Abramowicz,Danny,wr,1945,1967",\n"AdamBo00,Adams,Bob,te,1946,1969",\n"AdamCh00,Adams,Charlie,wr,1979,2003"\n')])])]),r("p",[e._v("此文件的内容映射到以下"),r("code",[e._v("Player")]),e._v("域对象:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('public class Player implements Serializable {\n\n private String ID;\n private String lastName;\n private String firstName;\n private String position;\n private int birthYear;\n private int debutYear;\n\n public String toString() {\n return "PLAYER:ID=" + ID + ",Last Name=" + lastName +\n ",First Name=" + firstName + ",Position=" + position +\n ",Birth Year=" + birthYear + ",DebutYear=" +\n debutYear;\n }\n\n // setters and getters...\n}\n')])])]),r("p",[e._v("要将"),r("code",[e._v("FieldSet")]),e._v("映射到"),r("code",[e._v("Player")]),e._v("对象中,需要定义一个返回播放机的"),r("code",[e._v("FieldSetMapper")]),e._v(",如下例所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("protected static class PlayerFieldSetMapper implements FieldSetMapper {\n public Player mapFieldSet(FieldSet fieldSet) {\n Player player = new Player();\n\n player.setID(fieldSet.readString(0));\n player.setLastName(fieldSet.readString(1));\n player.setFirstName(fieldSet.readString(2));\n player.setPosition(fieldSet.readString(3));\n player.setBirthYear(fieldSet.readInt(4));\n player.setDebutYear(fieldSet.readInt(5));\n\n return player;\n }\n}\n")])])]),r("p",[e._v("然后,可以通过正确地构造"),r("code",[e._v("FlatFileItemReader")]),e._v("并调用"),r("code",[e._v("read")]),e._v("来读取文件,如以下示例所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('FlatFileItemReader itemReader = new FlatFileItemReader<>();\nitemReader.setResource(new FileSystemResource("resources/players.csv"));\nDefaultLineMapper lineMapper = new DefaultLineMapper<>();\n//DelimitedLineTokenizer defaults to comma as its delimiter\nlineMapper.setLineTokenizer(new DelimitedLineTokenizer());\nlineMapper.setFieldSetMapper(new PlayerFieldSetMapper());\nitemReader.setLineMapper(lineMapper);\nitemReader.open(new ExecutionContext());\nPlayer player = itemReader.read();\n')])])]),r("p",[e._v("对"),r("code",[e._v("read")]),e._v("的每次调用都会从文件中的每一行返回一个新的"),r("code",[e._v("Player")]),e._v("对象。当到达文件的末尾时,将返回"),r("code",[e._v("null")]),e._v("。")]),e._v(" "),r("h5",{attrs:{id:"按名称映射字段"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#按名称映射字段"}},[e._v("#")]),e._v(" 按名称映射字段")]),e._v(" "),r("p",[e._v("还有一个额外的功能块是"),r("code",[e._v("DelimitedLineTokenizer")]),e._v("和"),r("code",[e._v("FixedLengthTokenizer")]),e._v("都允许的,它在功能上类似于 JDBC"),r("code",[e._v("ResultSet")]),e._v("。字段的名称可以被注入到这些"),r("code",[e._v("LineTokenizer")]),e._v("实现中,以增加映射函数的可读性。首先,将平面文件中所有字段的列名注入到记号生成器中,如下例所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('tokenizer.setNames(new String[] {"ID", "lastName", "firstName", "position", "birthYear", "debutYear"});\n')])])]),r("p",[e._v("a"),r("code",[e._v("FieldSetMapper")]),e._v("可以如下方式使用此信息:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('public class PlayerMapper implements FieldSetMapper {\n public Player mapFieldSet(FieldSet fs) {\n\n if (fs == null) {\n return null;\n }\n\n Player player = new Player();\n player.setID(fs.readString("ID"));\n player.setLastName(fs.readString("lastName"));\n player.setFirstName(fs.readString("firstName"));\n player.setPosition(fs.readString("position"));\n player.setDebutYear(fs.readInt("debutYear"));\n player.setBirthYear(fs.readInt("birthYear"));\n\n return player;\n }\n}\n')])])]),r("h5",{attrs:{id:"向域对象自动设置字段集"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#向域对象自动设置字段集"}},[e._v("#")]),e._v(" 向域对象自动设置字段集")]),e._v(" "),r("p",[e._v("对于许多人来说,必须为"),r("code",[e._v("FieldSetMapper")]),e._v("编写特定的"),r("code",[e._v("RowMapper")]),e._v(",就像为"),r("code",[e._v("JdbcTemplate")]),e._v("编写特定的"),r("code",[e._v("RowMapper")]),e._v("一样麻烦。 Spring 批处理通过提供"),r("code",[e._v("FieldSetMapper")]),e._v("使这一点变得更容易,该批处理通过使用 JavaBean 规范将字段名称与对象上的 setter 匹配来自动映射字段。")]),e._v(" "),r("p",[e._v("再次使用 Football 示例,"),r("code",[e._v("BeanWrapperFieldSetMapper")]),e._v("配置在 XML 中看起来像以下代码片段:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n\n\n\n')])])]),r("p",[e._v("再次使用 Football 示例,"),r("code",[e._v("BeanWrapperFieldSetMapper")]),e._v("配置在 Java 中看起来像以下代码片段:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic FieldSetMapper fieldSetMapper() {\n\tBeanWrapperFieldSetMapper fieldSetMapper = new BeanWrapperFieldSetMapper();\n\n\tfieldSetMapper.setPrototypeBeanName("player");\n\n\treturn fieldSetMapper;\n}\n\n@Bean\n@Scope("prototype")\npublic Player player() {\n\treturn new Player();\n}\n')])])]),r("p",[e._v("对于"),r("code",[e._v("FieldSet")]),e._v("中的每个条目,映射器在"),r("code",[e._v("Player")]),e._v("对象的新实例上查找相应的 setter(由于这个原因,需要原型作用域),就像 Spring 容器查找匹配属性名的 setter 一样。映射"),r("code",[e._v("FieldSet")]),e._v("中的每个可用字段,并返回结果"),r("code",[e._v("Player")]),e._v("对象,不需要任何代码。")]),e._v(" "),r("h5",{attrs:{id:"固定长度文件格式"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#固定长度文件格式"}},[e._v("#")]),e._v(" 固定长度文件格式")]),e._v(" "),r("p",[e._v("到目前为止,只对分隔的文件进行了详细的讨论。然而,它们只代表了文件阅读图片的一半。许多使用平面文件的组织使用固定长度格式。下面是固定长度文件的示例:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("UK21341EAH4121131.11customer1\nUK21341EAH4221232.11customer2\nUK21341EAH4321333.11customer3\nUK21341EAH4421434.11customer4\nUK21341EAH4521535.11customer5\n")])])]),r("p",[e._v("虽然这看起来像是一个很大的域,但它实际上代表了 4 个不同的域:")]),e._v(" "),r("ol",[r("li",[r("p",[e._v("ISIN:所订购商品的唯一标识符-12 个字符长。")])]),e._v(" "),r("li",[r("p",[e._v("数量:订购的商品数量-3 个字符长。")])]),e._v(" "),r("li",[r("p",[e._v("价格:该商品的价格-5 个字符长.")])]),e._v(" "),r("li",[r("p",[e._v("顾客:订购该商品的顾客的 ID-9 个字符长。")])])]),e._v(" "),r("p",[e._v("在配置"),r("code",[e._v("FixedLengthLineTokenizer")]),e._v("时,这些长度中的每一个都必须以范围的形式提供。")]),e._v(" "),r("p",[e._v("下面的示例展示了如何在 XML 中为"),r("code",[e._v("FixedLengthLineTokenizer")]),e._v("定义范围:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n\n')])])]),r("p",[e._v("因为"),r("code",[e._v("FixedLengthLineTokenizer")]),e._v("使用与前面讨论的相同的"),r("code",[e._v("LineTokenizer")]),e._v("接口,所以它返回相同的"),r("code",[e._v("FieldSet")]),e._v(",就像使用了分隔符一样。这允许在处理其输出时使用相同的方法,例如使用"),r("code",[e._v("BeanWrapperFieldSetMapper")]),e._v("。")]),e._v(" "),r("table",[r("thead",[r("tr",[r("th"),e._v(" "),r("th",[e._v("支持前面的范围语法需要在"),r("code",[e._v("ApplicationContext")]),e._v("中配置专门的属性编辑器"),r("code",[e._v("RangeArrayPropertyEditor")]),e._v("。然而,这 Bean "),r("br"),e._v("是在使用批处理名称空间的"),r("code",[e._v("ApplicationContext")]),e._v("中自动声明的。")])])]),e._v(" "),r("tbody")]),e._v(" "),r("p",[e._v("下面的示例展示了如何在 Java 中为"),r("code",[e._v("FixedLengthLineTokenizer")]),e._v("定义范围:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic FixedLengthTokenizer fixedLengthTokenizer() {\n\tFixedLengthTokenizer tokenizer = new FixedLengthTokenizer();\n\n\ttokenizer.setNames("ISIN", "Quantity", "Price", "Customer");\n\ttokenizer.setColumns(new Range(1, 12),\n\t\t\t\t\t\tnew Range(13, 15),\n\t\t\t\t\t\tnew Range(16, 20),\n\t\t\t\t\t\tnew Range(21, 29));\n\n\treturn tokenizer;\n}\n')])])]),r("p",[e._v("因为"),r("code",[e._v("FixedLengthLineTokenizer")]),e._v("使用与上面讨论的相同的"),r("code",[e._v("LineTokenizer")]),e._v("接口,所以它返回相同的"),r("code",[e._v("FieldSet")]),e._v(",就像使用了分隔符一样。这使得在处理其输出时可以使用相同的方法,例如使用"),r("code",[e._v("BeanWrapperFieldSetMapper")]),e._v("。")]),e._v(" "),r("h5",{attrs:{id:"单个文件中的多个记录类型"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#单个文件中的多个记录类型"}},[e._v("#")]),e._v(" 单个文件中的多个记录类型")]),e._v(" "),r("p",[e._v("到目前为止,所有的文件读取示例都为了简单起见做出了一个关键假设:文件中的所有记录都具有相同的格式。然而,情况可能并不总是如此。很常见的一种情况是,一个文件可能具有不同格式的记录,这些记录需要以不同的方式进行标记并映射到不同的对象。下面的文件摘录说明了这一点:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("USER;Smith;Peter;;T;20014539;F\nLINEA;1044391041ABC037.49G201XX1383.12H\nLINEB;2134776319DEF422.99M005LI\n")])])]),r("p",[e._v("在这个文件中,我们有三种类型的记录,“user”、“linea”和“lineb”。“user”行对应于"),r("code",[e._v("User")]),e._v("对象。“linea”和“lineb”都对应于"),r("code",[e._v("Line")]),e._v("对象,尽管“linea”比“lineb”有更多的信息。")]),e._v(" "),r("p",[r("code",[e._v("ItemReader")]),e._v("单独读取每一行,但是我们必须指定不同的"),r("code",[e._v("LineTokenizer")]),e._v("和"),r("code",[e._v("FieldSetMapper")]),e._v("对象,以便"),r("code",[e._v("ItemWriter")]),e._v("接收正确的项。"),r("code",[e._v("PatternMatchingCompositeLineMapper")]),e._v("允许配置模式到"),r("code",[e._v("LineTokenizers")]),e._v("的映射和模式到"),r("code",[e._v("FieldSetMappers")]),e._v("的映射,从而简化了这一过程。")]),e._v(" "),r("p",[e._v("下面的示例展示了如何在 XML 中为"),r("code",[e._v("FixedLengthLineTokenizer")]),e._v("定义范围:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n \n \n \n \n \n \n \n \n \n\n')])])]),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic PatternMatchingCompositeLineMapper orderFileLineMapper() {\n\tPatternMatchingCompositeLineMapper lineMapper =\n\t\tnew PatternMatchingCompositeLineMapper();\n\n\tMap tokenizers = new HashMap<>(3);\n\ttokenizers.put("USER*", userTokenizer());\n\ttokenizers.put("LINEA*", lineATokenizer());\n\ttokenizers.put("LINEB*", lineBTokenizer());\n\n\tlineMapper.setTokenizers(tokenizers);\n\n\tMap mappers = new HashMap<>(2);\n\tmappers.put("USER*", userFieldSetMapper());\n\tmappers.put("LINE*", lineFieldSetMapper());\n\n\tlineMapper.setFieldSetMappers(mappers);\n\n\treturn lineMapper;\n}\n')])])]),r("p",[e._v("在这个示例中,“linea”和“lineb”有单独的"),r("code",[e._v("LineTokenizer")]),e._v("实例,但它们都使用相同的"),r("code",[e._v("FieldSetMapper")]),e._v("。")]),e._v(" "),r("p",[r("code",[e._v("PatternMatchingCompositeLineMapper")]),e._v("使用"),r("code",[e._v("PatternMatcher#match")]),e._v("方法为每一行选择正确的委托。"),r("code",[e._v("PatternMatcher")]),e._v("允许两个具有特殊含义的通配符:问号(“?”)恰好匹配一个字符,而星号(“*”)匹配零个或更多字符。请注意,在前面的配置中,所有模式都以星号结尾,使它们有效地成为行的前缀。无论配置中的顺序如何,"),r("code",[e._v("PatternMatcher")]),e._v("始终匹配最特定的模式。因此,如果“line*”和“linea*”都被列为模式,那么“linea”将匹配模式“linea*”,而“lineb”将匹配模式“line*”。此外,单个星号(“*”)可以通过匹配任何其他模式不匹配的任何行来作为默认设置。")]),e._v(" "),r("p",[e._v("下面的示例展示了如何匹配 XML 中任何其他模式都不匹配的行:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n')])])]),r("p",[e._v("下面的示例展示了如何匹配 Java 中任何其他模式都不匹配的行:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('...\ntokenizers.put("*", defaultLineTokenizer());\n...\n')])])]),r("p",[e._v("还有一个"),r("code",[e._v("PatternMatchingCompositeLineTokenizer")]),e._v("可以单独用于标记化。")]),e._v(" "),r("p",[e._v("平面文件中包含的记录跨越多行也是很常见的。要处理这种情况,需要一种更复杂的策略。在"),r("code",[e._v("multiLineRecords")]),e._v("示例中可以找到这种常见模式的演示。")]),e._v(" "),r("h5",{attrs:{id:"平面文件中的异常处理"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#平面文件中的异常处理"}},[e._v("#")]),e._v(" 平面文件中的异常处理")]),e._v(" "),r("p",[e._v("在许多情况下,对一行进行标记化可能会导致抛出异常。许多平面文件是不完美的,包含格式不正确的记录。许多用户在记录问题、原始行号和行号时选择跳过这些错误行。这些日志稍后可以手动检查,也可以通过另一个批处理作业进行检查。出于这个原因, Spring Batch 为处理解析异常提供了一个异常层次结构:"),r("code",[e._v("FlatFileParseException")]),e._v("和"),r("code",[e._v("FlatFileFormatException")]),e._v("。当试图读取文件时遇到任何错误时,"),r("code",[e._v("FlatFileParseException")]),e._v("将抛出"),r("code",[e._v("FlatFileItemReader")]),e._v("。"),r("code",[e._v("FlatFileFormatException")]),e._v("由"),r("code",[e._v("LineTokenizer")]),e._v("接口的实现抛出,并指示在标记时遇到的更具体的错误。")]),e._v(" "),r("h6",{attrs:{id:"incorrecttokencountexception"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#incorrecttokencountexception"}},[e._v("#")]),e._v(" "),r("code",[e._v("IncorrectTokenCountException")])]),e._v(" "),r("p",[r("code",[e._v("DelimitedLineTokenizer")]),e._v("和"),r("code",[e._v("FixedLengthLineTokenizer")]),e._v("都可以指定可用于创建"),r("code",[e._v("FieldSet")]),e._v("的列名。但是,如果列名的数量与对一行进行标记时发现的列数不匹配,则无法创建"),r("code",[e._v("FieldSet")]),e._v(",并抛出一个"),r("code",[e._v("IncorrectTokenCountException")]),e._v(",其中包含遇到的令牌数量和预期的数量,如以下示例所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('tokenizer.setNames(new String[] {"A", "B", "C", "D"});\n\ntry {\n tokenizer.tokenize("a,b,c");\n}\ncatch (IncorrectTokenCountException e) {\n assertEquals(4, e.getExpectedCount());\n assertEquals(3, e.getActualCount());\n}\n')])])]),r("p",[e._v("因为标记器配置了 4 个列名,但在文件中只找到了 3 个令牌,所以抛出了一个"),r("code",[e._v("IncorrectTokenCountException")]),e._v("。")]),e._v(" "),r("h6",{attrs:{id:"incorrectlinelengthexception"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#incorrectlinelengthexception"}},[e._v("#")]),e._v(" "),r("code",[e._v("IncorrectLineLengthException")])]),e._v(" "),r("p",[e._v("以固定长度格式格式化的文件在解析时有额外的要求,因为与分隔格式不同,每个列必须严格遵守其预定义的宽度。如果行的总长度不等于此列的最大值,则抛出一个异常,如以下示例所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('tokenizer.setColumns(new Range[] { new Range(1, 5),\n new Range(6, 10),\n new Range(11, 15) });\ntry {\n tokenizer.tokenize("12345");\n fail("Expected IncorrectLineLengthException");\n}\ncatch (IncorrectLineLengthException ex) {\n assertEquals(15, ex.getExpectedLength());\n assertEquals(5, ex.getActualLength());\n}\n')])])]),r("p",[e._v("上面的记号生成器的配置范围是:1-5、6-10 和 11-1 5.因此,这条线的总长度是 1 5.但是,在前面的示例中,传入了长度为 5 的行,从而引发了"),r("code",[e._v("IncorrectLineLengthException")]),e._v("。在此抛出一个异常,而不是仅映射第一列,这样可以使行的处理更早失败,并且所包含的信息比在试图在"),r("code",[e._v("FieldSetMapper")]),e._v("中读取第 2 列时失败时所包含的信息更多。然而,在某些情况下,直线的长度并不总是恒定的。因此,可以通过“严格”属性关闭对行长的验证,如下例所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('tokenizer.setColumns(new Range[] { new Range(1, 5), new Range(6, 10) });\ntokenizer.setStrict(false);\nFieldSet tokens = tokenizer.tokenize("12345");\nassertEquals("12345", tokens.readString(0));\nassertEquals("", tokens.readString(1));\n')])])]),r("p",[e._v("前面的示例与前面的示例几乎相同,只是调用了"),r("code",[e._v("tokenizer.setStrict(false)")]),e._v("。这个设置告诉标记器在标记行时不要强制行长。现在正确地创建并返回了"),r("code",[e._v("FieldSet")]),e._v("。但是,对于其余的值,它只包含空标记。")]),e._v(" "),r("h4",{attrs:{id:"flatfileitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#flatfileitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("FlatFileItemWriter")])]),e._v(" "),r("p",[e._v("写入平面文件也存在从文件读入时必须克服的问题。一个步骤必须能够以事务性的方式编写分隔格式或固定长度格式。")]),e._v(" "),r("h5",{attrs:{id:"lineaggregator"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#lineaggregator"}},[e._v("#")]),e._v(" "),r("code",[e._v("LineAggregator")])]),e._v(" "),r("p",[e._v("正如"),r("code",[e._v("LineTokenizer")]),e._v("接口是获取一个项并将其转换为"),r("code",[e._v("String")]),e._v("所必需的一样,文件写入必须有一种方法,可以将多个字段聚合到一个字符串中,以便将其写入文件。在 Spring 批处理中,这是"),r("code",[e._v("LineAggregator")]),e._v(",如下面的接口定义所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public interface LineAggregator {\n\n public String aggregate(T item);\n\n}\n")])])]),r("p",[r("code",[e._v("LineAggregator")]),e._v("是"),r("code",[e._v("LineTokenizer")]),e._v("的逻辑对立面。"),r("code",[e._v("LineTokenizer")]),e._v("接受一个"),r("code",[e._v("String")]),e._v("并返回一个"),r("code",[e._v("FieldSet")]),e._v(",而"),r("code",[e._v("LineAggregator")]),e._v("接受一个"),r("code",[e._v("item")]),e._v("并返回一个"),r("code",[e._v("String")]),e._v("。")]),e._v(" "),r("h6",{attrs:{id:"passthroughlineaggregator"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#passthroughlineaggregator"}},[e._v("#")]),e._v(" "),r("code",[e._v("PassThroughLineAggregator")])]),e._v(" "),r("p",[r("code",[e._v("LineAggregator")]),e._v("接口的最基本的实现是"),r("code",[e._v("PassThroughLineAggregator")]),e._v(",它假定对象已经是一个字符串,或者它的字符串表示可以用于编写,如下面的代码所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public class PassThroughLineAggregator implements LineAggregator {\n\n public String aggregate(T item) {\n return item.toString();\n }\n}\n")])])]),r("p",[e._v("如果需要直接控制创建字符串,那么前面的实现是有用的,但是"),r("code",[e._v("FlatFileItemWriter")]),e._v("的优点,例如事务和重新启动支持,是必要的。")]),e._v(" "),r("h5",{attrs:{id:"简化文件编写示例"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#简化文件编写示例"}},[e._v("#")]),e._v(" 简化文件编写示例")]),e._v(" "),r("p",[e._v("既然"),r("code",[e._v("LineAggregator")]),e._v("接口及其最基本的实现"),r("code",[e._v("PassThroughLineAggregator")]),e._v("已经定义好了,那么编写的基本流程就可以解释了:")]),e._v(" "),r("ol",[r("li",[r("p",[e._v("要写入的对象被传递给"),r("code",[e._v("LineAggregator")]),e._v(",以获得"),r("code",[e._v("String")]),e._v("。")])]),e._v(" "),r("li",[r("p",[e._v("返回的"),r("code",[e._v("String")]),e._v("被写入配置的文件。")])])]),e._v(" "),r("p",[e._v("下面摘自"),r("code",[e._v("FlatFileItemWriter")]),e._v("的代码表达了这一点:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public void write(T item) throws Exception {\n write(lineAggregator.aggregate(item) + LINE_SEPARATOR);\n}\n")])])]),r("p",[e._v("在 XML 中,配置的一个简单示例可能如下所示:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n\n')])])]),r("p",[e._v("在 Java 中,配置的一个简单示例可能如下所示:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic FlatFileItemWriter itemWriter() {\n\treturn new FlatFileItemWriterBuilder()\n \t\t\t.name("itemWriter")\n \t\t\t.resource(new FileSystemResource("target/test-outputs/output.txt"))\n \t\t\t.lineAggregator(new PassThroughLineAggregator<>())\n \t\t\t.build();\n}\n')])])]),r("h5",{attrs:{id:"fieldextractor"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#fieldextractor"}},[e._v("#")]),e._v(" "),r("code",[e._v("FieldExtractor")])]),e._v(" "),r("p",[e._v("前面的示例对于对文件的写入的最基本使用可能是有用的。然而,"),r("code",[e._v("FlatFileItemWriter")]),e._v("的大多数用户都有一个需要写出的域对象,因此必须将其转换为一行。在文件阅读中,需要进行以下操作:")]),e._v(" "),r("ol",[r("li",[r("p",[e._v("从文件中读出一行。")])]),e._v(" "),r("li",[r("p",[e._v("将该行传递到"),r("code",[e._v("LineTokenizer#tokenize()")]),e._v("方法中,以便检索"),r("code",[e._v("FieldSet")]),e._v("。")])]),e._v(" "),r("li",[r("p",[e._v("将从标记化返回的"),r("code",[e._v("FieldSet")]),e._v("传递到"),r("code",[e._v("FieldSetMapper")]),e._v(",从"),r("code",[e._v("ItemReader#read()")]),e._v("方法返回结果。")])])]),e._v(" "),r("p",[e._v("编写文件也有类似但相反的步骤:")]),e._v(" "),r("ol",[r("li",[r("p",[e._v("把要写的东西交给作者。")])]),e._v(" "),r("li",[r("p",[e._v("将项目上的字段转换为数组。")])]),e._v(" "),r("li",[r("p",[e._v("将生成的数组聚合为一条线。")])])]),e._v(" "),r("p",[e._v("因为框架无法知道需要从对象中写出哪些字段,所以必须编写"),r("code",[e._v("FieldExtractor")]),e._v("才能完成将项转换为数组的任务,如下面的接口定义所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public interface FieldExtractor {\n\n Object[] extract(T item);\n\n}\n")])])]),r("p",[r("code",[e._v("FieldExtractor")]),e._v("接口的实现应该从提供的对象的字段创建一个数组,然后可以在元素之间使用分隔符写出该数组,或者作为固定宽度线的一部分。")]),e._v(" "),r("h6",{attrs:{id:"passthroughfieldextractor"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#passthroughfieldextractor"}},[e._v("#")]),e._v(" "),r("code",[e._v("PassThroughFieldExtractor")])]),e._v(" "),r("p",[e._v("在许多情况下,需要写出集合,例如一个数组,"),r("code",[e._v("Collection")]),e._v("或"),r("code",[e._v("FieldSet")]),e._v("。从这些集合类型中的一种“提取”一个数组是非常简单的。要做到这一点,将集合转换为一个数组。因此,在此场景中应该使用"),r("code",[e._v("PassThroughFieldExtractor")]),e._v("。应该注意的是,如果传入的对象不是集合的类型,那么"),r("code",[e._v("PassThroughFieldExtractor")]),e._v("将返回一个仅包含要提取的项的数组。")]),e._v(" "),r("h6",{attrs:{id:"beanwrapperfieldextractor"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#beanwrapperfieldextractor"}},[e._v("#")]),e._v(" "),r("code",[e._v("BeanWrapperFieldExtractor")])]),e._v(" "),r("p",[e._v("与文件读取部分中描述的"),r("code",[e._v("BeanWrapperFieldSetMapper")]),e._v("一样,通常更好的方法是配置如何将域对象转换为对象数组,而不是自己编写转换。"),r("code",[e._v("BeanWrapperFieldExtractor")]),e._v("提供了这种功能,如以下示例所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('BeanWrapperFieldExtractor extractor = new BeanWrapperFieldExtractor<>();\nextractor.setNames(new String[] { "first", "last", "born" });\n\nString first = "Alan";\nString last = "Turing";\nint born = 1912;\n\nName n = new Name(first, last, born);\nObject[] values = extractor.extract(n);\n\nassertEquals(first, values[0]);\nassertEquals(last, values[1]);\nassertEquals(born, values[2]);\n')])])]),r("p",[e._v("这个提取器实现只有一个必需的属性:要映射的字段的名称。正如"),r("code",[e._v("BeanWrapperFieldSetMapper")]),e._v("需要字段名称来将"),r("code",[e._v("FieldSet")]),e._v("上的字段映射到所提供对象上的 setter 一样,"),r("code",[e._v("BeanWrapperFieldExtractor")]),e._v("也需要名称来映射到 getter 以创建对象数组。值得注意的是,名称的顺序决定了数组中字段的顺序。")]),e._v(" "),r("h5",{attrs:{id:"分隔的文件编写示例"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#分隔的文件编写示例"}},[e._v("#")]),e._v(" 分隔的文件编写示例")]),e._v(" "),r("p",[e._v("最基本的平面文件格式是一种所有字段都用分隔符分隔的格式。这可以使用"),r("code",[e._v("DelimitedLineAggregator")]),e._v("来完成。下面的示例写出了一个简单的域对象,该对象表示对客户帐户的信用:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public class CustomerCredit {\n\n private int id;\n private String name;\n private BigDecimal credit;\n\n //getters and setters removed for clarity\n}\n")])])]),r("p",[e._v("由于正在使用域对象,因此必须提供"),r("code",[e._v("FieldExtractor")]),e._v("接口的实现以及要使用的分隔符。")]),e._v(" "),r("p",[e._v("下面的示例展示了如何在 XML 中使用带有分隔符的"),r("code",[e._v("FieldExtractor")]),e._v(":")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n \n \n \n \n \n \n \n\n')])])]),r("p",[e._v("下面的示例展示了如何在 Java 中使用带有分隔符的"),r("code",[e._v("FieldExtractor")]),e._v(":")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic FlatFileItemWriter itemWriter(Resource outputResource) throws Exception {\n\tBeanWrapperFieldExtractor fieldExtractor = new BeanWrapperFieldExtractor<>();\n\tfieldExtractor.setNames(new String[] {"name", "credit"});\n\tfieldExtractor.afterPropertiesSet();\n\n\tDelimitedLineAggregator lineAggregator = new DelimitedLineAggregator<>();\n\tlineAggregator.setDelimiter(",");\n\tlineAggregator.setFieldExtractor(fieldExtractor);\n\n\treturn new FlatFileItemWriterBuilder()\n\t\t\t\t.name("customerCreditWriter")\n\t\t\t\t.resource(outputResource)\n\t\t\t\t.lineAggregator(lineAggregator)\n\t\t\t\t.build();\n}\n')])])]),r("p",[e._v("在前面的示例中,本章前面描述的"),r("code",[e._v("BeanWrapperFieldExtractor")]),e._v("用于将"),r("code",[e._v("CustomerCredit")]),e._v("中的名称和信用字段转换为一个对象数组,然后在每个字段之间使用逗号写出该对象数组。")]),e._v(" "),r("p",[e._v("也可以使用"),r("code",[e._v("FlatFileItemWriterBuilder.DelimitedBuilder")]),e._v("自动创建"),r("code",[e._v("BeanWrapperFieldExtractor")]),e._v("和"),r("code",[e._v("DelimitedLineAggregator")]),e._v(",如以下示例所示:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic FlatFileItemWriter itemWriter(Resource outputResource) throws Exception {\n\treturn new FlatFileItemWriterBuilder()\n\t\t\t\t.name("customerCreditWriter")\n\t\t\t\t.resource(outputResource)\n\t\t\t\t.delimited()\n\t\t\t\t.delimiter("|")\n\t\t\t\t.names(new String[] {"name", "credit"})\n\t\t\t\t.build();\n}\n')])])]),r("h5",{attrs:{id:"固定宽度文件编写示例"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#固定宽度文件编写示例"}},[e._v("#")]),e._v(" 固定宽度文件编写示例")]),e._v(" "),r("p",[e._v("分隔符并不是唯一一种平面文件格式。许多人更喜欢为每个列使用一个设置的宽度来划分字段,这通常称为“固定宽度”。 Spring 批处理在用"),r("code",[e._v("FormatterLineAggregator")]),e._v("写文件时支持这一点。")]),e._v(" "),r("p",[e._v("使用上述相同的"),r("code",[e._v("CustomerCredit")]),e._v("域对象,可以在 XML 中进行如下配置:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n \n \n \n \n \n \n \n\n')])])]),r("p",[e._v("使用上面描述的相同的"),r("code",[e._v("CustomerCredit")]),e._v("域对象,可以在 Java 中进行如下配置:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic FlatFileItemWriter itemWriter(Resource outputResource) throws Exception {\n\tBeanWrapperFieldExtractor fieldExtractor = new BeanWrapperFieldExtractor<>();\n\tfieldExtractor.setNames(new String[] {"name", "credit"});\n\tfieldExtractor.afterPropertiesSet();\n\n\tFormatterLineAggregator lineAggregator = new FormatterLineAggregator<>();\n\tlineAggregator.setFormat("%-9s%-2.0f");\n\tlineAggregator.setFieldExtractor(fieldExtractor);\n\n\treturn new FlatFileItemWriterBuilder()\n\t\t\t\t.name("customerCreditWriter")\n\t\t\t\t.resource(outputResource)\n\t\t\t\t.lineAggregator(lineAggregator)\n\t\t\t\t.build();\n}\n')])])]),r("p",[e._v("前面的大多数示例看起来应该很熟悉。但是,格式属性的值是新的。")]),e._v(" "),r("p",[e._v("下面的示例显示了 XML 中的格式属性:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n')])])]),r("p",[e._v("下面的示例显示了 Java 中的 format 属性:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('...\nFormatterLineAggregator lineAggregator = new FormatterLineAggregator<>();\nlineAggregator.setFormat("%-9s%-2.0f");\n...\n')])])]),r("p",[e._v("底层实现是使用作为 Java5 的一部分添加的相同的"),r("code",[e._v("Formatter")]),e._v("构建的。Java"),r("code",[e._v("Formatter")]),e._v("基于 C 编程语言的"),r("code",[e._v("printf")]),e._v("功能。关于如何配置格式化程序的大多数详细信息可以在"),r("a",{attrs:{href:"https://docs.oracle.com/javase/8/docs/api/java/util/Formatter.html",target:"_blank",rel:"noopener noreferrer"}},[e._v("Formatter"),r("OutboundLink")],1),e._v("的 Javadoc 中找到。")]),e._v(" "),r("p",[e._v("也可以使用"),r("code",[e._v("FlatFileItemWriterBuilder.FormattedBuilder")]),e._v("自动创建"),r("code",[e._v("BeanWrapperFieldExtractor")]),e._v("和"),r("code",[e._v("FormatterLineAggregator")]),e._v(",如以下示例所示:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic FlatFileItemWriter itemWriter(Resource outputResource) throws Exception {\n\treturn new FlatFileItemWriterBuilder()\n\t\t\t\t.name("customerCreditWriter")\n\t\t\t\t.resource(outputResource)\n\t\t\t\t.formatted()\n\t\t\t\t.format("%-9s%-2.0f")\n\t\t\t\t.names(new String[] {"name", "credit"})\n\t\t\t\t.build();\n}\n')])])]),r("h5",{attrs:{id:"处理文件创建"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#处理文件创建"}},[e._v("#")]),e._v(" 处理文件创建")]),e._v(" "),r("p",[r("code",[e._v("FlatFileItemReader")]),e._v("与文件资源的关系非常简单。当读取器被初始化时,它会打开该文件(如果它存在的话),如果它不存在,则会抛出一个异常。写文件并不是那么简单。乍一看,对于"),r("code",[e._v("FlatFileItemWriter")]),e._v("似乎应该存在类似的直接契约:如果文件已经存在,则抛出一个异常,如果不存在,则创建它并开始写入。然而,重新启动"),r("code",[e._v("Job")]),e._v("可能会导致问题。在正常的重启场景中,契约是相反的:如果文件存在,则从最后一个已知的良好位置开始向它写入,如果不存在,则抛出一个异常。但是,如果此作业的文件名总是相同,会发生什么情况?在这种情况下,如果文件存在,你可能想要删除它,除非是重新启动。由于这种可能性,"),r("code",[e._v("FlatFileItemWriter")]),e._v("包含属性"),r("code",[e._v("shouldDeleteIfExists")]),e._v("。将此属性设置为 true 将导致在打开 Writer 时删除同名的现有文件。")]),e._v(" "),r("h3",{attrs:{id:"xml-项读取器和编写器"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#xml-项读取器和编写器"}},[e._v("#")]),e._v(" XML 项读取器和编写器")]),e._v(" "),r("p",[e._v("Spring Batch 提供了用于读取 XML 记录并将它们映射到 Java 对象以及将 Java 对象写为 XML 记录的事务基础设施。")]),e._v(" "),r("table",[r("thead",[r("tr",[r("th"),e._v(" "),r("th",[e._v("流 XML 上的约束"),r("br"),r("br"),e._v("STAX API 用于 I/O,因为其他标准的 XML 解析 API 不符合批处理"),r("br"),e._v("的要求(DOM 一次将整个输入加载到内存中,SAX 通过允许用户仅提供回调来控制"),r("br"),e._v("解析过程)。")])])]),e._v(" "),r("tbody")]),e._v(" "),r("p",[e._v("我们需要考虑 XML 输入和输出如何在 Spring 批处理中工作。首先,有几个概念与文件读写不同,但在 Spring 批 XML 处理中很常见。使用 XML 处理,不是需要标记的记录行("),r("code",[e._v("FieldSet")]),e._v("实例),而是假设 XML 资源是与单个记录相对应的“片段”的集合,如下图所示:")]),e._v(" "),r("p",[r("img",{attrs:{src:"https://docs.spring.io/spring-batch/docs/current/reference/html/images/xmlinput.png",alt:"XML Input"}})]),e._v(" "),r("p",[e._v("图 1.XML 输入")]),e._v(" "),r("p",[e._v("在上面的场景中,“trade”标记被定义为“root 元素”。“”和“”之间的所有内容都被视为一个“片段”。 Spring 批处理使用对象/XML 映射(OXM)将片段绑定到对象。然而, Spring 批处理并不绑定到任何特定的 XML 绑定技术。典型的用途是委托给"),r("a",{attrs:{href:"https://docs.spring.io/spring/docs/current/spring-framework-reference/data-access.html#oxm",target:"_blank",rel:"noopener noreferrer"}},[e._v("Spring OXM"),r("OutboundLink")],1),e._v(",这为最流行的 OXM 技术提供了统一的抽象。对 Spring OXM 的依赖是可选的,如果需要,可以选择实现 Spring 批处理特定接口。与 OXM 支持的技术之间的关系如下图所示:")]),e._v(" "),r("p",[r("img",{attrs:{src:"https://docs.spring.io/spring-batch/docs/current/reference/html/images/oxm-fragments.png",alt:"OXM 绑定"}})]),e._v(" "),r("p",[e._v("图 2.OXM 绑定")]),e._v(" "),r("p",[e._v("通过介绍 OXM 以及如何使用 XML 片段来表示记录,我们现在可以更仔细地研究阅读器和编写器。")]),e._v(" "),r("h4",{attrs:{id:"staxeventitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#staxeventitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("StaxEventItemReader")])]),e._v(" "),r("p",[r("code",[e._v("StaxEventItemReader")]),e._v("配置为处理来自 XML 输入流的记录提供了一个典型的设置。首先,考虑"),r("code",[e._v("StaxEventItemReader")]),e._v("可以处理的以下一组 XML 记录:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n\n \n XYZ0001\n 5\n 11.39\n Customer1\n \n \n XYZ0002\n 2\n 72.99\n Customer2c\n \n \n XYZ0003\n 9\n 99.99\n Customer3\n \n\n')])])]),r("p",[e._v("为了能够处理 XML 记录,需要具备以下条件:")]),e._v(" "),r("ul",[r("li",[r("p",[e._v("根元素名称:构成要映射的对象的片段的根元素的名称。示例配置用“交易价值”演示了这一点。")])]),e._v(" "),r("li",[r("p",[e._v("资源:表示要读取的文件的 Spring 资源。")])]),e._v(" "),r("li",[r("p",[r("code",[e._v("Unmarshaller")]),e._v(": Spring OXM 提供的一种解组功能,用于将 XML 片段映射到对象。")])])]),e._v(" "),r("p",[e._v("下面的示例展示了如何定义一个"),r("code",[e._v("StaxEventItemReader")]),e._v(",它与一个名为"),r("code",[e._v("trade")]),e._v("的根元素、一个资源"),r("code",[e._v("data/iosample/input/input.xml")]),e._v("和一个在 XML 中名为"),r("code",[e._v("tradeMarshaller")]),e._v("的解组器一起工作:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n\n')])])]),r("p",[e._v("下面的示例展示了如何定义一个"),r("code",[e._v("StaxEventItemReader")]),e._v(",它与一个名为"),r("code",[e._v("trade")]),e._v("的根元素、一个资源"),r("code",[e._v("data/iosample/input/input.xml")]),e._v("和一个在 Java 中名为"),r("code",[e._v("tradeMarshaller")]),e._v("的解组器一起工作:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic StaxEventItemReader itemReader() {\n\treturn new StaxEventItemReaderBuilder()\n\t\t\t.name("itemReader")\n\t\t\t.resource(new FileSystemResource("org/springframework/batch/item/xml/domain/trades.xml"))\n\t\t\t.addFragmentRootElements("trade")\n\t\t\t.unmarshaller(tradeMarshaller())\n\t\t\t.build();\n\n}\n')])])]),r("p",[e._v("请注意,在本例中,我们选择使用"),r("code",[e._v("XStreamMarshaller")]),e._v(",它接受作为映射传入的别名,第一个键和值是片段的名称(即根元素)和要绑定的对象类型。然后,类似于"),r("code",[e._v("FieldSet")]),e._v(",映射到对象类型中的字段的其他元素的名称在映射中被描述为键/值对。在配置文件中,我们可以使用 Spring 配置实用程序来描述所需的别名。")]),e._v(" "),r("p",[e._v("下面的示例展示了如何用 XML 描述别名:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n \n \n \n \n \n\n')])])]),r("p",[e._v("下面的示例展示了如何在 Java 中描述别名:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic XStreamMarshaller tradeMarshaller() {\n\tMap aliases = new HashMap<>();\n\taliases.put("trade", Trade.class);\n\taliases.put("price", BigDecimal.class);\n\taliases.put("isin", String.class);\n\taliases.put("customer", String.class);\n\taliases.put("quantity", Long.class);\n\n\tXStreamMarshaller marshaller = new XStreamMarshaller();\n\n\tmarshaller.setAliases(aliases);\n\n\treturn marshaller;\n}\n')])])]),r("p",[e._v("在输入时,读取器读取 XML 资源,直到它识别出一个新的片段即将开始。默认情况下,读取器匹配元素名,以识别一个新片段即将开始。阅读器从片段中创建一个独立的 XML 文档,并将该文档传递给一个反序列化器(通常是围绕 Spring OXM"),r("code",[e._v("Unmarshaller")]),e._v("的包装器),以将 XML 映射到一个 Java 对象。")]),e._v(" "),r("p",[e._v("总之,这个过程类似于下面的 Java 代码,它使用由 Spring 配置提供的注入:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('StaxEventItemReader xmlStaxEventItemReader = new StaxEventItemReader<>();\nResource resource = new ByteArrayResource(xmlResource.getBytes());\n\nMap aliases = new HashMap();\naliases.put("trade","org.springframework.batch.sample.domain.trade.Trade");\naliases.put("price","java.math.BigDecimal");\naliases.put("customer","java.lang.String");\naliases.put("isin","java.lang.String");\naliases.put("quantity","java.lang.Long");\nXStreamMarshaller unmarshaller = new XStreamMarshaller();\nunmarshaller.setAliases(aliases);\nxmlStaxEventItemReader.setUnmarshaller(unmarshaller);\nxmlStaxEventItemReader.setResource(resource);\nxmlStaxEventItemReader.setFragmentRootElementName("trade");\nxmlStaxEventItemReader.open(new ExecutionContext());\n\nboolean hasNext = true;\n\nTrade trade = null;\n\nwhile (hasNext) {\n trade = xmlStaxEventItemReader.read();\n if (trade == null) {\n hasNext = false;\n }\n else {\n System.out.println(trade);\n }\n}\n')])])]),r("h4",{attrs:{id:"staxeventitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#staxeventitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("StaxEventItemWriter")])]),e._v(" "),r("p",[e._v("输出与输入对称地工作。"),r("code",[e._v("StaxEventItemWriter")]),e._v("需要一个"),r("code",[e._v("Resource")]),e._v("、一个编组器和一个"),r("code",[e._v("rootTagName")]),e._v("。将 Java 对象传递给编组器(通常是标准的 Spring OXM 编组器),该编组器通过使用自定义事件编写器将 OXM 工具为每个片段产生的"),r("code",[e._v("StartDocument")]),e._v("和"),r("code",[e._v("EndDocument")]),e._v("事件进行过滤,从而将其写到"),r("code",[e._v("Resource")]),e._v("。")]),e._v(" "),r("p",[e._v("下面的 XML 示例使用"),r("code",[e._v("MarshallingEventWriterSerializer")]),e._v(":")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n\n')])])]),r("p",[e._v("下面的 Java 示例使用"),r("code",[e._v("MarshallingEventWriterSerializer")]),e._v(":")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic StaxEventItemWriter itemWriter(Resource outputResource) {\n\treturn new StaxEventItemWriterBuilder()\n\t\t\t.name("tradesWriter")\n\t\t\t.marshaller(tradeMarshaller())\n\t\t\t.resource(outputResource)\n\t\t\t.rootTagName("trade")\n\t\t\t.overwriteOutput(true)\n\t\t\t.build();\n\n}\n')])])]),r("p",[e._v("前面的配置设置了三个必需的属性,并设置了可选的"),r("code",[e._v("overwriteOutput=true")]),e._v("attrbute,这在本章前面提到过,用于指定现有文件是否可以重写。")]),e._v(" "),r("p",[e._v("下面的 XML 示例使用了与本章前面所示的阅读示例中使用的相同的编组器:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n \n \n \n \n \n\n')])])]),r("p",[e._v("下面的 Java 示例使用了与本章前面所示的阅读示例中使用的收集器相同的收集器:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic XStreamMarshaller customerCreditMarshaller() {\n\tXStreamMarshaller marshaller = new XStreamMarshaller();\n\n\tMap aliases = new HashMap<>();\n\taliases.put("trade", Trade.class);\n\taliases.put("price", BigDecimal.class);\n\taliases.put("isin", String.class);\n\taliases.put("customer", String.class);\n\taliases.put("quantity", Long.class);\n\n\tmarshaller.setAliases(aliases);\n\n\treturn marshaller;\n}\n')])])]),r("p",[e._v("作为 Java 示例的总结,下面的代码演示了讨论的所有要点,并演示了所需属性的编程设置:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('FileSystemResource resource = new FileSystemResource("data/outputFile.xml")\n\nMap aliases = new HashMap();\naliases.put("trade","org.springframework.batch.sample.domain.trade.Trade");\naliases.put("price","java.math.BigDecimal");\naliases.put("customer","java.lang.String");\naliases.put("isin","java.lang.String");\naliases.put("quantity","java.lang.Long");\nMarshaller marshaller = new XStreamMarshaller();\nmarshaller.setAliases(aliases);\n\nStaxEventItemWriter staxItemWriter =\n\tnew StaxEventItemWriterBuilder()\n\t\t\t\t.name("tradesWriter")\n\t\t\t\t.marshaller(marshaller)\n\t\t\t\t.resource(resource)\n\t\t\t\t.rootTagName("trade")\n\t\t\t\t.overwriteOutput(true)\n\t\t\t\t.build();\n\nstaxItemWriter.afterPropertiesSet();\n\nExecutionContext executionContext = new ExecutionContext();\nstaxItemWriter.open(executionContext);\nTrade trade = new Trade();\ntrade.setPrice(11.39);\ntrade.setIsin("XYZ0001");\ntrade.setQuantity(5L);\ntrade.setCustomer("Customer1");\nstaxItemWriter.write(trade);\n')])])]),r("h3",{attrs:{id:"json-条目阅读器和编写器"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#json-条目阅读器和编写器"}},[e._v("#")]),e._v(" JSON 条目阅读器和编写器")]),e._v(" "),r("p",[e._v("Spring Batch 以以下格式提供对读取和写入 JSON 资源的支持:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('[\n {\n "isin": "123",\n "quantity": 1,\n "price": 1.2,\n "customer": "foo"\n },\n {\n "isin": "456",\n "quantity": 2,\n "price": 1.4,\n "customer": "bar"\n }\n]\n')])])]),r("p",[e._v("假定 JSON 资源是与单个项对应的 JSON 对象数组。 Spring 批处理不绑定到任何特定的 JSON 库。")]),e._v(" "),r("h4",{attrs:{id:"jsonitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#jsonitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("JsonItemReader")])]),e._v(" "),r("p",[r("code",[e._v("JsonItemReader")]),e._v("将 JSON 解析和绑定委托给"),r("code",[e._v("org.springframework.batch.item.json.JsonObjectReader")]),e._v("接口的实现。该接口旨在通过使用流 API 以块形式读取 JSON 对象来实现。目前提供了两种实现方式:")]),e._v(" "),r("ul",[r("li",[r("p",[r("a",{attrs:{href:"https://github.com/FasterXML/jackson",target:"_blank",rel:"noopener noreferrer"}},[e._v("Jackson"),r("OutboundLink")],1),e._v("通过"),r("code",[e._v("org.springframework.batch.item.json.JacksonJsonObjectReader")])])]),e._v(" "),r("li",[r("p",[r("a",{attrs:{href:"https://github.com/google/gson",target:"_blank",rel:"noopener noreferrer"}},[e._v("Gson"),r("OutboundLink")],1),e._v("通过"),r("code",[e._v("org.springframework.batch.item.json.GsonJsonObjectReader")])])])]),e._v(" "),r("p",[e._v("要能够处理 JSON 记录,需要具备以下条件:")]),e._v(" "),r("ul",[r("li",[r("p",[r("code",[e._v("Resource")]),e._v(":表示要读取的 JSON 文件的 Spring 资源。")])]),e._v(" "),r("li",[r("p",[r("code",[e._v("JsonObjectReader")]),e._v(":用于解析并将 JSON 对象绑定到项的 JSON 对象阅读器")])])]),e._v(" "),r("p",[e._v("下面的示例展示了如何基于 Jackson 定义一个"),r("code",[e._v("JsonItemReader")]),e._v("并与前面的 JSON 资源"),r("code",[e._v("org/springframework/batch/item/json/trades.json")]),e._v("一起工作的"),r("code",[e._v("JsonObjectReader")]),e._v(":")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic JsonItemReader jsonItemReader() {\n return new JsonItemReaderBuilder()\n .jsonObjectReader(new JacksonJsonObjectReader<>(Trade.class))\n .resource(new ClassPathResource("trades.json"))\n .name("tradeJsonItemReader")\n .build();\n}\n')])])]),r("h4",{attrs:{id:"jsonfileitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#jsonfileitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("JsonFileItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("JsonFileItemWriter")]),e._v("将项的编组委托给"),r("code",[e._v("org.springframework.batch.item.json.JsonObjectMarshaller")]),e._v("接口。这个接口的契约是将一个对象带到一个 JSON"),r("code",[e._v("String")]),e._v("。目前提供了两种实现方式:")]),e._v(" "),r("ul",[r("li",[r("p",[r("a",{attrs:{href:"https://github.com/FasterXML/jackson",target:"_blank",rel:"noopener noreferrer"}},[e._v("Jackson"),r("OutboundLink")],1),e._v("通过"),r("code",[e._v("org.springframework.batch.item.json.JacksonJsonObjectMarshaller")])])]),e._v(" "),r("li",[r("p",[r("a",{attrs:{href:"https://github.com/google/gson",target:"_blank",rel:"noopener noreferrer"}},[e._v("Gson"),r("OutboundLink")],1),e._v("通过"),r("code",[e._v("org.springframework.batch.item.json.GsonJsonObjectMarshaller")])])])]),e._v(" "),r("p",[e._v("为了能够编写 JSON 记录,需要具备以下条件:")]),e._v(" "),r("ul",[r("li",[r("p",[r("code",[e._v("Resource")]),e._v(":表示要写入的 JSON 文件的一个 Spring "),r("code",[e._v("Resource")])])]),e._v(" "),r("li",[r("p",[r("code",[e._v("JsonObjectMarshaller")]),e._v(":一个 JSON 对象编组器将 Marshall 对象转换为 JSON 格式")])])]),e._v(" "),r("p",[e._v("下面的示例展示了如何定义"),r("code",[e._v("JsonFileItemWriter")]),e._v(":")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic JsonFileItemWriter jsonFileItemWriter() {\n return new JsonFileItemWriterBuilder()\n .jsonObjectMarshaller(new JacksonJsonObjectMarshaller<>())\n .resource(new ClassPathResource("trades.json"))\n .name("tradeJsonFileItemWriter")\n .build();\n}\n')])])]),r("h3",{attrs:{id:"多文件输入"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#多文件输入"}},[e._v("#")]),e._v(" 多文件输入")]),e._v(" "),r("p",[e._v("在一个"),r("code",[e._v("Step")]),e._v("中处理多个文件是一个常见的要求。假设所有文件都具有相同的格式,"),r("code",[e._v("MultiResourceItemReader")]),e._v("在 XML 和平面文件处理中都支持这种类型的输入。考虑目录中的以下文件:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("file-1.txt file-2.txt ignored.txt\n")])])]),r("p",[e._v("File-1.TXT 和 File-2.TXT 的格式相同,出于业务原因,应该一起处理。"),r("code",[e._v("MultiResourceItemReader")]),e._v("可以通过使用通配符在两个文件中读取。")]),e._v(" "),r("p",[e._v("下面的示例展示了如何使用 XML 中的通配符读取文件:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n\n')])])]),r("p",[e._v("下面的示例展示了如何在 Java 中使用通配符读取文件:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("@Bean\npublic MultiResourceItemReader multiResourceReader() {\n\treturn new MultiResourceItemReaderBuilder()\n\t\t\t\t\t.delegate(flatFileItemReader())\n\t\t\t\t\t.resources(resources())\n\t\t\t\t\t.build();\n}\n")])])]),r("p",[e._v("引用的委托是一个简单的"),r("code",[e._v("FlatFileItemReader")]),e._v("。上面的配置读取两个文件的输入,处理回滚和重新启动场景。应该注意的是,与任何"),r("code",[e._v("ItemReader")]),e._v("一样,在重新启动时添加额外的输入(在这种情况下是一个文件)可能会导致潜在的问题。建议批处理作业使用它们自己的独立目录,直到成功完成为止。")]),e._v(" "),r("table",[r("thead",[r("tr",[r("th"),e._v(" "),r("th",[e._v("通过使用"),r("code",[e._v("MultiResourceItemReader#setComparator(Comparator)")]),e._v("对输入资源进行排序,以确保在重新启动场景中的作业运行之间保留资源排序。")])])]),e._v(" "),r("tbody")]),e._v(" "),r("h3",{attrs:{id:"数据库"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#数据库"}},[e._v("#")]),e._v(" 数据库")]),e._v(" "),r("p",[e._v("像大多数 Enterprise 应用程序样式一样,数据库是批处理的中心存储机制。然而,由于系统必须使用的数据集的巨大规模,批处理与其他应用程序样式不同。如果 SQL 语句返回 100 万行,那么结果集可能会将所有返回的结果保存在内存中,直到所有行都被读取为止。 Spring Batch 为此问题提供了两种类型的解决方案:")]),e._v(" "),r("ul",[r("li",[r("p",[e._v("[基于游标的"),r("code",[e._v("ItemReader")]),e._v("实现]")])]),e._v(" "),r("li",[r("p",[e._v("[分页"),r("code",[e._v("ItemReader")]),e._v("实现]")])])]),e._v(" "),r("h4",{attrs:{id:"基于光标的itemreader实现"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#基于光标的itemreader实现"}},[e._v("#")]),e._v(" 基于光标的"),r("code",[e._v("ItemReader")]),e._v("实现")]),e._v(" "),r("p",[e._v("使用数据库游标通常是大多数批处理开发人员的默认方法,因为它是数据库解决关系数据“流”问题的方法。Java"),r("code",[e._v("ResultSet")]),e._v("类本质上是一种用于操作游标的面向对象机制。a"),r("code",[e._v("ResultSet")]),e._v("维护当前数据行的游标。在"),r("code",[e._v("ResultSet")]),e._v("上调用"),r("code",[e._v("next")]),e._v("将光标移动到下一行。 Spring 基于批处理游标的"),r("code",[e._v("ItemReader")]),e._v("实现在初始化时打开游标,并在每次调用"),r("code",[e._v("read")]),e._v("时将游标向前移动一行,返回可用于处理的映射对象。然后调用"),r("code",[e._v("close")]),e._v("方法,以确保释放所有资源。 Spring 核心"),r("code",[e._v("JdbcTemplate")]),e._v("通过使用回调模式来完全映射"),r("code",[e._v("ResultSet")]),e._v("中的所有行,并在将控制权返回给方法调用方之前关闭,从而绕过了这个问题。然而,在批处理中,这必须等到步骤完成。下图显示了基于游标的"),r("code",[e._v("ItemReader")]),e._v("如何工作的通用关系图。请注意,虽然示例使用 SQL(因为 SQL 是广为人知的),但任何技术都可以实现基本方法。")]),e._v(" "),r("p",[r("img",{attrs:{src:"https://docs.spring.io/spring-batch/docs/current/reference/html/images/cursorExample.png",alt:"游标示例"}})]),e._v(" "),r("p",[e._v("图 3.游标示例")]),e._v(" "),r("p",[e._v("这个例子说明了基本模式。给定一个有三列的“foo”表:"),r("code",[e._v("ID")]),e._v("、"),r("code",[e._v("NAME")]),e._v("和"),r("code",[e._v("BAR")]),e._v(",选择 ID 大于 1 但小于 7 的所有行。这将把游标的开头(第 1 行)放在 ID2 上。该行的结果应该是一个完全映射的"),r("code",[e._v("Foo")]),e._v("对象。调用"),r("code",[e._v("read()")]),e._v("再次将光标移动到下一行,即 ID 为 3 的"),r("code",[e._v("Foo")]),e._v("。在每个"),r("code",[e._v("read")]),e._v("之后写出这些读取的结果,从而允许对对象进行垃圾收集(假设没有实例变量维护对它们的引用)。")]),e._v(" "),r("h5",{attrs:{id:"jdbccursoritemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#jdbccursoritemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("JdbcCursorItemReader")])]),e._v(" "),r("p",[r("code",[e._v("JdbcCursorItemReader")]),e._v("是基于光标的技术的 JDBC 实现。它可以直接与"),r("code",[e._v("ResultSet")]),e._v("一起工作,并且需要针对从"),r("code",[e._v("DataSource")]),e._v("获得的连接运行 SQL 语句。下面的数据库模式用作示例:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("CREATE TABLE CUSTOMER (\n ID BIGINT IDENTITY PRIMARY KEY,\n NAME VARCHAR(45),\n CREDIT FLOAT\n);\n")])])]),r("p",[e._v("许多人更喜欢为每一行使用域对象,因此下面的示例使用"),r("code",[e._v("RowMapper")]),e._v("接口的实现来映射"),r("code",[e._v("CustomerCredit")]),e._v("对象:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('public class CustomerCreditRowMapper implements RowMapper {\n\n public static final String ID_COLUMN = "id";\n public static final String NAME_COLUMN = "name";\n public static final String CREDIT_COLUMN = "credit";\n\n public CustomerCredit mapRow(ResultSet rs, int rowNum) throws SQLException {\n CustomerCredit customerCredit = new CustomerCredit();\n\n customerCredit.setId(rs.getInt(ID_COLUMN));\n customerCredit.setName(rs.getString(NAME_COLUMN));\n customerCredit.setCredit(rs.getBigDecimal(CREDIT_COLUMN));\n\n return customerCredit;\n }\n}\n')])])]),r("p",[e._v("因为"),r("code",[e._v("JdbcCursorItemReader")]),e._v("与"),r("code",[e._v("JdbcTemplate")]),e._v("共享关键接口,所以查看如何使用"),r("code",[e._v("JdbcTemplate")]),e._v("在此数据中读取数据的示例非常有用,以便将其与"),r("code",[e._v("ItemReader")]),e._v("进行对比。为了这个示例的目的,假设"),r("code",[e._v("CUSTOMER")]),e._v("数据库中有 1,000 行。第一个示例使用"),r("code",[e._v("JdbcTemplate")]),e._v(":")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('//For simplicity sake, assume a dataSource has already been obtained\nJdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource);\nList customerCredits = jdbcTemplate.query("SELECT ID, NAME, CREDIT from CUSTOMER",\n new CustomerCreditRowMapper());\n')])])]),r("p",[e._v("在运行前面的代码片段之后,"),r("code",[e._v("customerCredits")]),e._v("列表包含 1,000 个"),r("code",[e._v("CustomerCredit")]),e._v("对象。在查询方法中,从"),r("code",[e._v("DataSource")]),e._v("获得连接,对其运行所提供的 SQL,并对"),r("code",[e._v("mapRow")]),e._v("中的每一行调用"),r("code",[e._v("ResultSet")]),e._v("方法。将其与"),r("code",[e._v("JdbcCursorItemReader")]),e._v("的方法进行对比,如下例所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('JdbcCursorItemReader itemReader = new JdbcCursorItemReader();\nitemReader.setDataSource(dataSource);\nitemReader.setSql("SELECT ID, NAME, CREDIT from CUSTOMER");\nitemReader.setRowMapper(new CustomerCreditRowMapper());\nint counter = 0;\nExecutionContext executionContext = new ExecutionContext();\nitemReader.open(executionContext);\nObject customerCredit = new Object();\nwhile(customerCredit != null){\n customerCredit = itemReader.read();\n counter++;\n}\nitemReader.close();\n')])])]),r("p",[e._v("在运行前面的代码片段之后,计数器等于 1,00 0.如果上面的代码将返回的"),r("code",[e._v("customerCredit")]),e._v("放入一个列表中,结果将与"),r("code",[e._v("JdbcTemplate")]),e._v("示例完全相同。然而,"),r("code",[e._v("ItemReader")]),e._v("的一大优势在于,它允许项目被“流化”。"),r("code",[e._v("read")]),e._v("方法可以调用一次,该项可以由一个"),r("code",[e._v("ItemWriter")]),e._v("写出,然后可以用"),r("code",[e._v("read")]),e._v("获得下一个项。这使得项目的读写可以在“块”中完成,并定期提交,这是高性能批处理的本质。此外,很容易地将其配置为将"),r("code",[e._v("Step")]),e._v("注入到 Spring 批中。")]),e._v(" "),r("p",[e._v("下面的示例展示了如何在 XML 中将"),r("code",[e._v("ItemReader")]),e._v("插入到"),r("code",[e._v("Step")]),e._v("中:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n \n\n')])])]),r("p",[e._v("下面的示例展示了如何在 Java 中将"),r("code",[e._v("ItemReader")]),e._v("注入"),r("code",[e._v("Step")]),e._v(":")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic JdbcCursorItemReader itemReader() {\n\treturn new JdbcCursorItemReaderBuilder()\n\t\t\t.dataSource(this.dataSource)\n\t\t\t.name("creditReader")\n\t\t\t.sql("select ID, NAME, CREDIT from CUSTOMER")\n\t\t\t.rowMapper(new CustomerCreditRowMapper())\n\t\t\t.build();\n\n}\n')])])]),r("h6",{attrs:{id:"附加属性"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#附加属性"}},[e._v("#")]),e._v(" 附加属性")]),e._v(" "),r("p",[e._v("因为在 Java 中有很多不同的打开光标的选项,所以"),r("code",[e._v("JdbcCursorItemReader")]),e._v("上有很多可以设置的属性,如下表所示:")]),e._v(" "),r("table",[r("thead",[r("tr",[r("th",[e._v("ignoreWarnings")]),e._v(" "),r("th",[e._v("确定是否记录了 SQLwarns 或是否导致异常。"),r("br"),e._v("默认值是"),r("code",[e._v("true")]),e._v("(这意味着记录了警告)。")])])]),e._v(" "),r("tbody",[r("tr",[r("td",[e._v("fetchSize")]),e._v(" "),r("td",[e._v("当"),r("code",[e._v("ResultSet")]),e._v("对象所使用的"),r("code",[e._v("ResultSet")]),e._v("对象需要更多行时,向 JDBC 驱动程序提供有关应该从数据库中获取"),r("br"),e._v("的行数的提示。默认情况下,不会给出任何提示。")])]),e._v(" "),r("tr",[r("td",[e._v("maxRows")]),e._v(" "),r("td",[e._v("设置底层"),r("code",[e._v("ResultSet")]),e._v("在任何时候都可以"),r("br"),e._v("的最大行数的限制。")])]),e._v(" "),r("tr",[r("td",[e._v("queryTimeout")]),e._v(" "),r("td",[e._v("将驱动程序等待"),r("code",[e._v("Statement")]),e._v("对象的秒数设置为"),r("br"),e._v("运行。如果超过限制,则抛出"),r("code",[e._v("DataAccessException")]),e._v("。(有关详细信息,请咨询你的驱动程序"),r("br"),e._v("供应商文档)。")])]),e._v(" "),r("tr",[r("td",[e._v("verifyCursorPosition")]),e._v(" "),r("td",[e._v("因为由"),r("code",[e._v("ItemReader")]),e._v("持有的相同"),r("code",[e._v("ResultSet")]),e._v("被传递到"),r("br"),r("code",[e._v("RowMapper")]),e._v(",所以用户可以自己调用"),r("code",[e._v("ResultSet.next()")]),e._v(",这可能会导致阅读器的内部计数出现问题。将该值设置为"),r("code",[e._v("true")]),e._v("会导致"),r("br"),e._v("在"),r("code",[e._v("RowMapper")]),e._v("调用后,如果光标位置与以前不同,将引发一个异常。")])]),e._v(" "),r("tr",[r("td",[e._v("saveState")]),e._v(" "),r("td",[e._v("指示是否应将读取器的状态保存在"),r("code",[e._v("ExecutionContext")]),e._v("提供的"),r("code",[e._v("ItemStream#update(ExecutionContext)")]),e._v("中。默认值为"),r("code",[e._v("true")]),e._v("。")])]),e._v(" "),r("tr",[r("td",[e._v("driverSupportsAbsolute")]),e._v(" "),r("td",[e._v("指示 JDBC 驱动程序是否支持"),r("br"),e._v("设置"),r("code",[e._v("ResultSet")]),e._v("上的绝对行。对于支持"),r("code",[e._v("ResultSet.absolute()")]),e._v("的 JDBC 驱动程序,建议将其设置为"),r("code",[e._v("true")]),e._v(",因为这可能会提高性能,"),r("br"),e._v("特别是在使用大数据集时发生步骤失败时。默认值为"),r("code",[e._v("false")]),e._v("。")])]),e._v(" "),r("tr",[r("td",[e._v("setUseSharedExtendedConnection")]),e._v(" "),r("td",[e._v("指示用于光标的连接"),r("br"),e._v("是否应由所有其他处理使用,从而共享相同的"),r("br"),e._v("事务。如果将其设置为"),r("code",[e._v("false")]),e._v(",然后用它自己的连接"),r("br"),e._v("打开光标,并且不参与启动的任何事务对于步骤处理的其余部分,"),r("br"),e._v("如果将此标志设置为"),r("code",[e._v("true")]),e._v(",则必须将数据源包装在"),r("code",[e._v("ExtendedConnectionDataSourceProxy")]),e._v("中,以防止连接被关闭,并在每次提交后释放"),r("br"),e._v("。当你将此选项设置为"),r("code",[e._v("true")]),e._v("时,用于"),r("br"),e._v("打开光标的语句将使用’只读’和’持有 _ 游标 _over_commit’选项创建。"),r("br"),e._v("这允许在事务启动时保持光标打开,并在"),r("br"),e._v("步骤处理中执行提交。要使用此功能,你需要一个支持此功能的数据库,以及一个支持 JDBC3.0 或更高版本的 JDBC"),r("br"),e._v("驱动程序。默认值为"),r("code",[e._v("false")]),e._v("。")])])])]),e._v(" "),r("h5",{attrs:{id:"hibernatecursoritemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#hibernatecursoritemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("HibernateCursorItemReader")])]),e._v(" "),r("p",[e._v("正如正常的 Spring 用户对是否使用 ORM 解决方案做出重要的决定,这会影响他们是否使用"),r("code",[e._v("JdbcTemplate")]),e._v("或"),r("code",[e._v("HibernateTemplate")]),e._v(", Spring 批处理用户具有相同的选项。"),r("code",[e._v("HibernateCursorItemReader")]),e._v("是 Hibernate 游标技术的实现。 Hibernate 的批量使用一直颇具争议。这在很大程度上是因为 Hibernate 最初是为了支持在线应用程序样式而开发的。然而,这并不意味着它不能用于批处理。解决这个问题的最简单的方法是使用"),r("code",[e._v("StatelessSession")]),e._v(",而不是使用标准会话。这删除了 Hibernate 使用的所有缓存和脏检查,这可能会在批处理场景中导致问题。有关无状态会话和正常 Hibernate 会话之间的差异的更多信息,请参阅你的特定 Hibernate 版本的文档。"),r("code",[e._v("HibernateCursorItemReader")]),e._v("允许你声明一个 HQL 语句,并传入一个"),r("code",[e._v("SessionFactory")]),e._v(",它将在每个调用中传回一个项,以与"),r("code",[e._v("JdbcCursorItemReader")]),e._v("相同的基本方式进行读取。下面的示例配置使用了与 JDBC 阅读器相同的“客户信用”示例:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('HibernateCursorItemReader itemReader = new HibernateCursorItemReader();\nitemReader.setQueryString("from CustomerCredit");\n//For simplicity sake, assume sessionFactory already obtained.\nitemReader.setSessionFactory(sessionFactory);\nitemReader.setUseStatelessSession(true);\nint counter = 0;\nExecutionContext executionContext = new ExecutionContext();\nitemReader.open(executionContext);\nObject customerCredit = new Object();\nwhile(customerCredit != null){\n customerCredit = itemReader.read();\n counter++;\n}\nitemReader.close();\n')])])]),r("p",[e._v("这个配置的"),r("code",[e._v("ItemReader")]),e._v("以与"),r("code",[e._v("JdbcCursorItemReader")]),e._v("所描述的完全相同的方式返回"),r("code",[e._v("CustomerCredit")]),e._v("对象,假设 Hibernate 已经为"),r("code",[e._v("Customer")]),e._v("表正确地创建了映射文件。“useStatelession”属性默认为 true,但在此添加此属性是为了提请注意打开或关闭它的能力。还值得注意的是,可以使用"),r("code",[e._v("setFetchSize")]),e._v("属性设置底层游标的 fetch 大小。与"),r("code",[e._v("JdbcCursorItemReader")]),e._v("一样,配置也很简单。")]),e._v(" "),r("p",[e._v("下面的示例展示了如何在 XML 中注入 Hibernate "),r("code",[e._v("ItemReader")]),e._v(":")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n\n')])])]),r("p",[e._v("下面的示例展示了如何在 Java 中注入 Hibernate "),r("code",[e._v("ItemReader")]),e._v(":")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic HibernateCursorItemReader itemReader(SessionFactory sessionFactory) {\n\treturn new HibernateCursorItemReaderBuilder()\n\t\t\t.name("creditReader")\n\t\t\t.sessionFactory(sessionFactory)\n\t\t\t.queryString("from CustomerCredit")\n\t\t\t.build();\n}\n')])])]),r("h5",{attrs:{id:"storedprocedureitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#storedprocedureitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("StoredProcedureItemReader")])]),e._v(" "),r("p",[e._v("有时需要使用存储过程来获取游标数据。"),r("code",[e._v("StoredProcedureItemReader")]),e._v("的工作原理与"),r("code",[e._v("JdbcCursorItemReader")]),e._v("类似,不同的是,它运行的是返回光标的存储过程,而不是运行查询来获取光标。存储过程可以以三种不同的方式返回光标:")]),e._v(" "),r("ul",[r("li",[r("p",[e._v("作为返回的"),r("code",[e._v("ResultSet")]),e._v("(由 SQL Server、Sybase、DB2、Derby 和 MySQL 使用)。")])]),e._v(" "),r("li",[r("p",[e._v("作为 ref-cursor 作为 out 参数返回(Oracle 和 PostgreSQL 使用)。")])]),e._v(" "),r("li",[r("p",[e._v("作为存储函数调用的返回值。")])])]),e._v(" "),r("p",[e._v("下面的 XML 示例配置使用了与前面的示例相同的“客户信用”示例:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n \n\n')])])]),r("p",[e._v("下面的 Java 示例配置使用了与前面的示例相同的“客户信用”示例:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic StoredProcedureItemReader reader(DataSource dataSource) {\n\tStoredProcedureItemReader reader = new StoredProcedureItemReader();\n\n\treader.setDataSource(dataSource);\n\treader.setProcedureName("sp_customer_credit");\n\treader.setRowMapper(new CustomerCreditRowMapper());\n\n\treturn reader;\n}\n')])])]),r("p",[e._v("前面的示例依赖于存储过程来提供"),r("code",[e._v("ResultSet")]),e._v("作为返回的结果(前面的选项 1)。")]),e._v(" "),r("p",[e._v("如果存储过程返回了"),r("code",[e._v("ref-cursor")]),e._v("(选项 2),那么我们将需要提供输出参数的位置,即返回的"),r("code",[e._v("ref-cursor")]),e._v("。")]),e._v(" "),r("p",[e._v("下面的示例展示了如何使用第一个参数作为 XML 中的 ref-cursor:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n \n \n\n')])])]),r("p",[e._v("下面的示例展示了如何使用第一个参数作为 Java 中的 ref-cursor:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic StoredProcedureItemReader reader(DataSource dataSource) {\n\tStoredProcedureItemReader reader = new StoredProcedureItemReader();\n\n\treader.setDataSource(dataSource);\n\treader.setProcedureName("sp_customer_credit");\n\treader.setRowMapper(new CustomerCreditRowMapper());\n\treader.setRefCursorPosition(1);\n\n\treturn reader;\n}\n')])])]),r("p",[e._v("如果光标是从存储函数返回的(选项 3),则需要将属性“function”设置为"),r("code",[e._v("true")]),e._v("。它的默认值为"),r("code",[e._v("false")]),e._v("。")]),e._v(" "),r("p",[e._v("下面的示例在 XML 中向"),r("code",[e._v("true")]),e._v("显示了属性:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n \n \n\n')])])]),r("p",[e._v("下面的示例在 Java 中向"),r("code",[e._v("true")]),e._v("显示了属性:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic StoredProcedureItemReader reader(DataSource dataSource) {\n\tStoredProcedureItemReader reader = new StoredProcedureItemReader();\n\n\treader.setDataSource(dataSource);\n\treader.setProcedureName("sp_customer_credit");\n\treader.setRowMapper(new CustomerCreditRowMapper());\n\treader.setFunction(true);\n\n\treturn reader;\n}\n')])])]),r("p",[e._v("在所有这些情况下,我们需要定义一个"),r("code",[e._v("RowMapper")]),e._v("以及一个"),r("code",[e._v("DataSource")]),e._v("和实际的过程名称。")]),e._v(" "),r("p",[e._v("如果存储过程或函数接受参数,则必须使用"),r("code",[e._v("parameters")]),e._v("属性声明和设置参数。下面的示例为 Oracle 声明了三个参数。第一个参数是返回 ref-cursor 的"),r("code",[e._v("out")]),e._v("参数,第二个和第三个参数是参数中的"),r("code",[e._v("INTEGER")]),e._v("类型的值。")]),e._v(" "),r("p",[e._v("下面的示例展示了如何使用 XML 中的参数:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n\n')])])]),r("p",[e._v("下面的示例展示了如何使用 Java 中的参数:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic StoredProcedureItemReader reader(DataSource dataSource) {\n\tList parameters = new ArrayList<>();\n\tparameters.add(new SqlOutParameter("newId", OracleTypes.CURSOR));\n\tparameters.add(new SqlParameter("amount", Types.INTEGER);\n\tparameters.add(new SqlParameter("custId", Types.INTEGER);\n\n\tStoredProcedureItemReader reader = new StoredProcedureItemReader();\n\n\treader.setDataSource(dataSource);\n\treader.setProcedureName("spring.cursor_func");\n\treader.setParameters(parameters);\n\treader.setRefCursorPosition(1);\n\treader.setRowMapper(rowMapper());\n\treader.setPreparedStatementSetter(parameterSetter());\n\n\treturn reader;\n}\n')])])]),r("p",[e._v("除了参数声明外,我们还需要指定一个"),r("code",[e._v("PreparedStatementSetter")]),e._v("实现,该实现为调用设置参数值。这与上面的"),r("code",[e._v("JdbcCursorItemReader")]),e._v("的工作原理相同。"),r("a",{attrs:{href:"#JdbcCursorItemReaderProperties"}},[e._v("附加属性")]),e._v("中列出的所有附加属性也适用于"),r("code",[e._v("StoredProcedureItemReader")]),e._v("。")]),e._v(" "),r("h4",{attrs:{id:"分页itemreader实现"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#分页itemreader实现"}},[e._v("#")]),e._v(" 分页"),r("code",[e._v("ItemReader")]),e._v("实现")]),e._v(" "),r("p",[e._v("使用数据库游标的一种替代方法是运行多个查询,其中每个查询获取部分结果。我们把这一部分称为一个页面。每个查询必须指定起始行号和我们希望在页面中返回的行数。")]),e._v(" "),r("h5",{attrs:{id:"jdbcpagingitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#jdbcpagingitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("JdbcPagingItemReader")])]),e._v(" "),r("p",[e._v("分页"),r("code",[e._v("ItemReader")]),e._v("的一个实现是"),r("code",[e._v("JdbcPagingItemReader")]),e._v("。"),r("code",[e._v("JdbcPagingItemReader")]),e._v("需要一个"),r("code",[e._v("PagingQueryProvider")]),e._v(",负责提供用于检索构成页面的行的 SQL 查询。由于每个数据库都有自己的策略来提供分页支持,因此我们需要为每个受支持的数据库类型使用不同的"),r("code",[e._v("PagingQueryProvider")]),e._v("。还有"),r("code",[e._v("SqlPagingQueryProviderFactoryBean")]),e._v("自动检测正在使用的数据库,并确定适当的"),r("code",[e._v("PagingQueryProvider")]),e._v("实现。这简化了配置,是推荐的最佳实践。")]),e._v(" "),r("p",[r("code",[e._v("SqlPagingQueryProviderFactoryBean")]),e._v("要求你指定"),r("code",[e._v("select")]),e._v("子句和"),r("code",[e._v("from")]),e._v("子句。你还可以提供一个可选的"),r("code",[e._v("where")]),e._v("子句。这些子句和所需的"),r("code",[e._v("sortKey")]),e._v("用于构建 SQL 语句。")]),e._v(" "),r("table",[r("thead",[r("tr",[r("th"),e._v(" "),r("th",[e._v("在"),r("code",[e._v("sortKey")]),e._v("上有一个唯一的键约束是很重要的,以保证"),r("br"),e._v("在两次执行之间不会丢失任何数据。")])])]),e._v(" "),r("tbody")]),e._v(" "),r("p",[e._v("打开读取器后,它会以与任何其他"),r("code",[e._v("ItemReader")]),e._v("相同的基本方式,将每个调用返回一个项到"),r("code",[e._v("read")]),e._v("。当需要额外的行时,分页会在幕后进行。")]),e._v(" "),r("p",[e._v("下面的 XML 示例配置使用了与前面显示的基于游标的"),r("code",[e._v("ItemReaders")]),e._v("类似的“客户信用”示例:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n\n')])])]),r("p",[e._v("下面的 Java 示例配置使用了与前面显示的基于游标的"),r("code",[e._v("ItemReaders")]),e._v("类似的“客户信用”示例:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic JdbcPagingItemReader itemReader(DataSource dataSource, PagingQueryProvider queryProvider) {\n\tMap parameterValues = new HashMap<>();\n\tparameterValues.put("status", "NEW");\n\n\treturn new JdbcPagingItemReaderBuilder()\n \t\t\t\t.name("creditReader")\n \t\t\t\t.dataSource(dataSource)\n \t\t\t\t.queryProvider(queryProvider)\n \t\t\t\t.parameterValues(parameterValues)\n \t\t\t\t.rowMapper(customerCreditMapper())\n \t\t\t\t.pageSize(1000)\n \t\t\t\t.build();\n}\n\n@Bean\npublic SqlPagingQueryProviderFactoryBean queryProvider() {\n\tSqlPagingQueryProviderFactoryBean provider = new SqlPagingQueryProviderFactoryBean();\n\n\tprovider.setSelectClause("select id, name, credit");\n\tprovider.setFromClause("from customer");\n\tprovider.setWhereClause("where status=:status");\n\tprovider.setSortKey("id");\n\n\treturn provider;\n}\n')])])]),r("p",[e._v("此配置的"),r("code",[e._v("ItemReader")]),e._v("使用"),r("code",[e._v("RowMapper")]),e._v("返回"),r("code",[e._v("CustomerCredit")]),e._v("对象,该对象必须指定。“PageSize”属性确定每次运行查询时从数据库中读取的实体的数量。")]),e._v(" "),r("p",[e._v("“parametervalues”属性可用于为查询指定一个"),r("code",[e._v("Map")]),e._v("参数值。如果在"),r("code",[e._v("where")]),e._v("子句中使用命名参数,则每个条目的键应该与命名参数的名称匹配。如果使用传统的“?”占位符,那么每个条目的键应该是占位符的编号,从 1 开始。")]),e._v(" "),r("h5",{attrs:{id:"jpapagingitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#jpapagingitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("JpaPagingItemReader")])]),e._v(" "),r("p",[e._v("分页"),r("code",[e._v("ItemReader")]),e._v("的另一个实现是"),r("code",[e._v("JpaPagingItemReader")]),e._v("。 JPA 不具有类似于 Hibernate 的概念,因此我们不得不使用由 JPA 规范提供的其他特征。由于 JPA 支持分页,所以当涉及到使用 JPA 进行批处理时,这是一个自然的选择。在读取每个页面之后,这些实体将被分离,持久性上下文将被清除,从而允许在页面被处理之后对这些实体进行垃圾收集。")]),e._v(" "),r("p",[r("code",[e._v("JpaPagingItemReader")]),e._v("允许你声明一个 JPQL 语句,并传入一个"),r("code",[e._v("EntityManagerFactory")]),e._v("。然后,它在每个调用中传回一个项,以与任何其他"),r("code",[e._v("ItemReader")]),e._v("相同的基本方式进行读取。当需要额外的实体时,寻呼就会在幕后进行。")]),e._v(" "),r("p",[e._v("下面的 XML 示例配置使用了与前面显示的 JDBC 阅读器相同的“客户信用”示例:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n\n')])])]),r("p",[e._v("下面的 Java 示例配置使用了与前面显示的 JDBC 阅读器相同的“客户信用”示例:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic JpaPagingItemReader itemReader() {\n\treturn new JpaPagingItemReaderBuilder()\n \t\t\t\t.name("creditReader")\n \t\t\t\t.entityManagerFactory(entityManagerFactory())\n \t\t\t\t.queryString("select c from CustomerCredit c")\n \t\t\t\t.pageSize(1000)\n \t\t\t\t.build();\n}\n')])])]),r("p",[e._v("这个配置的"),r("code",[e._v("ItemReader")]),e._v("以与上面描述的"),r("code",[e._v("JdbcPagingItemReader")]),e._v("对象完全相同的方式返回"),r("code",[e._v("CustomerCredit")]),e._v("对象,假设"),r("code",[e._v("CustomerCredit")]),e._v("对象具有正确的 JPA 注释或 ORM 映射文件。“PageSize”属性确定每个查询执行从数据库中读取的实体的数量。")]),e._v(" "),r("h4",{attrs:{id:"数据库项目编写器"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#数据库项目编写器"}},[e._v("#")]),e._v(" 数据库项目编写器")]),e._v(" "),r("p",[e._v("虽然平面文件和 XML 文件都有一个特定的"),r("code",[e._v("ItemWriter")]),e._v("实例,但在数据库世界中没有完全相同的实例。这是因为事务提供了所需的所有功能。"),r("code",[e._v("ItemWriter")]),e._v("实现对于文件来说是必要的,因为它们必须像事务一样工作,跟踪写好的项目,并在适当的时候刷新或清除。数据库不需要此功能,因为写操作已经包含在事务中了。用户可以创建自己的 DAO 来实现"),r("code",[e._v("ItemWriter")]),e._v("接口,或者使用自定义的"),r("code",[e._v("ItemWriter")]),e._v("接口,这是为通用处理问题编写的。无论哪种方式,它们的工作都应该没有任何问题。需要注意的一点是批处理输出所提供的性能和错误处理能力。当使用 Hibernate 作为"),r("code",[e._v("ItemWriter")]),e._v("时,这是最常见的,但是当使用 JDBC 批处理模式时,可能会有相同的问题。批处理数据库输出没有任何固有的缺陷,前提是我们要小心刷新,并且数据中没有错误。然而,书写时的任何错误都可能导致混淆,因为无法知道是哪个单独的项目导致了异常,或者即使是任何单独的项目是负责任的,如下图所示:")]),e._v(" "),r("p",[r("img",{attrs:{src:"https://docs.spring.io/spring-batch/docs/current/reference/html/images/errorOnFlush.png",alt:"刷新错误"}})]),e._v(" "),r("p",[e._v("图 4.刷新错误")]),e._v(" "),r("p",[e._v("如果项目在写入之前被缓冲,则在提交之前刷新缓冲区之前不会抛出任何错误。例如,假设每个块写 20 个项,第 15 个项抛出一个"),r("code",[e._v("DataIntegrityViolationException")]),e._v("。就"),r("code",[e._v("Step")]),e._v("而言,所有 20 个项都已成功写入,因为只有在实际写入它们之前,才能知道发生了错误。一旦调用"),r("code",[e._v("Session#flush()")]),e._v(",将清空缓冲区并命中异常。在这一点上,"),r("code",[e._v("Step")]),e._v("是无能为力的。事务必须回滚。通常,此异常可能会导致跳过该项(取决于跳过/重试策略),然后不会再次写入该项。但是,在批处理场景中,无法知道是哪个项导致了问题。当故障发生时,整个缓冲区正在被写入。解决此问题的唯一方法是在每个项目之后进行刷新,如下图所示:")]),e._v(" "),r("p",[r("img",{attrs:{src:"https://docs.spring.io/spring-batch/docs/current/reference/html/images/errorOnWrite.png",alt:"写错误"}})]),e._v(" "),r("p",[e._v("图 5.写错误")]),e._v(" "),r("p",[e._v("这是一个常见的用例,尤其是在使用 Hibernate 时,而"),r("code",[e._v("ItemWriter")]),e._v("的实现的简单准则是在每次调用"),r("code",[e._v("write()")]),e._v("时刷新。这样做允许可靠地跳过项, Spring 批处理在内部处理错误后对"),r("code",[e._v("ItemWriter")]),e._v("的调用的粒度。")]),e._v(" "),r("h3",{attrs:{id:"重用现有服务"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#重用现有服务"}},[e._v("#")]),e._v(" 重用现有服务")]),e._v(" "),r("p",[e._v("批处理系统通常与其他应用程序样式结合使用。最常见的是在线系统,但它也可以通过移动每个应用程序样式使用的必要的大容量数据来支持集成,甚至支持厚客户机应用程序。由于这个原因,许多用户希望在其批处理作业中重用现有的 DAO 或其他服务是很常见的。 Spring 容器本身通过允许注入任何必要的类,使这一点变得相当容易。然而,可能存在现有服务需要充当"),r("code",[e._v("ItemReader")]),e._v("或"),r("code",[e._v("ItemWriter")]),e._v("的情况,要么是为了满足另一个 Spring 批处理类的依赖关系,要么是因为它确实是主要的"),r("code",[e._v("ItemReader")]),e._v("的一个步骤。为每个需要包装的服务编写一个适配器类是相当琐碎的,但是由于这是一个常见的问题, Spring Batch 提供了实现:"),r("code",[e._v("ItemReaderAdapter")]),e._v("和"),r("code",[e._v("ItemWriterAdapter")]),e._v("。这两个类都通过调用委托模式来实现标准 Spring 方法,并且设置起来相当简单。")]),e._v(" "),r("p",[e._v("下面的 XML 示例使用"),r("code",[e._v("ItemReaderAdapter")]),e._v(":")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n\n\n\n')])])]),r("p",[e._v("下面的 Java 示例使用"),r("code",[e._v("ItemReaderAdapter")]),e._v(":")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic ItemReaderAdapter itemReader() {\n\tItemReaderAdapter reader = new ItemReaderAdapter();\n\n\treader.setTargetObject(fooService());\n\treader.setTargetMethod("generateFoo");\n\n\treturn reader;\n}\n\n@Bean\npublic FooService fooService() {\n\treturn new FooService();\n}\n')])])]),r("p",[e._v("需要注意的一点是,"),r("code",[e._v("targetMethod")]),e._v("的契约必须与"),r("code",[e._v("read")]),e._v("的契约相同:当耗尽时,它返回"),r("code",[e._v("null")]),e._v("。否则,它返回一个"),r("code",[e._v("Object")]),e._v("。根据"),r("code",[e._v("ItemWriter")]),e._v("的实现,任何其他方法都会阻止框架知道处理应该何时结束,从而导致无限循环或错误失败。")]),e._v(" "),r("p",[e._v("下面的 XML 示例使用"),r("code",[e._v("ItemWriterAdapter")]),e._v(":")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n\n\n\n')])])]),r("p",[e._v("下面的 Java 示例使用"),r("code",[e._v("ItemWriterAdapter")]),e._v(":")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic ItemWriterAdapter itemWriter() {\n\tItemWriterAdapter writer = new ItemWriterAdapter();\n\n\twriter.setTargetObject(fooService());\n\twriter.setTargetMethod("processFoo");\n\n\treturn writer;\n}\n\n@Bean\npublic FooService fooService() {\n\treturn new FooService();\n}\n')])])]),r("h3",{attrs:{id:"防止状态持久性"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#防止状态持久性"}},[e._v("#")]),e._v(" 防止状态持久性")]),e._v(" "),r("p",[e._v("默认情况下,所有"),r("code",[e._v("ItemReader")]),e._v("和"),r("code",[e._v("ItemWriter")]),e._v("实现在提交之前将其当前状态存储在"),r("code",[e._v("ExecutionContext")]),e._v("中。然而,这可能并不总是理想的行为。例如,许多开发人员选择通过使用过程指示器使他们的数据库阅读器“可重新运行”。在输入数据中添加一个额外的列,以指示是否对其进行了处理。当读取(或写入)特定记录时,处理后的标志从"),r("code",[e._v("false")]),e._v("翻转到"),r("code",[e._v("true")]),e._v("。然后,SQL 语句可以在"),r("code",[e._v("where")]),e._v("子句中包含一个额外的语句,例如"),r("code",[e._v("where PROCESSED_IND = false")]),e._v(",从而确保在重新启动的情况下仅返回未处理的记录。在这种情况下,最好不要存储任何状态,例如当前行号,因为它在重新启动时是不相关的。由于这个原因,所有的读者和作者都包括“SaveState”财产。")]),e._v(" "),r("p",[e._v("Bean 下面的定义展示了如何防止 XML 中的状态持久性:")]),e._v(" "),r("p",[e._v("XML 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('\n \n \n \n \n \n \n \n SELECT games.player_id, games.year_no, SUM(COMPLETES),\n SUM(ATTEMPTS), SUM(PASSING_YARDS), SUM(PASSING_TD),\n SUM(INTERCEPTIONS), SUM(RUSHES), SUM(RUSH_YARDS),\n SUM(RECEPTIONS), SUM(RECEPTIONS_YARDS), SUM(TOTAL_TD)\n from games, players where players.player_id =\n games.player_id group by games.player_id, games.year_no\n \n \n\n')])])]),r("p",[e._v("Bean 下面的定义展示了如何在 Java 中防止状态持久性:")]),e._v(" "),r("p",[e._v("Java 配置")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('@Bean\npublic JdbcCursorItemReader playerSummarizationSource(DataSource dataSource) {\n\treturn new JdbcCursorItemReaderBuilder()\n\t\t\t\t.dataSource(dataSource)\n\t\t\t\t.rowMapper(new PlayerSummaryMapper())\n\t\t\t\t.saveState(false)\n\t\t\t\t.sql("SELECT games.player_id, games.year_no, SUM(COMPLETES),"\n\t\t\t\t + "SUM(ATTEMPTS), SUM(PASSING_YARDS), SUM(PASSING_TD),"\n\t\t\t\t + "SUM(INTERCEPTIONS), SUM(RUSHES), SUM(RUSH_YARDS),"\n\t\t\t\t + "SUM(RECEPTIONS), SUM(RECEPTIONS_YARDS), SUM(TOTAL_TD)"\n\t\t\t\t + "from games, players where players.player_id ="\n\t\t\t\t + "games.player_id group by games.player_id, games.year_no")\n\t\t\t\t.build();\n\n}\n')])])]),r("p",[e._v("上面配置的"),r("code",[e._v("ItemReader")]),e._v("不会在"),r("code",[e._v("ExecutionContext")]),e._v("中为其参与的任何执行创建任何条目。")]),e._v(" "),r("h3",{attrs:{id:"创建自定义项目阅读器和项目编写器"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#创建自定义项目阅读器和项目编写器"}},[e._v("#")]),e._v(" 创建自定义项目阅读器和项目编写器")]),e._v(" "),r("p",[e._v("到目前为止,本章已经讨论了 Spring 批处理中的读和写的基本契约,以及这样做的一些常见实现。然而,这些都是相当通用的,并且有许多潜在的场景可能不会被开箱即用的实现所覆盖。本节通过使用一个简单的示例,展示了如何创建自定义"),r("code",[e._v("ItemReader")]),e._v("和"),r("code",[e._v("ItemWriter")]),e._v("实现,并正确地实现它们的契约。"),r("code",[e._v("ItemReader")]),e._v("还实现了"),r("code",[e._v("ItemStream")]),e._v(",以说明如何使读取器或写入器重新启动。")]),e._v(" "),r("h4",{attrs:{id:"自定义itemreader示例"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#自定义itemreader示例"}},[e._v("#")]),e._v(" 自定义"),r("code",[e._v("ItemReader")]),e._v("示例")]),e._v(" "),r("p",[e._v("为了这个示例的目的,我们创建了一个简单的"),r("code",[e._v("ItemReader")]),e._v("实现,该实现从提供的列表中读取数据。我们首先实现"),r("code",[e._v("ItemReader")]),e._v("的最基本契约,即"),r("code",[e._v("read")]),e._v("方法,如以下代码所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public class CustomItemReader implements ItemReader {\n\n List items;\n\n public CustomItemReader(List items) {\n this.items = items;\n }\n\n public T read() throws Exception, UnexpectedInputException,\n NonTransientResourceException, ParseException {\n\n if (!items.isEmpty()) {\n return items.remove(0);\n }\n return null;\n }\n}\n")])])]),r("p",[e._v("前面的类获取一个项目列表,并一次返回一个项目,将每个项目从列表中删除。当列表为空时,它返回"),r("code",[e._v("null")]),e._v(",从而满足"),r("code",[e._v("ItemReader")]),e._v("的最基本要求,如下面的测试代码所示:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('List items = new ArrayList<>();\nitems.add("1");\nitems.add("2");\nitems.add("3");\n\nItemReader itemReader = new CustomItemReader<>(items);\nassertEquals("1", itemReader.read());\nassertEquals("2", itemReader.read());\nassertEquals("3", itemReader.read());\nassertNull(itemReader.read());\n')])])]),r("h5",{attrs:{id:"使itemreader可重启"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#使itemreader可重启"}},[e._v("#")]),e._v(" 使"),r("code",[e._v("ItemReader")]),e._v("可重启")]),e._v(" "),r("p",[e._v("最后的挑战是使"),r("code",[e._v("ItemReader")]),e._v("重新启动。目前,如果处理被中断并重新开始,"),r("code",[e._v("ItemReader")]),e._v("必须在开始时开始。这实际上在许多场景中都是有效的,但有时更可取的做法是,在批处理作业停止的地方重新启动它。关键的判别式通常是读者是有状态的还是无状态的。无状态的读者不需要担心重启性,但是有状态的读者必须尝试在重新启动时重建其最后已知的状态。出于这个原因,我们建议你在可能的情况下保持自定义阅读器的无状态,这样你就不必担心重启性了。")]),e._v(" "),r("p",[e._v("如果确实需要存储状态,那么应该使用"),r("code",[e._v("ItemStream")]),e._v("接口:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('public class CustomItemReader implements ItemReader, ItemStream {\n\n List items;\n int currentIndex = 0;\n private static final String CURRENT_INDEX = "current.index";\n\n public CustomItemReader(List items) {\n this.items = items;\n }\n\n public T read() throws Exception, UnexpectedInputException,\n ParseException, NonTransientResourceException {\n\n if (currentIndex < items.size()) {\n return items.get(currentIndex++);\n }\n\n return null;\n }\n\n public void open(ExecutionContext executionContext) throws ItemStreamException {\n if (executionContext.containsKey(CURRENT_INDEX)) {\n currentIndex = new Long(executionContext.getLong(CURRENT_INDEX)).intValue();\n }\n else {\n currentIndex = 0;\n }\n }\n\n public void update(ExecutionContext executionContext) throws ItemStreamException {\n executionContext.putLong(CURRENT_INDEX, new Long(currentIndex).longValue());\n }\n\n public void close() throws ItemStreamException {}\n}\n')])])]),r("p",[e._v("在每次调用"),r("code",[e._v("ItemStream``update")]),e._v("方法时,"),r("code",[e._v("ItemReader")]),e._v("的当前索引都存储在提供的"),r("code",[e._v("ExecutionContext")]),e._v("中,其键为“current.index”。当调用"),r("code",[e._v("ItemStream``open")]),e._v("方法时,将检查"),r("code",[e._v("ExecutionContext")]),e._v("是否包含带有该键的条目。如果找到了键,则将当前索引移动到该位置。这是一个相当微不足道的例子,但它仍然符合一般合同:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v('ExecutionContext executionContext = new ExecutionContext();\n((ItemStream)itemReader).open(executionContext);\nassertEquals("1", itemReader.read());\n((ItemStream)itemReader).update(executionContext);\n\nList items = new ArrayList<>();\nitems.add("1");\nitems.add("2");\nitems.add("3");\nitemReader = new CustomItemReader<>(items);\n\n((ItemStream)itemReader).open(executionContext);\nassertEquals("2", itemReader.read());\n')])])]),r("p",[e._v("大多数"),r("code",[e._v("ItemReaders")]),e._v("都有更复杂的重启逻辑。例如,"),r("code",[e._v("JdbcCursorItemReader")]),e._v("将最后处理的行的行 ID 存储在游标中。")]),e._v(" "),r("p",[e._v("还值得注意的是,"),r("code",[e._v("ExecutionContext")]),e._v("中使用的键不应该是微不足道的。这是因为相同的"),r("code",[e._v("ExecutionContext")]),e._v("用于"),r("code",[e._v("ItemStreams")]),e._v("中的所有"),r("code",[e._v("Step")]),e._v("。在大多数情况下,只需在键前加上类名就足以保证唯一性。然而,在很少的情况下,在相同的步骤中使用两个相同类型的"),r("code",[e._v("ItemStream")]),e._v("(如果需要输出两个文件,可能会发生这种情况),则需要一个更唯一的名称。由于这个原因,许多 Spring 批处理"),r("code",[e._v("ItemReader")]),e._v("和"),r("code",[e._v("ItemWriter")]),e._v("实现都有一个"),r("code",[e._v("setName()")]),e._v("属性,该属性允许重写这个键名。")]),e._v(" "),r("h4",{attrs:{id:"自定义itemwriter示例"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#自定义itemwriter示例"}},[e._v("#")]),e._v(" 自定义"),r("code",[e._v("ItemWriter")]),e._v("示例")]),e._v(" "),r("p",[e._v("实现自定义"),r("code",[e._v("ItemWriter")]),e._v("在许多方面与上面的"),r("code",[e._v("ItemReader")]),e._v("示例相似,但在足够多的方面有所不同,以保证它自己的示例。然而,添加可重启性本质上是相同的,因此在本例中不涉及它。与"),r("code",[e._v("ItemReader")]),e._v("示例一样,使用"),r("code",[e._v("List")]),e._v("是为了使示例尽可能简单:")]),e._v(" "),r("div",{staticClass:"language- extra-class"},[r("pre",{pre:!0,attrs:{class:"language-text"}},[r("code",[e._v("public class CustomItemWriter implements ItemWriter {\n\n List output = TransactionAwareProxyFactory.createTransactionalList();\n\n public void write(List items) throws Exception {\n output.addAll(items);\n }\n\n public List getOutput() {\n return output;\n }\n}\n")])])]),r("h5",{attrs:{id:"使itemwriter重新启动"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#使itemwriter重新启动"}},[e._v("#")]),e._v(" 使"),r("code",[e._v("ItemWriter")]),e._v("重新启动")]),e._v(" "),r("p",[e._v("要使"),r("code",[e._v("ItemWriter")]),e._v("可重启,我们将遵循与"),r("code",[e._v("ItemReader")]),e._v("相同的过程,添加并实现"),r("code",[e._v("ItemStream")]),e._v("接口以同步执行上下文。在这个示例中,我们可能必须计算处理的项目的数量,并将其添加为页脚记录。如果需要这样做,我们可以在"),r("code",[e._v("ItemWriter")]),e._v("中实现"),r("code",[e._v("ItemStream")]),e._v(",这样,如果流被重新打开,计数器将从执行上下文中重新构造。")]),e._v(" "),r("p",[e._v("在许多实际的情况下,自定义"),r("code",[e._v("ItemWriters")]),e._v("也会委托给另一个本身是可重启的编写器(例如,当写到文件时),或者它会写到事务资源,因此不需要重启,因为它是无状态的。当你有一个有状态的编写器时,你可能应该确保实现"),r("code",[e._v("ItemStream")]),e._v("以及"),r("code",[e._v("ItemWriter")]),e._v("。还请记住,Writer 的客户机需要知道"),r("code",[e._v("ItemStream")]),e._v(",因此你可能需要在配置中将其注册为流。")]),e._v(" "),r("h3",{attrs:{id:"项读取器和编写器实现"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#项读取器和编写器实现"}},[e._v("#")]),e._v(" 项读取器和编写器实现")]),e._v(" "),r("p",[e._v("在本节中,我们将向你介绍在前几节中尚未讨论过的读者和作者。")]),e._v(" "),r("h4",{attrs:{id:"装饰者"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#装饰者"}},[e._v("#")]),e._v(" 装饰者")]),e._v(" "),r("p",[e._v("在某些情况下,用户需要将专门的行为附加到预先存在的"),r("code",[e._v("ItemReader")]),e._v("。 Spring Batch 提供了一些开箱即用的装饰器,它们可以将额外的行为添加到你的"),r("code",[e._v("ItemReader")]),e._v("和"),r("code",[e._v("ItemWriter")]),e._v("实现中。")]),e._v(" "),r("p",[e._v("Spring 批处理包括以下装饰器:")]),e._v(" "),r("ul",[r("li",[r("p",[e._v("["),r("code",[e._v("SynchronizedItemStreamReader")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("SingleItemPeekableItemReader")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("SynchronizedItemStreamWriter")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("MultiResourceItemWriter")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("ClassifierCompositeItemWriter")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("ClassifierCompositeItemProcessor")]),e._v("]")])])]),e._v(" "),r("h5",{attrs:{id:"synchronizeditemstreamreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#synchronizeditemstreamreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("SynchronizedItemStreamReader")])]),e._v(" "),r("p",[e._v("当使用不是线程安全的"),r("code",[e._v("ItemReader")]),e._v("时, Spring Batch 提供"),r("code",[e._v("SynchronizedItemStreamReader")]),e._v("decorator,该 decorator 可用于使"),r("code",[e._v("ItemReader")]),e._v("线程安全。 Spring 批处理提供了一个"),r("code",[e._v("SynchronizedItemStreamReaderBuilder")]),e._v("来构造"),r("code",[e._v("SynchronizedItemStreamReader")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"singleitempeekableitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#singleitempeekableitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("SingleItemPeekableItemReader")])]),e._v(" "),r("p",[e._v("Spring 批处理包括向"),r("code",[e._v("ItemReader")]),e._v("添加 PEEK 方法的装饰器。这种 peek 方法允许用户提前查看一项。对 Peek 的重复调用返回相同的项,这是从"),r("code",[e._v("read")]),e._v("方法返回的下一个项。 Spring 批处理提供了一个"),r("code",[e._v("SingleItemPeekableItemReaderBuilder")]),e._v("来构造"),r("code",[e._v("SingleItemPeekableItemReader")]),e._v("的实例。")]),e._v(" "),r("table",[r("thead",[r("tr",[r("th"),e._v(" "),r("th",[e._v("SingleitemPeekableitemreader 的 Peek 方法不是线程安全的,因为它不可能"),r("br"),e._v("在多个线程中执行 Peek。窥视"),r("br"),e._v("的线程中只有一个会在下一次调用中获得要读取的项。")])])]),e._v(" "),r("tbody")]),e._v(" "),r("h5",{attrs:{id:"synchronizeditemstreamwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#synchronizeditemstreamwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("SynchronizedItemStreamWriter")])]),e._v(" "),r("p",[e._v("当使用不是线程安全的"),r("code",[e._v("ItemWriter")]),e._v("时, Spring Batch 提供"),r("code",[e._v("SynchronizedItemStreamWriter")]),e._v("decorator,该 decorator 可用于使"),r("code",[e._v("ItemWriter")]),e._v("线程安全。 Spring 批处理提供了一个"),r("code",[e._v("SynchronizedItemStreamWriterBuilder")]),e._v("来构造"),r("code",[e._v("SynchronizedItemStreamWriter")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"multiresourceitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#multiresourceitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("MultiResourceItemWriter")])]),e._v(" "),r("p",[e._v("当当前资源中写入的项数超过"),r("code",[e._v("itemCountLimitPerResource")]),e._v("时,"),r("code",[e._v("MultiResourceItemWriter")]),e._v("包装一个"),r("code",[e._v("ResourceAwareItemWriterItemStream")]),e._v("并创建一个新的输出资源。 Spring 批处理提供了一个"),r("code",[e._v("MultiResourceItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("MultiResourceItemWriter")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"classifiercompositeitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#classifiercompositeitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("ClassifierCompositeItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("ClassifierCompositeItemWriter")]),e._v("调用用于每个项的"),r("code",[e._v("ItemWriter")]),e._v("实现的集合之一,该实现基于通过提供的"),r("code",[e._v("Classifier")]),e._v("实现的路由器模式。如果所有委托都是线程安全的,则实现是线程安全的。 Spring 批处理提供了一个"),r("code",[e._v("ClassifierCompositeItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("ClassifierCompositeItemWriter")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"classifiercompositeitemprocessor"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#classifiercompositeitemprocessor"}},[e._v("#")]),e._v(" "),r("code",[e._v("ClassifierCompositeItemProcessor")])]),e._v(" "),r("p",[r("code",[e._v("ClassifierCompositeItemProcessor")]),e._v("是一个"),r("code",[e._v("ItemProcessor")]),e._v(",它调用"),r("code",[e._v("ItemProcessor")]),e._v("实现的集合之一,该实现基于通过所提供的"),r("code",[e._v("Classifier")]),e._v("实现的路由器模式。 Spring 批处理提供了一个"),r("code",[e._v("ClassifierCompositeItemProcessorBuilder")]),e._v("来构造"),r("code",[e._v("ClassifierCompositeItemProcessor")]),e._v("的实例。")]),e._v(" "),r("h4",{attrs:{id:"消息阅读器和消息编写器"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#消息阅读器和消息编写器"}},[e._v("#")]),e._v(" 消息阅读器和消息编写器")]),e._v(" "),r("p",[e._v("Spring Batch 为常用的消息传递系统提供了以下读取器和编写器:")]),e._v(" "),r("ul",[r("li",[r("p",[e._v("["),r("code",[e._v("AmqpItemReader")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("AmqpItemWriter")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("JmsItemReader")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("JmsItemWriter")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("KafkaItemReader")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("KafkaItemWriter")]),e._v("]")])])]),e._v(" "),r("h5",{attrs:{id:"amqpitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#amqpitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("AmqpItemReader")])]),e._v(" "),r("p",[r("code",[e._v("AmqpItemReader")]),e._v("是一个"),r("code",[e._v("ItemReader")]),e._v(",它使用"),r("code",[e._v("AmqpTemplate")]),e._v("来接收或转换来自交换的消息。 Spring 批处理提供了一个"),r("code",[e._v("AmqpItemReaderBuilder")]),e._v("来构造"),r("code",[e._v("AmqpItemReader")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"amqpitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#amqpitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("AmqpItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("AmqpItemWriter")]),e._v("是一个"),r("code",[e._v("ItemWriter")]),e._v(",它使用"),r("code",[e._v("AmqpTemplate")]),e._v("向 AMQP 交换发送消息。如果提供的"),r("code",[e._v("AmqpTemplate")]),e._v("中未指定名称,则将消息发送到无名交换机。 Spring 批处理提供了"),r("code",[e._v("AmqpItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("AmqpItemWriter")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"jmsitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#jmsitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("JmsItemReader")])]),e._v(" "),r("p",[e._v("对于使用"),r("code",[e._v("JmsTemplate")]),e._v("的 JMS,"),r("code",[e._v("ItemReader")]),e._v("是"),r("code",[e._v("ItemReader")]),e._v("。模板应该有一个默认的目标,它用于为"),r("code",[e._v("read()")]),e._v("方法提供项。 Spring 批处理提供了一个"),r("code",[e._v("JmsItemReaderBuilder")]),e._v("来构造"),r("code",[e._v("JmsItemReader")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"jmsitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#jmsitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("JmsItemWriter")])]),e._v(" "),r("p",[e._v("对于使用"),r("code",[e._v("JmsTemplate")]),e._v("的 JMS,"),r("code",[e._v("ItemWriter")]),e._v("是"),r("code",[e._v("ItemWriter")]),e._v("。模板应该有一个默认的目的地,用于在"),r("code",[e._v("write(List)")]),e._v("中发送项。 Spring 批处理提供了一个"),r("code",[e._v("JmsItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("JmsItemWriter")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"kafkaitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#kafkaitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("KafkaItemReader")])]),e._v(" "),r("p",[e._v("对于 Apache Kafka 主题,"),r("code",[e._v("KafkaItemReader")]),e._v("是"),r("code",[e._v("ItemReader")]),e._v("。可以将其配置为从同一主题的多个分区中读取消息。它在执行上下文中存储消息偏移量,以支持重新启动功能。 Spring 批处理提供了一个"),r("code",[e._v("KafkaItemReaderBuilder")]),e._v("来构造"),r("code",[e._v("KafkaItemReader")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"kafkaitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#kafkaitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("KafkaItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("KafkaItemWriter")]),e._v("是用于 Apache Kafka 的"),r("code",[e._v("ItemWriter")]),e._v(",它使用"),r("code",[e._v("KafkaTemplate")]),e._v("将事件发送到默认主题。 Spring 批处理提供了一个"),r("code",[e._v("KafkaItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("KafkaItemWriter")]),e._v("的实例。")]),e._v(" "),r("h4",{attrs:{id:"数据库阅读器"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#数据库阅读器"}},[e._v("#")]),e._v(" 数据库阅读器")]),e._v(" "),r("p",[e._v("Spring Batch 提供以下数据库阅读器:")]),e._v(" "),r("ul",[r("li",[r("p",[e._v("["),r("code",[e._v("Neo4jItemReader")]),e._v("](#NEO4jitemreader)")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("MongoItemReader")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("HibernateCursorItemReader")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("HibernatePagingItemReader")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("RepositoryItemReader")]),e._v("]")])])]),e._v(" "),r("h5",{attrs:{id:"neo4jitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#neo4jitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("Neo4jItemReader")])]),e._v(" "),r("p",[r("code",[e._v("Neo4jItemReader")]),e._v("是一个"),r("code",[e._v("ItemReader")]),e._v(",它使用分页技术从图数据库 NEO4j 中读取对象。 Spring 批处理提供了一个"),r("code",[e._v("Neo4jItemReaderBuilder")]),e._v("来构造"),r("code",[e._v("Neo4jItemReader")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"mongoitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#mongoitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("MongoItemReader")])]),e._v(" "),r("p",[r("code",[e._v("MongoItemReader")]),e._v("是一个"),r("code",[e._v("ItemReader")]),e._v(",它使用分页技术从 MongoDB 读取文档。 Spring 批处理提供了一个"),r("code",[e._v("MongoItemReaderBuilder")]),e._v("来构造"),r("code",[e._v("MongoItemReader")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"hibernatecursoritemreader-2"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#hibernatecursoritemreader-2"}},[e._v("#")]),e._v(" "),r("code",[e._v("HibernateCursorItemReader")])]),e._v(" "),r("p",[r("code",[e._v("HibernateCursorItemReader")]),e._v("是用于读取在 Hibernate 之上构建的数据库记录的"),r("code",[e._v("ItemStreamReader")]),e._v("。它执行 HQL 查询,然后在初始化时,在调用"),r("code",[e._v("read()")]),e._v("方法时对结果集进行迭代,依次返回与当前行对应的对象。 Spring 批处理提供了一个"),r("code",[e._v("HibernateCursorItemReaderBuilder")]),e._v("来构造"),r("code",[e._v("HibernateCursorItemReader")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"hibernatepagingitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#hibernatepagingitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("HibernatePagingItemReader")])]),e._v(" "),r("p",[r("code",[e._v("HibernatePagingItemReader")]),e._v("是一个"),r("code",[e._v("ItemReader")]),e._v(",用于读取建立在 Hibernate 之上的数据库记录,并且一次只读取固定数量的项。 Spring 批处理提供了一个"),r("code",[e._v("HibernatePagingItemReaderBuilder")]),e._v("来构造"),r("code",[e._v("HibernatePagingItemReader")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"repositoryitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#repositoryitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("RepositoryItemReader")])]),e._v(" "),r("p",[r("code",[e._v("RepositoryItemReader")]),e._v("是通过使用"),r("code",[e._v("PagingAndSortingRepository")]),e._v("读取记录的"),r("code",[e._v("ItemReader")]),e._v("。 Spring 批处理提供了一个"),r("code",[e._v("RepositoryItemReaderBuilder")]),e._v("来构造"),r("code",[e._v("RepositoryItemReader")]),e._v("的实例。")]),e._v(" "),r("h4",{attrs:{id:"数据库编写者"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#数据库编写者"}},[e._v("#")]),e._v(" 数据库编写者")]),e._v(" "),r("p",[e._v("Spring Batch 提供以下数据库编写器:")]),e._v(" "),r("ul",[r("li",[r("p",[e._v("["),r("code",[e._v("Neo4jItemWriter")]),e._v("](#NEO4jitemwriter)")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("MongoItemWriter")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("RepositoryItemWriter")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("HibernateItemWriter")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("JdbcBatchItemWriter")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("JpaItemWriter")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("GemfireItemWriter")]),e._v("]")])])]),e._v(" "),r("h5",{attrs:{id:"neo4jitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#neo4jitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("Neo4jItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("Neo4jItemWriter")]),e._v("是一个"),r("code",[e._v("ItemWriter")]),e._v("实现,它将写到 NEO4J 数据库。 Spring 批处理提供了一个"),r("code",[e._v("Neo4jItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("Neo4jItemWriter")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"mongoitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#mongoitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("MongoItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("MongoItemWriter")]),e._v("是一个"),r("code",[e._v("ItemWriter")]),e._v("实现,它使用 Spring data 的"),r("code",[e._v("MongoOperations")]),e._v("的实现将数据写到 MongoDB 存储。 Spring 批处理提供了一个"),r("code",[e._v("MongoItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("MongoItemWriter")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"repositoryitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#repositoryitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("RepositoryItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("RepositoryItemWriter")]),e._v("是来自 Spring 数据的"),r("code",[e._v("ItemWriter")]),e._v("包装器。 Spring 批处理提供了一个"),r("code",[e._v("RepositoryItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("RepositoryItemWriter")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"hibernateitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#hibernateitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("HibernateItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("HibernateItemWriter")]),e._v("是一个"),r("code",[e._v("ItemWriter")]),e._v(",它使用一个 Hibernate 会话来保存或更新不是当前 Hibernate 会话的一部分的实体。 Spring 批处理提供了一个"),r("code",[e._v("HibernateItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("HibernateItemWriter")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"jdbcbatchitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#jdbcbatchitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("JdbcBatchItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("JdbcBatchItemWriter")]),e._v("是一个"),r("code",[e._v("ItemWriter")]),e._v(",它使用"),r("code",[e._v("NamedParameterJdbcTemplate")]),e._v("中的批处理特性来为提供的所有项执行一批语句。 Spring 批处理提供了一个"),r("code",[e._v("JdbcBatchItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("JdbcBatchItemWriter")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"jpaitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#jpaitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("JpaItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("JpaItemWriter")]),e._v("是一个"),r("code",[e._v("ItemWriter")]),e._v(",它使用 JPA "),r("code",[e._v("EntityManagerFactory")]),e._v("来合并不属于持久性上下文的任何实体。 Spring 批处理提供了一个"),r("code",[e._v("JpaItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("JpaItemWriter")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"gemfireitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#gemfireitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("GemfireItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("GemfireItemWriter")]),e._v("是一个"),r("code",[e._v("ItemWriter")]),e._v(",它使用一个"),r("code",[e._v("GemfireTemplate")]),e._v("将项目存储在 Gemfire 中,作为键/值对。 Spring 批处理提供了一个"),r("code",[e._v("GemfireItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("GemfireItemWriter")]),e._v("的实例。")]),e._v(" "),r("h4",{attrs:{id:"专业阅读器"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#专业阅读器"}},[e._v("#")]),e._v(" 专业阅读器")]),e._v(" "),r("p",[e._v("Spring Batch 提供以下专门的阅读器:")]),e._v(" "),r("ul",[r("li",[r("p",[e._v("["),r("code",[e._v("LdifReader")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("MappingLdifReader")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("AvroItemReader")]),e._v("]")])])]),e._v(" "),r("h5",{attrs:{id:"ldifreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#ldifreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("LdifReader")])]),e._v(" "),r("p",[r("code",[e._v("AvroItemWriter")]),e._v("读取来自"),r("code",[e._v("Resource")]),e._v("的 LDIF(LDAP 数据交换格式)记录,对它们进行解析,并为执行的每个"),r("code",[e._v("LdapAttribute")]),e._v("返回一个"),r("code",[e._v("LdapAttribute")]),e._v("对象。 Spring 批处理提供了一个"),r("code",[e._v("LdifReaderBuilder")]),e._v("来构造"),r("code",[e._v("LdifReader")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"mappingldifreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#mappingldifreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("MappingLdifReader")])]),e._v(" "),r("p",[r("code",[e._v("MappingLdifReader")]),e._v("从"),r("code",[e._v("Resource")]),e._v("读取 LDIF(LDAP 数据交换格式)记录,解析它们,然后将每个 LDIF 记录映射到 POJO(普通的旧 Java 对象)。每个读都返回一个 POJO。 Spring 批处理提供了一个"),r("code",[e._v("MappingLdifReaderBuilder")]),e._v("来构造"),r("code",[e._v("MappingLdifReader")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"avroitemreader"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#avroitemreader"}},[e._v("#")]),e._v(" "),r("code",[e._v("AvroItemReader")])]),e._v(" "),r("p",[r("code",[e._v("AvroItemReader")]),e._v("从资源中读取序列化的 AVRO 数据。每个读取返回由 Java 类或 AVRO 模式指定的类型的实例。读取器可以被可选地配置为嵌入 AVRO 模式的输入或不嵌入该模式的输入。 Spring 批处理提供了一个"),r("code",[e._v("AvroItemReaderBuilder")]),e._v("来构造"),r("code",[e._v("AvroItemReader")]),e._v("的实例。")]),e._v(" "),r("h4",{attrs:{id:"专业作家"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#专业作家"}},[e._v("#")]),e._v(" 专业作家")]),e._v(" "),r("p",[e._v("Spring Batch 提供以下专业的写作人员:")]),e._v(" "),r("ul",[r("li",[r("p",[e._v("["),r("code",[e._v("SimpleMailMessageItemWriter")]),e._v("]")])]),e._v(" "),r("li",[r("p",[e._v("["),r("code",[e._v("AvroItemWriter")]),e._v("]")])])]),e._v(" "),r("h5",{attrs:{id:"simplemailmessageitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#simplemailmessageitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("SimpleMailMessageItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("SimpleMailMessageItemWriter")]),e._v("是可以发送邮件的"),r("code",[e._v("ItemWriter")]),e._v("。它将消息的实际发送委托给"),r("code",[e._v("MailSender")]),e._v("的实例。 Spring 批处理提供了一个"),r("code",[e._v("SimpleMailMessageItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("SimpleMailMessageItemWriter")]),e._v("的实例。")]),e._v(" "),r("h5",{attrs:{id:"avroitemwriter"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#avroitemwriter"}},[e._v("#")]),e._v(" "),r("code",[e._v("AvroItemWriter")])]),e._v(" "),r("p",[r("code",[e._v("AvroItemWrite")]),e._v("根据给定的类型或模式将 Java 对象序列化到一个 WriteableResource。编写器可以被可选地配置为在输出中嵌入或不嵌入 AVRO 模式。 Spring 批处理提供了一个"),r("code",[e._v("AvroItemWriterBuilder")]),e._v("来构造"),r("code",[e._v("AvroItemWriter")]),e._v("的实例。")]),e._v(" "),r("h4",{attrs:{id:"专用处理器"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#专用处理器"}},[e._v("#")]),e._v(" 专用处理器")]),e._v(" "),r("p",[e._v("Spring Batch 提供以下专门的处理器:")]),e._v(" "),r("ul",[r("li",[e._v("["),r("code",[e._v("ScriptItemProcessor")]),e._v("]")])]),e._v(" "),r("h5",{attrs:{id:"scriptitemprocessor"}},[r("a",{staticClass:"header-anchor",attrs:{href:"#scriptitemprocessor"}},[e._v("#")]),e._v(" "),r("code",[e._v("ScriptItemProcessor")])]),e._v(" "),r("p",[r("code",[e._v("ScriptItemProcessor")]),e._v("是一个"),r("code",[e._v("ItemProcessor")]),e._v(",它将当前项目传递给提供的脚本,并且该脚本的结果将由处理器返回。 Spring 批处理提供了一个"),r("code",[e._v("ScriptItemProcessorBuilder")]),e._v("来构造"),r("code",[e._v("ScriptItemProcessor")]),e._v("的实例。")])])}),[],!1,null,null,null);t.default=n.exports}}]);