日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Java 如何優雅的拷貝對象屬性

瀏覽:17日期:2022-08-20 18:52:01

場景

在 Java 項目中,經常遇到需要在對象之間拷貝屬性的問題。然而,除了直接使用 Getter/Stter 方法,我們還有其他的方法么?當然有,例如 Apache Common Lang3 的 BeanUtils,然而 BeanUtils 卻無法完全滿足吾輩的需求,所以吾輩便自己封裝了一個,這里分享出來以供參考。

需要大量復制對象的屬性 對象之間的屬性名可能是不同的 對象之間的屬性類型可能是不同的

目標

簡單易用的 API

copy: 指定需要拷貝的源對象和目標對象 prop: 拷貝指定對象的字段 props: 拷貝指定對象的多個字段 exec: 執行真正的拷貝操作 from: 重新開始添加其他對象的屬性 get: 返回當前的目標對象 config: 配置拷貝的一些策略

思路

定義門面類 BeanCopyUtil 用以暴露出一些 API 定義每個字段的操作類 BeanCopyField,保存對每個字段的操作 定義 BeanCopyConfig,用于配置拷貝屬性的策略 定義 BeanCopyOperator 作為拷貝的真正實現

圖解

Java 如何優雅的拷貝對象屬性

實現

注:反射部分依賴于 joor, JDK1.8 請使用 joor-java-8

定義門面類 BeanCopyUtil 用以暴露出一些 API

/** * java bean 復制操作的工具類 * * @author rxliuli */public class BeanCopyUtil<F, T> { /** * 源對象 */ private final F from; /** * 目標對象 */ private final T to; /** * 拷貝的字段信息列表 */ private final List<BeanCopyField> copyFieldList = new LinkedList<>(); /** * 配置信息 */ private BeanCopyConfig config = new BeanCopyConfig(); private BeanCopyUtil(F from, T to) { this.from = from; this.to = to; } /** * 指定需要拷貝的源對象和目標對象 * * @param from 源對象 * @param to 目標對象 * @param <F> 源對象類型 * @param <T> 目標對象類型 * @return 一個 {@link BeanCopyUtil} 對象 */ public static <F, T> BeanCopyUtil<F, T> copy(F from, T to) { return new BeanCopyUtil<>(from, to); } /** * 拷貝指定對象的字段 * * @param fromField 源對象中的字段名 * @param toField 目標對象中的字段名 * @param converter 將源對象中字段轉換為目標對象字段類型的轉換器 * @return 返回 {@code this} */ public BeanCopyUtil<F, T> prop(String fromField, String toField, Function<? super Object, ? super Object> converter) { copyFieldList.add(new BeanCopyField(fromField, toField, converter)); return this; } /** * 拷貝指定對象的字段 * * @param fromField 源對象中的字段名 * @param toField 目標對象中的字段名 * @return 返回 {@code this} */ public BeanCopyUtil<F, T> prop(String fromField, String toField) { return prop(fromField, toField, null); } /** * 拷貝指定對象的字段 * * @param field 源對象中與目標對象中的字段名 * @param converter 將源對象中字段轉換為目標對象字段類型的轉換器 * @return 返回 {@code this} */ public BeanCopyUtil<F, T> prop(String field, Function<? super Object, ? super Object> converter) { return prop(field, field, converter); } /** * 拷貝指定對象的字段 * * @param field 源對象中與目標對象中的字段名 * @return 返回 {@code this} */ public BeanCopyUtil<F, T> prop(String field) { return prop(field, field, null); } /** * 拷貝指定對象的多個字段 * * @param fields 源對象中與目標對象中的多個字段名 * @return 返回 {@code this} */ public BeanCopyUtil<F, T> props(String... fields) { for (String field : fields) { prop(field); } return this; } /** * 執行真正的拷貝操作 * * @return 返回 {@code this} */ public BeanCopyUtil<F, T> exec() { new BeanCopyOperator<>(from, to, copyFieldList, config).copy(); return this; } /** * 重新開始添加其他對象的屬性 * 用于在執行完 {@link #exec()} 之后還想復制其它對象的屬性 * * @param from 源對象 * @param <R> 源對象類型 * @return 一個新的 {@link BeanCopyUtil} 對象 */ public <R> BeanCopyUtil<R, T> from(R from) { return new BeanCopyUtil<>(from, to); } /** * 返回當前的目標對象 * * @return 當前的目標對象 */ public T get() { return to; } /** * 配置拷貝的一些策略 * * @param config 拷貝配置對象 * @return 返回 {@code this} */ public BeanCopyUtil<F, T> config(BeanCopyConfig config) { this.config = config; return this; }}

定義每個字段的操作類 BeanCopyField,保存對每個字段的操作

/** * 拷貝屬性的每一個字段的選項 * * @author rxliuli */public class BeanCopyField { private String from; private String to; private Function<? super Object, ? super Object> converter; public BeanCopyField() { } public BeanCopyField(String from, String to, Function<? super Object, ? super Object> converter) { this.from = from; this.to = to; this.converter = converter; } public String getFrom() { return from; } public BeanCopyField setFrom(String from) { this.from = from; return this; } public String getTo() { return to; } public BeanCopyField setTo(String to) { this.to = to; return this; } public Function<? super Object, ? super Object> getConverter() { return converter; } public BeanCopyField setConverter(Function<? super Object, ? super Object> converter) { this.converter = converter; return this; }}

定義 BeanCopyConfig,用于配置拷貝屬性的策略

/** * 拷貝屬性的配置 * * @author rxliuli */public class BeanCopyConfig { /** * 同名的字段自動復制 */ private boolean same = true; /** * 覆蓋同名的字段 */ private boolean override = true; /** * 忽略 {@code null} 的源對象屬性 */ private boolean ignoreNull = true; /** * 嘗試進行自動轉換 */ private boolean converter = true; public BeanCopyConfig() { } public BeanCopyConfig(boolean same, boolean override, boolean ignoreNull, boolean converter) { this.same = same; this.override = override; this.ignoreNull = ignoreNull; this.converter = converter; } public boolean isSame() { return same; } public BeanCopyConfig setSame(boolean same) { this.same = same; return this; } public boolean isOverride() { return override; } public BeanCopyConfig setOverride(boolean override) { this.override = override; return this; } public boolean isIgnoreNull() { return ignoreNull; } public BeanCopyConfig setIgnoreNull(boolean ignoreNull) { this.ignoreNull = ignoreNull; return this; } public boolean isConverter() { return converter; } public BeanCopyConfig setConverter(boolean converter) { this.converter = converter; return this; }}

定義 BeanCopyOperator 作為拷貝的真正實現

/** * 真正執行 copy 屬性的類 * * @author rxliuli */public class BeanCopyOperator<F, T> { private static final Logger log = LoggerFactory.getLogger(BeanCopyUtil.class); private final F from; private final T to; private final BeanCopyConfig config; private List<BeanCopyField> copyFieldList; public BeanCopyOperator(F from, T to, List<BeanCopyField> copyFieldList, BeanCopyConfig config) { this.from = from; this.to = to; this.copyFieldList = copyFieldList; this.config = config; } public void copy() { //獲取到兩個對象所有的屬性 final Map<String, Reflect> fromFields = Reflect.on(from).fields(); final Reflect to = Reflect.on(this.to); final Map<String, Reflect> toFields = to.fields(); //過濾出所有相同字段名的字段并進行拷貝 if (config.isSame()) { final Map<ListUtil.ListDiffState, List<String>> different = ListUtil.different(new ArrayList<>(fromFields.keySet()), new ArrayList<>(toFields.keySet())); copyFieldList = Stream.concat(different.get(ListUtil.ListDiffState.common).stream() .map(s -> new BeanCopyField(s, s, null)), copyFieldList.stream()) .collect(Collectors.toList()); } //根據拷貝字段列表進行拷貝 copyFieldList.stream()//忽略空值.filter(beanCopyField -> !config.isIgnoreNull() || fromFields.get(beanCopyField.getFrom()).get() != null)//覆蓋屬性.filter(beanCopyField -> config.isOverride() || toFields.get(beanCopyField.getTo()).get() == null)//如果沒有轉換器,則使用默認的轉換器.peek(beanCopyField -> { if (beanCopyField.getConverter() == null) { beanCopyField.setConverter(Function.identity()); }}).forEach(beanCopyField -> { final String fromField = beanCopyField.getFrom(); final F from = fromFields.get(fromField).get(); final String toField = beanCopyField.getTo(); try { to.set(toField, beanCopyField.getConverter().apply(from)); } catch (ReflectException e) { log.warn('Copy field failed, from {} to {}, exception is {}', fromField, toField, e.getMessage()); }}); }}

使用

使用流程圖

Java 如何優雅的拷貝對象屬性

測試

代碼寫完了,讓我們測試一下!

public class BeanCopyUtilTest { private final Logger log = LoggerFactory.getLogger(getClass()); private Student student; private Teacher teacher; @Before public void before() { student = new Student('琉璃', 10, '女', 4); teacher = new Teacher(); } @Test public void copy() { //簡單的復制(類似于 BeanUtils.copyProperties) BeanCopyUtil.copy(student, teacher).exec(); log.info('teacher: {}', teacher); assertThat(teacher).extracting('age').containsOnlyOnce(student.getAge()); } @Test public void prop() { //不同名字的屬性 BeanCopyUtil.copy(student, teacher).prop('sex', 'sex', sex -> Objects.equals(sex, '男')).prop('realname', 'name').exec(); assertThat(teacher).extracting('name', 'age', 'sex').containsOnlyOnce(student.getRealname(), student.getAge(), false); } @Test public void prop1() { //不存的屬性 assertThat(BeanCopyUtil.copy(student, teacher).prop('sex', 'sex', sex -> Objects.equals(sex, '男')).prop('realname', 'name2').exec().get()).extracting('age', 'sex').containsOnlyOnce(student.getAge(), false); } @Test public void from() { final Teacher lingMeng = new Teacher().setName('靈夢').setAge(17); //測試 from 是否覆蓋 assertThat(BeanCopyUtil.copy(student, teacher).prop('sex', 'sex', sex -> Objects.equals(sex, '男')).prop('realname', 'name').exec().from(lingMeng).exec().get()).extracting('name', 'age', 'sex').containsOnlyOnce(lingMeng.getName(), lingMeng.getAge(), false); } @Test public void get() { //測試 get 是否有效 assertThat(BeanCopyUtil.copy(student, teacher).prop('sex', 'sex', sex -> Objects.equals(sex, '男')).prop('realname', 'name').exec().get()).extracting('name', 'age', 'sex').containsOnlyOnce(student.getRealname(), student.getAge(), false); } @Test public void config() { //不自動復制同名屬性 assertThat(BeanCopyUtil.copy(new Student().setAge(15), new Teacher()).config(new BeanCopyConfig().setSame(false)).exec().get()).extracting('age').containsOnlyNulls(); //不覆蓋不為空的屬性 assertThat(BeanCopyUtil.copy(new Student().setAge(15), new Teacher().setAge(10)).config(new BeanCopyConfig().setOverride(false)).exec().get()).extracting('age').containsOnlyOnce(10); //不忽略源對象不為空的屬性 assertThat(BeanCopyUtil.copy(new Student(), student).config(new BeanCopyConfig().setIgnoreNull(false)).exec().get()).extracting('realname', 'age', 'sex', 'grade').containsOnlyNulls(); } /** * 測試學生類 */ private static class Student { /** * 姓名 */ private String realname; /** * 年齡 */ private Integer age; /** * 性別,男/女 */ private String sex; /** * 年級,1 - 6 */ private Integer grade; public Student() { } public Student(String realname, Integer age, String sex, Integer grade) { this.realname = realname; this.age = age; this.sex = sex; this.grade = grade; } public String getRealname() { return realname; } public Student setRealname(String realname) { this.realname = realname; return this; } public Integer getAge() { return age; } public Student setAge(Integer age) { this.age = age; return this; } public String getSex() { return sex; } public Student setSex(String sex) { this.sex = sex; return this; } public Integer getGrade() { return grade; } public Student setGrade(Integer grade) { this.grade = grade; return this; } @Override public String toString() { return ToStringBuilder.reflectionToString(this); } } /** * 測試教師類 */ private static class Teacher { /** * 姓名 */ private String name; /** * 年齡 */ private Integer age; /** * 性別,true 男,false 女 */ private Boolean sex; /** * 職位 */ private String post; public Teacher() { } public Teacher(String name, Integer age, Boolean sex, String post) { this.name = name; this.age = age; this.sex = sex; this.post = post; } public String getName() { return name; } public Teacher setName(String name) { this.name = name; return this; } public Integer getAge() { return age; } public Teacher setAge(Integer age) { this.age = age; return this; } public Boolean getSex() { return sex; } public Teacher setSex(Boolean sex) { this.sex = sex; return this; } public String getPost() { return post; } public Teacher setPost(String post) { this.post = post; return this; } @Override public String toString() { return ToStringBuilder.reflectionToString(this); } }}

如果沒有發生什么意外,那么一切將能夠正常運行!

好了,那么關于在 Java 中優雅的拷貝對象屬性就到這里啦

以上就是Java 如何優雅的拷貝對象屬性的詳細內容,更多關于Java 拷貝對象屬性的資料請關注好吧啦網其它相關文章!

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
91成人精品在线| 欧美福利在线| 中文一区一区三区免费在线观| 亚洲www啪成人一区二区| 日韩在线综合| 日韩久久精品| 波多野结衣一区| 亚洲精品1区2区| 视频一区中文字幕精品| 97久久亚洲| 美女视频网站久久| 国产日韩一区二区三免费高清| 日韩动漫一区| 国产精品香蕉| 黑森林国产精品av| 99精品电影| 亚洲激情二区| 日韩欧美中文字幕一区二区三区| 国产日产一区| 91日韩免费| 99香蕉国产精品偷在线观看| 视频一区国产视频| 久久激五月天综合精品| 国产在线一区不卡| 久久蜜桃av| 亚洲久久视频| 国产福利一区二区精品秒拍| 久久xxx视频| 日韩精品免费一区二区三区| 日韩中文字幕av电影| 国产美女视频一区二区| 国产一区二区亚洲| 91精品精品| 亚洲一区二区三区无吗| 国产欧美精品| 999精品色在线播放| 人人爽香蕉精品| 国产欧美久久一区二区三区| 亚洲爱爱视频| 日韩成人一级| 久久婷婷一区| 亚洲精品日本| 免费日韩一区二区三区| 欧美日韩中文一区二区| 日本综合精品一区| 亚洲黄色网址| 视频一区日韩精品| 日韩成人精品一区| 亚洲深夜福利| 精品女同一区二区三区在线观看| 久久久91麻豆精品国产一区| 婷婷综合激情| 国产精品115| 亚洲自啪免费| 久久三级毛片| 亚洲三级国产| 久久精品免费一区二区三区 | 日韩欧美一区二区三区免费看| 国产一级一区二区| 国产成人精选| 欧美日韩一区二区三区不卡视频 | 精品免费av在线| 97久久中文字幕| 欧美一级精品| 六月丁香综合在线视频| 蜜桃久久久久久| 久久久久免费av| 免费看久久久| 日本电影久久久| 欧美精品自拍| 亚洲国产福利| 国产精品色在线网站| 久久av在线| 成人精品天堂一区二区三区| 日韩1区2区日韩1区2区| 亚洲小说欧美另类婷婷| 国语精品一区| 国产亚洲一区| 一二三区精品| 香蕉视频成人在线观看| 日韩精品水蜜桃| 色一区二区三区四区| 国产精品一区二区免费福利视频| 妖精视频成人观看www| 日韩中文在线电影| 精品一区二区三区的国产在线观看| 三级一区在线视频先锋| 性欧美69xoxoxoxo| 色婷婷久久久| 日本午夜大片a在线观看| 日韩av黄色在线| 在线国产精品一区| 日韩午夜av| 欧美精品自拍| 99成人在线| 欧美另类综合| 婷婷激情图片久久| 午夜日韩福利| 国产一区白浆| 美女日韩在线中文字幕| 婷婷综合社区| av成人国产| 天堂成人国产精品一区| 久久国产88| 麻豆亚洲精品| 在线综合亚洲| 国产一区二区三区国产精品| 久久99国产精品视频| 欧美极品一区二区三区| 国产精品亚洲二区| 欧美精品第一区| 国产精品免费看| 免费在线观看日韩欧美| 男人的天堂亚洲一区| 亚洲影院天堂中文av色| 亚洲深夜福利在线观看| 日本在线一区二区三区| 国产欧美激情| 日韩成人a**站| 激情综合自拍| 极品裸体白嫩激情啪啪国产精品| 欧美日韩国产一区二区三区不卡 | 国产麻豆精品| 精品精品99| av在线日韩| 91久久黄色| 日本高清久久| 国产专区精品| 一区二区三区四区日本视频| 美美哒免费高清在线观看视频一区二区| 老色鬼久久亚洲一区二区| 少妇精品在线| 欧美激情日韩| 国产欧美在线观看免费| 久久免费福利| 久久久亚洲一区| 亚洲色图国产| 国产劲爆久久| 999国产精品视频| 蜜桃久久精品一区二区| 欧美久久香蕉| 国产66精品| 国产一区白浆| 国产精品久久久久久久免费软件| 91av亚洲| 亚洲三级视频| 91亚洲国产高清| 另类激情亚洲| 国产九九精品| 91精品推荐| 777久久精品| 欧美日韩在线播放视频| 欧美日韩伊人| 女人av一区| 国产精品theporn| 精品一区免费| 国产精品九九| 尤物精品在线| 麻豆精品视频在线| 免费av一区| 国产精品99精品一区二区三区∴| 激情欧美一区二区三区| 国产日韩欧美三级| 秋霞国产精品| 国产精品久久久久久模特| 亚洲激精日韩激精欧美精品| 麻豆精品在线观看| 日韩中文字幕亚洲一区二区va在线| 久久精品亚洲| 日韩精品中文字幕吗一区二区| 国产欧美一区二区三区精品酒店| 亚洲精品欧美| 亚洲一级高清| 精品三级在线观看视频| 蜜桃一区二区三区在线| 四虎4545www国产精品| 国产日韩欧美一区| 美女精品一区| 在线天堂中文资源最新版| 97se亚洲| 亚洲一区中文| 成人在线网站| 精品视频免费| 国产亚洲一区二区三区啪| 亚洲欧美日韩视频二区| 高清不卡亚洲| 精品一区不卡| 久久国产精品免费一区二区三区| 午夜一级在线看亚洲| 日韩一区三区| 成人台湾亚洲精品一区二区| 欧美日韩xxxx| 欧美精品影院| 亚洲一级大片| 亚洲欧美网站| av不卡在线看| 性欧美长视频| 首页国产欧美日韩丝袜| 91精品91| 亚洲视频播放|