<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
日常開發過程中,尤其在 DDD 過程中,經常遇到 VO/MODEL/PO 等領域模型的相互轉換。此時我們會一個欄位一個欄位進行 set|get 設定。要麼使用工具類進行暴力的屬性拷貝,在這個暴力屬性拷貝過程中好的工具更能提高程式的執行效率,反之引起效能低下、隱藏細節設定 OOM 等極端情況出現。
本擴充套件元件基於 mapstruct 進行擴充套件,簡單介紹 mapstruct 實現原理。
mapstruct 是基於 JSR 269 實現的,JSR 269 是 JDK 引進的一種規範。有了它,能夠實現在編譯期處理註解,並且讀取、修改和新增抽象語法樹中的內容。JSR 269 使用 Annotation Processor 在編譯期間處理註解,Annotation Processor 相當於編譯器的一種外掛,因此又稱為插入式註解處理。
我們知道,java 的類載入機制是需要通過編譯期執行期。如下圖所示
mapstruct 正是在上面的編譯期編譯原始碼的過程中,通過修改語法樹二次生成位元組碼,如下圖所示
以上大概可以概括如下幾個步驟:
1、生成抽象語法樹。Java 編譯器對 Java 原始碼進行編譯,生成抽象語法樹(Abstract Syntax Tree,AST)。
2、呼叫實現了 JSR 269 API 的程式。只要程式實現了 JSR 269 API,就會在編譯期間呼叫實現的註解處理器。
3、修改抽象語法樹。在實現 JSR 269 API 的程式中,可以修改抽象語法樹,插入自己的實現邏輯。
4、生成位元組碼。修改完抽象語法樹後,Java 編譯器會生成修改後的抽象語法樹對應的位元組碼檔案件。
從 mapstruct 實現原理來看,我們發現 mapstruct 屬性轉換邏輯清晰,具備良好的擴充套件性,問題是需要單獨寫一層轉換介面或者新增一個轉換方法。能否將轉換介面或者方法做到自動擴充套件呢?
上面所說 mapstruct 方案,有個弊端。就是如果有新的領域模型轉換,我們不得不手動寫一層轉換介面,如果出現 A/B 兩個模型互轉,一般需定義四個方法:
鑑於此,本方案通過將原 mapstruct 定義在轉換介面類註解和轉換方法的註解,通過對映,形成新包裝註解。將此註解直接定義在模型的類或者欄位上,然後根據模型上的自定義註解直接編譯期生成轉換介面,然後 mapstruct 根據自動生成的介面再次生成具體的轉換實現類。
注意:自動生成的介面中類和方法的註解為原 mapstruct 的註解,所以 mapstruct 原有功能上沒有丟失。詳細調整如下圖:
1)繼承 AbstractProcessor 類,並且重寫 process 方法,在 process 方法中實現自己的註解處理邏輯。
2)在 META-INF/services 目錄下建立 javax.annotation.processing.Processor 檔案註冊自己實現的
知識點: 使用 AutoService 的好處是幫助我們不需要手動維護 Annotation Processor 所需要的 META-INF 檔案目錄和檔案內容。它會自動幫我們生產,使用方法也很簡單,只需要在自定義的 Annotation Processor 類上加上以下的註解即可 @AutoService (Processor.class)
JavaPoet 是一款可以自動生成 Java 檔案的第三方依賴。
簡潔易懂的 API,上手快。
讓繁雜、重複的 Java 檔案,自動化生成,提高工作效率,簡化流程。
1) AlpacaMap:定義在類上,屬性 target 指定所轉換目標模型;屬性 uses 指定雷專轉換過程中所依賴的外部物件。
2)AlpacaMapField:原始 mapstruct 所支援的所有註解做一次別名包裝,使用 spring 提供的 AliasFor 註解。
知識點: @AliasFor 是 Spring 框架的一個註解,用於宣告註解屬性的別名。它有兩種不同的應用場景:
註解內的別名
後設資料的別名
兩者主要的區別在於是否在同一個註解內。
AutoMapFieldDescriptor descriptor = new AutoMapFieldDescriptor(); descriptor.target = fillString(alpacaMapField.target()); descriptor.dateFormat = fillString(alpacaMapField.dateFormat()); descriptor.numberFormat = fillString(alpacaMapField.numberFormat()); descriptor.constant = fillString(alpacaMapField.constant()); descriptor.expression = fillString(alpacaMapField.expression()); descriptor.defaultExpression = fillString(alpacaMapField.defaultExpression()); descriptor.ignore = alpacaMapField.ignore(); ..........
生成類資訊:TypeSpec createTypeSpec(AlpacaMapMapperDescriptor descriptor)
生成類註解資訊 AnnotationSpec buildGeneratedMapperConfigAnnotationSpec(AlpacaMapMapperDescriptor descriptor) {
生成類方法資訊: MethodSpec buildMappingMethods(AlpacaMapMapperDescriptor descriptor)
生成方法註解資訊:List<AnnotationSpec> buildMethodMappingAnnotations(AlpacaMapMapperDescriptor descriptor){
在實現生成類資訊過程中,需要指定生成類的介面類 AlpacaBaseAutoAssembler,此類主要定義四個方法如下:
public interface AlpacaBaseAutoAssembler<S,T>{ T copy(S source); default List<T> copyL(List<S> sources){ return sources.stream().map(c->copy(c)).collect(Collectors.toList()); } @InheritInverseConfiguration(name = "copy") S reverseCopy(T source); default List<S> reverseCopyL(List<T> sources){ return sources.stream().map(c->reverseCopy(c)).collect(Collectors.toList()); } }
private AnnotationSpec buildGeneratedMapperConfigAnnotationSpec() { return AnnotationSpec.builder(ClassName.get("org.mapstruct", "MapperConfig")) .addMember("componentModel", "$S", "spring") .build(); }
private void writeAutoMapperClassFile(AlpacaMapMapperDescriptor descriptor){ System.out.println("開始生成介面:"+descriptor.sourcePackageName() + "."+ descriptor.mapperName()); try (final Writer outputWriter = processingEnv .getFiler() .createSourceFile( descriptor.sourcePackageName() + "."+ descriptor.mapperName()) .openWriter()) { alpacaMapMapperGenerator.write(descriptor, outputWriter); } catch (IOException e) { processingEnv .getMessager() .printMessage( ERROR, "Error while opening "+ descriptor.mapperName() + " output file: " + e.getMessage()); } }
知識點: 在 javapoet 中核心類第一大概有一下幾個類,可參考如下:
JavaFile 用於構造輸出包含一個頂級類的 Java 檔案,是對.java 檔案的抽象定義
TypeSpec TypeSpec 是類 / 介面 / 列舉的抽象型別
MethodSpec MethodSpec 是方法 / 建構函式的抽象定義
FieldSpec FieldSpec 是成員變數 / 欄位的抽象定義
ParameterSpec ParameterSpec 用於建立方法引數
AnnotationSpec AnnotationSpec 用於建立標記註解
下面舉例說明如何使用,在這裡我們定義一個模型 Person 和模型 Student,其中涉及欄位轉換的普通字串、列舉、時間格式化和複雜的型別換磚,具體運用如下步驟。
程式碼已上傳程式碼庫,如需特定需求可重新拉去分支打包使用
<dependency> <groupId>com.jdl</groupId> <artifactId>alpaca-mapstruct-processor</artifactId> <version>1.1-SNAPSHOT</version> </dependency>
uses 方法必須為正常的 spring 容器中的 bean,此 bean 提供 @Named 註解的方法可供類欄位註解 AlpacaMapField 中的 qualifiedByName 屬性以字串的方式指定,如下圖所示
@Data @AlpacaMap(targetType = Student.class,uses = {Person.class}) @Service public class Person { private String make; private SexType type; @AlpacaMapField(target = "age") private Integer sax; @AlpacaMapField(target="dateStr" ,dateFormat = "yyyy-MM-dd") private Date date; @AlpacaMapField(target = "brandTypeName",qualifiedByName ="convertBrandTypeName") private Integer brandType; @Named("convertBrandTypeName") public String convertBrandTypeName(Integer brandType){ return BrandTypeEnum.getDescByValue(brandType); } @Named("convertBrandTypeName") public Integer convertBrandType(String brandTypeName){ return BrandTypeEnum.getValueByDesc(brandTypeName); } }
使用 maven 打包或者編譯後觀察,此時在 target/generated-source/annotatins 目錄中生成兩個檔案 PersonToStudentAssembler 和 PersonToStudentAssemblerImpl
類檔案 PersonToStudentAssembler 是由自定義註解器自動生成,內容如下
@Mapper( config = AutoMapSpringConfig.class, uses = {Person.class} ) public interface PersonToStudentAssembler extends AlpacaBaseAutoAssembler<Person, Student> { @Override @Mapping( target = "age", source = "sax", ignore = false ) @Mapping( target = "dateStr", dateFormat = "yyyy-MM-dd", source = "date", ignore = false ) @Mapping( target = "brandTypeName", source = "brandType", ignore = false, qualifiedByName = "convertBrandTypeName" ) Student copy(final Person source); }
PersonToStudentAssemblerImpl 是 mapstruct 根據 PersonToStudentAssembler 介面註解器自動生成,內容如下
@Component public class PersonToStudentAssemblerImpl implements PersonToStudentAssembler { @Autowired private Person person; @Override public Person reverseCopy(Student arg0) { if ( arg0 == null ) { return null; } Person person = new Person(); person.setSax( arg0.getAge() ); try { if ( arg0.getDateStr() != null ) { person.setDate( new SimpleDateFormat( "yyyy-MM-dd" ).parse( arg0.getDateStr() ) ); } } catch ( ParseException e ) { throw new RuntimeException( e ); } person.setBrandType( person.convertBrandType( arg0.getBrandTypeName() ) ); person.setMake( arg0.getMake() ); person.setType( arg0.getType() ); return person; } @Override public Student copy(Person source) { if ( source == null ) { return null; } Student student = new Student(); student.setAge( source.getSax() ); if ( source.getDate() != null ) { student.setDateStr( new SimpleDateFormat( "yyyy-MM-dd" ).format( source.getDate() ) ); } student.setBrandTypeName( person.convertBrandTypeName( source.getBrandType() ) ); student.setMake( source.getMake() ); student.setType( source.getType() ); return student; } }
此時在我們的 spring 容器中可直接 @Autowired 引入介面 PersonToStudentAssembler 範例進行四種維護資料相互轉換
AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(); applicationContext.scan("com.jdl.alpaca.mapstruct"); applicationContext.refresh(); PersonToStudentAssembler personToStudentAssembler = applicationContext.getBean(PersonToStudentAssembler.class); Person person = new Person(); person.setMake("make"); person.setType(SexType.BOY); person.setSax(100); person.setDate(new Date()); person.setBrandType(1); Student student = personToStudentAssembler.copy(person); System.out.println(student); System.out.println(personToStudentAssembler.reverseCopy(student)); List<Person> personList = Lists.newArrayList(); personList.add(person); System.out.println(personToStudentAssembler.copyL(personList)); System.out.println(personToStudentAssembler.reverseCopyL(personToStudentAssembler.copyL(personList)));
控制檯列印:
personToStudentStudent(make=make, type=BOY, age=100, dateStr=2022-11-09, brandTypeName=集團KA)
studentToPersonPerson(make=make, type=BOY, sax=100, date=Wed Nov 09 00:00:00 CST 2022, brandType=1)
personListToStudentList[Student(make=make, type=BOY, age=100, dateStr=2022-11-09, brandTypeName=集團KA)]
studentListToPersonList[Person(make=make, type=BOY, sax=100, date=Wed Nov 09 00:00:00 CST 2022, brandType=1)]
注意:
@InheritInverseConfiguration(name = "copy")
比如從 S 轉換 T 會使用第一個方法,從 T 轉 S 的時候必須定義一個同名 Named 註解的方法,方法引數和前面方法是入參變出參、出參變入參。
@Named("convertBrandTypeName") public String convertBrandTypeName(Integer brandType){ return BrandTypeEnum.getDescByValue(brandType); } @Named("convertBrandTypeName") public Integer convertBrandType(String brandTypeName){ return BrandTypeEnum.getValueByDesc(brandTypeName); }
知識點:
InheritInverseConfiguration 功能很強大,可以逆向對映,從上面 PersonToStudentAssemblerImpl 看到上面屬性 sax 可以正對映到 sex,逆對映可自動從 sex 對映到 sax。但是正對映的 @Mapping#expression、#defaultExpression、#defaultValue 和 #constant 會被逆對映忽略。此外某個欄位的逆對映可以被 ignore,expression 或 constant 覆蓋
參考檔案:
以上就是AbstractProcessor擴充套件MapStruct自動生成實體對映工具類的詳細內容,更多關於AbstractProcessor MapStruct的資料請關注it145.com其它相關文章!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45