《重构:改善既有代码的设计》学习总结

重构是改善代码结构的过程,旨在保持外部行为不变的同时提高可理解性和降低修改成本。文章列举了如重复代码、过长方法、过大的类等需要重构的代码特征,并提供了诸如移动函数、提炼类、引入断言等重构技巧。重构应在代码审查、添加功能或修复bug时进行,以保持代码清晰和维护性。
摘要由CSDN通过智能技术生成

目录

一、重构概述

1.1、定义

1.2、为何重构

1.3、何时重构

二、什么样的代码需要重构

2.1、重复的代码

2.2、过长的方法名

2.3、过大的类

2.4、过长参数列表

2.5、发散式变化

2.6、霰(xian)弹式修改

2.7、依恋情结

2.8、数据泥团

2.9、基本类型偏执

2.10、冗赘类

2.11、Switch语句

2.12、平行继承体系

2.13、夸夸其谈未来性

2.14、令人迷惑的暂时字段

2.15、过度耦合的消息链

2.16、中间人

2.17、亲密关系

2.18、异曲同工的类

2.19、不完美的类库

2.20、纯稚的数据类

2.21、被拒绝的遗赠

2.22、过多的注释

三、重构技巧

3.1、重新组织函数

3.1.1、引入解释性变量

3.1.2、分解临时变量

3.1.3、移除对参数的赋值

3.1.4、以卫语句取代嵌套if-else语句

3.1.5、以函数对象取代函数

3.1.6、替换算法

3.2、在对象之间搬移特性

3.2.1、移动函数

3.2.2、搬移字段

3.2.3、提炼类

3.2.4、将类内联化

3.2.5、隐藏委托关系

3.2.6、移除中间人

3.2.7、引入外加函数

3.2.8、引入本地扩展

3.3、重新组织数据

3.3.1、自封装字段

3.3.2、对象取代数据值

3.3.3、将值对象改为引用对象

3.3.4、以对象取代数组

3.3.5、将单向关联改为双向关联

3.3.6、将双向关联改为单向关联

3.3.7、封装字段

3.4、简化表达式

3.4.1、分解条件表达式

3.4.2、合并重复的条件片段

3.4.3、以卫语句取代嵌套条件表达式

3.4.4、以多态取代条件表达式

3.4.5、引入断言

3.5、简化函数调用

3.5.1、函数改名

3.5.2、添加参数

3.5.3、移除参数

3.5.4、将查询函数和修改函数分离

3.5.5、以明确函数取代参数

3.5.6、保持对象完整

3.5.7、引入参数对象

3.5.8、以异常取代错误码

3.5.9、以测试取代异常

3.6、处理概况关系

3.7、大型重构


一、重构概述

任何一个人都可以写出计算机可以理解的程序,只有写出人类容易理解的程序才是优秀的程序员

                                                                                                                                ----- Martin Fowler

1.1、定义

名词形式,对程序内部结构的一种调整,目的是在不改变程序外部行为下,提高其可理解性,降低其修改成本。

动词形式,使用一系列重构准则,在不改变程序外部行为的前提下,对代码作出修改,以改进其内部结构。

1.2、为何重构

改进程序设计,代码结构的流失是累积性的,经常性的重构可以帮助代码维持自己的形态和结构。

使程序更容易理解,重构会使代码渐趋简洁,越简洁就越容易理解,越容易理解就越容易修改。

帮助找到隐藏的Bug,重构过程中,不断深入地理解代码,弄清程序结构,可能就会发现Bug。

提高变成速度,良好的设计是维持软件开发速度的根本。

1.3、何时重构

>>>>>>>>>>>>>>>>>>>>>> 重构应该随时随地的进行 <<<<<<<<<<<<<<<<<<<<<

三次法则,类似的事情,第一次去做时尽管去做,第二次重复做变化产生方案,但是第三次再做时就应该重构了。

添加功能时重构,给代码添加新功能时,不可避免的需要修改旧代码,如果不能快速理解相关代码,就应该重构,这样下次看到更容易理解。

修补错误时重构,当收到错误或者Bug报告的时候,便是需要重构代码的信号,因为代码还不够清晰,没有清晰到一眼就能看出Bug或问题的程度。

Review代码时重构,代码Review对于编写清晰代码很重要,会让更多的人提出更好更有用的建议,对于这些建议可考虑是否可以通过重构来实现它们。

二、什么样的代码需要重构

能够识别出劣质或不友好的代码,是正确重构的前提

2.1、重复的代码

一个类中的两个方法有重复代码,可以通过抽取方法将重复的代码放到另一个方法中供调用。

互为兄弟的子类中如果有重复代码,可以将重复代码抽取到父类中,

两个没有关系的类中如果有重复代码,可以重新抽取一个类将重复代码放到这个类中。

2.2、过长的方法名

使用清晰简洁的命名,如果只是为了给方法起一个好名字而多花了一两分钟开发时间,也是值得的。

2.3、过大的类

类的设计应当遵循单一职责原则(SRP),重构一个巨大的类可以使用抽取接口的方式来搞清楚这个类应该如何拆解。

2.4、过长参数列表

比较常见的是将相关参数组织成一个对象来替换掉这些参数,比如常见开源工具类中线程池的创建或HttpClient的创建等。

2.5、发散式变化

现有一个类,如果先加入一个数据库,必须修改三个方法;如果新出现一个金融工具,必须修改另外四个方法,此时将这个类分成两个类更好。往往只有在加入新数据库或新金融工具后,才会发现这点。

2.6、霰(xian)弹式修改

每遇到变化,需要修改多个类,容易遗漏,应该把需要修改的部分放到一个类中。

发散式变化指“一个类受多种变化的影响”,霰弹式修改则指“一种变化引发多个类响应修改”。

2.7、依恋情结

函数大量地使用了另外类的数据,这种情况下最好将此函数移动到那个类中。

2.8、数据泥团

两个类中大量相同的字段、方法签名中相同的参数等,都适合提取成一个单独的数据类。

2.9、基本类型偏执

如果有大量的基本数据类型字段,就有可能将其中部分存在逻辑联系的字段(类属性或者方法参数)组织起来,形成一个类。更进一步的是,将与这些数据有关联的方法也一并移入类中。

2.10、冗赘类

如果一个类不值得或者不必存在,那么它就应该消失。比如一个类是为了处理“新冠”相关问题的,那等疫情结束了,这个类也就没有存在的必要了。

2.11、Switch语句

一套代码部署在几个环境,在项目中存在许多环境的判断(switch语句实现),不同环境走不同逻辑,如果要添加一个新环境,所有switch语句都要添加一个case。面向对象中“多态”的概念可以优雅的解决这个问题。

可以将switch语句提炼成独立的方法,然后再将此方法搬移到需要“多态”的类里。

2.12、平行继承体系

是“霰弹式修改”的一种特殊情况,每当为某个类增加一个子类,必须也为另一个类相应的增加一个子类。

看一个例子:蜡笔有大、中、小三种型号,12中颜色,那么总共必须有36中蜡笔,每增加一种颜色,都必须增加大、中、小三种型号,颜色和型号紧紧耦合在一起。再来看毛笔,不同的毛笔型号抽象成五种,不同颜色抽象成颜料,毛笔和颜料两个基类形成关联,避免了“霰弹式修改”,这就是Bridge模式。

2.13、夸夸其谈未来性

如果你的抽象类、委托、方法的参数没有实际的作用,那么就应该被移除掉。

比如某个抽象类没有太大的作用,那么就不必抽象这一层。如果方法参数完全没有用到,就应该被移除。还有方法类的唯一调用方是测试用例,应该把方法和测试用例一起删除。

2.14、令人迷惑的暂时字段

类中某个字段只为某些特殊情况而设置。

比如类中有一个复杂的算法,需要好几个变量,为了避免传递过长的参数列表,而把这些变量放到类属性里,但是这些属性只有在使用该算法时才有效,其他情况下会令人迷惑。这时可把变量和相关算法提炼到一个新的类中。

2.15、过度耦合的消息链

常常是因为数据结构的层次很深,需要层层调用getter获取内层数据。如果频繁出现,就应该考虑这个字段是否应该移动到较外层的类,或者把调用链封装在较外层类的方法。

2.16、中间人

对象的基本特征之一就是“封装”,对外部世界隐藏其内部细节,封装往往伴随着“委托”,但是可能会过度运用委托。

如果一个类的很多功能都通过委托给其他类来完成,只是多了一层简单的调用,那么就不如去掉这些中间人直接和真正负责的对象打交道。要是被委托的中间类还有很多其他行为,可以变成子类,扩展原对象的行为。

2.17、亲密关系

两个类太过亲密,比如数据访问权限互相暴露太多,就应该提炼类,将两个类的共同点提炼到新类中,让它们共同使用新类。

继承往往造成过度亲密,因为子类对超类的了解总是超过后者的主观愿望,可以运用委托来取代继承。

2.18、异曲同工的类

两个类做类似的事,抽取超类。

2.19、不完美的类库

类库的开发者没有未卜先知的能力,我们经常遇到类库没有我们需要的方法,但是我们又不能直接修改类库里的代码,这时若是只想修改一两个方法,可以引入外加方法,如果想要添加一大堆额外行为,就得引入本地扩展。

2.20、纯稚的数据类

一些数据类型不应该把全部字段单纯的通过getter/setter暴露出来,而应该暴露抽象接口,封装内部结构。

2.21、被拒绝的遗赠

子类不想继承父类的所有方法和数据,只挑选几样来使用,为子类新建一个兄弟类,再运用下移方法和下移字段把用不到的方法下推给兄弟类。

子类只复用父类的行为,却不想支持父类的接口,运用委托替代继承来达到目的。

2.22、过多的注释

一些人常用注释来补救劣质代码,事实上当重构移除所有劣质代码的时候,注释已经变得多余,因为代码已经讲清楚了一切。

三、重构技巧

Java开发,由于IDE(Intellij Idea)能够很好的支持大多数情况下的重构,有各种自动提示,所以写代码的时候要多注意提示,及时消除警告。

3.1、重新组织函数

3.1.1、引入解释性变量

将该复杂表达式的结果放进一个临时变量,以变量名来解释其用途。

3.1.2、分解临时变量

一个临时变量多次被赋值(除了“循环变量”和“结果收集变量”),应该针对每次赋值,创造独立的临时变量。

临时变量会被多次赋值,容易产生理解歧义。

3.1.3、移除对参数的赋值

以一个临时变量取代改参数的位置,对参数赋值容易降低代码的清晰度,容易混淆按值传递和按引用传递的方式。

3.1.4、以卫语句取代嵌套if-else语句

复杂嵌套的条件语句使人难以看清正常的执行路径。

3.1.5、以函数对象取代函数

一个大型函数如果包含了很多临时变量,用Extract Method很难拆解,可以把函数放到一个新创建的类中,把临时变量变成类的实体变量,再用Extract Method拆解。

3.1.6、替换算法

复杂的算法会增加维护的成本,替换成较简单的算法实现,往往能明显提高代码的可读性和可维护性。

3.2、在对象之间搬移特性

3.2.1、移动函数

类的行为做到单一职责,不要越俎代庖。如果一个类有太多行为,或一个类与另一个类有太多合作而形成高度耦合,就需要搬移函数。

3.2.2、搬移字段

如果一个类的字段在另一个类中使用更频繁,就应该考虑搬移它。

3.2.3、提炼类

一个类做了过多的工作,就应该创建一个新的类,将多余特性相关的字段和方法从源类搬移到目标类。一个类应该是一个清晰的抽象,有明确的责任。

3.2.4、将类内联化

与提炼类相反,如果一个类没有做太多事情(没有承担足够责任),就将此类所有特性搬移到另一个类中,删除原类。

3.2.5、隐藏委托关系

委托关系发生变化,变化将被限制在委托类中,使用方感知不到。

3.2.6、移除中间人

封装委托对象也是要付出代价的:每当客户要使用受托类的新特性时,就必须在服务端添加一个委托函数。随着委托类的特性(功能)越来越多,服务类完全变成了“中间人”,此时就应该让客户直接调用受托类。

很难说什么程度的隐藏才是合适的,随着系统不断变化,需要不断调整。

3.2.7、引入外加函数

你需要为提供服务的类增加一个函数,单你无法修改这个类。在客户类中建立一个函数,并以第一参数形式传入一个服务类实例。

Date nextDayDate = new Date(year, month, day + 1);

新增外部函数Date

nextDay(Date date) {return new Date(date.year, date.month, date.day + 1);}

3.2.8、引入本地扩展

你需要为服务类提供一些额外函数,但你无法修改这个类。建立一个新类,使它包含这些额外函数,让这个扩展类成为源类的子类或者包装类。

3.3、重新组织数据

3.3.1、自封装字段

为这个字段建立getter/setter函数,并且只以这些函数访问字段。

直接访问一个字段会导致出现强耦合关系。

直接反问的好处是易阅读,间接访问的好处是好管理以及子类好覆写。

3.3.2、对象取代数据值

一个数据项,需要与其他数据和行为一起使用才有意义。随着设计深入,数据之间的关系逐渐显现出来,就需要将相关数据及其操作封装成对象。

3.3.3、将值对象改为引用对象

如果希望修改某个值对象的数据,并且影响到所有引用此对象的地方,将这个值对象变成引用对象。

3.3.4、以对象取代数组

如果一个数组中的元素各自代表不同的东西,以对象替换数组,对于数组中的每个元素,以一个字段来表示。

3.3.5、将单向关联改为双向关联

两个类都需要使用对方特性,但其间只有一条单向连接。添加一个反向指针,并使修改函数能够同时更新两条连接。

3.3.6、将双向关联改为单向关联

两个类之间有双向关联,但其中一个类如今不再需要另一个类的特性,去除不必要的关联。

3.3.7、封装字段

类中存在一个public字段。将它声明为private,并提供相应的访问函数。

3.4、简化表达式

3.4.1、分解条件表达式

程序中,复杂的条件逻辑是最长导致复杂度上升的地点之一。

可以将它拆解为多个独立函数,根据每个小块代码的用途,为拆解的新函数命名,从而更清楚的表达意图。

3.4.2、合并重复的条件片段

在条件表达式的每个分支上有着相同的一段代码,将这段重复代码移到条件表达式之外。

3.4.3、以卫语句取代嵌套条件表达式

可参看3.1.4的内容

3.4.4、以多态取代条件表达式

一个条件表达式,它根据对象类型的丌(ji)同而选择丌同的行为。将这个条件表达式的每个分支放进一个子类的覆写函数中,然后将原始函数声明为抽象函数。

3.4.5、引入断言

某一段代码需要对程序状态作出某种假设,以断言明确表现出这种假设。

使用断言明确标明对输入条件的严格要求和限制。

断言可以辅助交流和调试。

3.5、简化函数调用

3.5.1、函数改名

将名字过于复杂的函数进行简化

3.5.2、添加参数

某个函数需要从调用端得到跟多信息,为此函数添加一个对象参数,让该对象带进函数所需信息。

3.5.3、移除参数

当函数不再需要某个参数时,将其移除。

3.5.4、将查询函数和修改函数分离

3.5.5、以明确函数取代参数

某个函数完全取决于参数值而采取不同行为,为了获得一个清晰的接口,针对该参数的每一个可能值,建立一个独立函数。

3.5.6、保持对象完整

如果从对象中取出若干值,将它们作为某一次函数调用时的参数,改为传递整个对象。除了可以使参数列更稳固外,还能简化参数列表,提高代码的可读性。此外,使用完整对象,被调用函数可以利用完整对象中的函数来计算某些中间值。

如果这些使对象间依赖结构恶化,就不该使用。

3.5.7、引入参数对象

如果一组参数总是一起被传递,以一个对象取代这些参数。

3.5.8、以异常取代错误码

通过异常捕获统一处理错误,而不是随处直接返回错误码,影响可读性,逻辑完整性。

3.5.9、以测试取代异常

面对调用者可以预先检查的条件,在调用函数之前应该先做检查,而不是直接捕获异常。

3.6、处理概况关系

  • 字段上移
  • 函数上移
  • 构造方法本体上移
  • 函数下移
  • 字段下移
  • 提炼子类
  • 提炼超类
  • 塑造模板方法

3.7、大型重构

在一知半解的情况下做出的设计决策,一旦堆积起来,也会使你的程序陷于瘫痪。通过重构,可以保证随时在程序中反映出完整的设计思路。

  • 梳理并分解继承体系
  • 将过程化设计转化为对象设计
  • 将领域和表述/显示分离
  • 提炼继承体系

一直很喜欢重构这本书,但是由于自己记性不太好,书看过之后其中的方法总是记不住,于是想如果有电子版的重构书就好了,工作中遇到重构的问题可以随时打开查阅。在网上搜索了许久,发现重构这本书有英文chm版本的,而中文版的电子书只有扫描的PDF版本,用起来非常不方便。于是萌生想做一本重构工具书的想法,本来打算自己重新将重构书的内容再整理归类一下,后来发现原书的目录编排就很适合做工具书,包括坏味道分类,重构手法归类等,都有了一个比较系统的整理。因此,我利用空余时间制作了这样的一本中文的chm版重构,希望对大家有所帮助,也算对中国软件业做出一点小小的贡献。 本书基本上是取自”重构”中文版一书的内容,但格式上参照的是chm英文版的格式,还有一些格式小修改,比如第一章的重构前后代码对比。因为时间匆促,个人能力有限,本书难免存在一些缺漏,如果大家发现有问题,随时可以给我发邮件,我会尽快更新错误的内容。 最后再次感谢几位大师 Martin Fowler、Kent Beck等,还有翻译的侯捷和熊节先生,为我们带来这么精彩的一本书。谢谢。 免责声明:本书仅供个人学习研究之用,不得用于任何商业目的,不得以任何方式修改本作品,基于此产生的法律责任本人不承担任何连带责任。
第1章 重构,第一个案例 1 1.1 起点 1 1.2 重构的第一步 7 1.3 分解并重组statement() 8 1.4 运用多态取代与价格相关的条件逻辑 34 1.5 结语 52 第2章 重构原则 53 2.1 何谓重构 53 2.2 为何重构 55 2.3 何时重构 57 2.4 怎么对经理说 60 2.5 重构的难题 62 2.6 重构设计 66 2.7 重构与性能 69 2.8 重构起源何处 71 第3章 代码的坏味道 75 3.1 Duplicated Code(重复代码) 76 3.2 Long Method(过长函数) 76 3.3 Large Class(过大的类) 78 3.4 Long Parameter List(过长参数列) 78 3.5 Divergent Change(发散式变化) 79 3.6 Shotgun Surgery(霰弹式修改) 80 3.7 Feature Envy(依恋情结) 80 3.8 Data Clumps(数据泥团) 81 3.9 Primitive Obsession(基本类型偏执) 81 3.10 Switch Statements(switch惊悚现身) 82 3.11 Parallel InheritanceHierarchies(平行继承体系) 83 3.12 Lazy Class(冗赘类) 83 3.13 Speculative Generality(夸夸其谈未来性) 83 3.14 Temporary Field(令人迷惑的暂时字段) 84 3.15 Message Chains(过度耦合的消息链) 84 3.16 Middle Man(中间人) 85 3.17 Inappropriate Intimacy(狎昵关系) 85 3.18 Alternative Classes with Different Interfaces(异曲同工的类) 85 3.19 Incomplete Library Class(不完美的库类) 86 3.20 Data Class(纯稚的数据类) 86 3.21 Refused Bequest(被拒绝的遗赠) 87 3.22 Comments(过多的注释) 87 第4章 构筑测试体系 89 4.1 自测试代码的价值 89 4.2 JUnit测试框架 91 4.3 添加更多测试 97 第5章 重构列表 103 5.1 重构的记录格式 103 5.2 寻找引用点 105 5.3 这些重构手法有多成熟 106 第6章 重新组织函数 109 6.1 Extract Method(提炼函数) 110 6.2 Inline Method(内联函数) 117 6.3 Inline Temp(内联临时变量) 119 6.4 Replace Temp with Query(以查询取代临时变量) 120 6.5 Introduce Explaining Variable(引入解释性变量) 124 6.6 Split Temporary Variable(分解临时变量) 128 6.7 Remove Assignments to Parameters(移除对参数的赋值) 131 6.8 Replace Method with Method Object(以函数对象取代函数) 135 6.9 Substitute Algorithm(替换算法) 139 第7章 在对象之间搬移特性 141 7.1 Move Method(搬移函数) 142 7.2 Move Field(搬移字段) 146 7.3 Extract Class(提炼类) 149 7.4 Inline Class(将类内联化) 154 7.5 Hide Delegate(隐藏“委托关系”) 157 7.6 Remove Middle Man(移除中间人) 160 7.7 Introduce Foreign Method(引入外加函数) 162 7.8 Introduce Local Extension(引入本地扩展) 164 第8章 重新组织数据 169 8.1 Self Encapsulate Field(自封装字段) 171 8.2 Replace Data Value with Object(以对象取代数据值) 175 8.3 Change Value to Reference(将值对象改为引用对象) 179 8.4 Change Reference to Value(将引用对象改为值对象) 183 8.5 Replace Array with Object(以对象取代数组) 186 8.6 Duplicate Observed Data(复制“被监视数据”) 189 8.7 Change Unidirectional Association to Bidirectional(将单向关联改为双向关联) 197 8.8 Change Bidirectional Association to Unidirectional(将双向关联改为单向关联) 200 8.9 Replace Magic Number with Symbolic Constant(以字面常量取代魔法数) 204 8.10 Encapsulate Field(封装字段) 206 8.11 Encapsulate Collection(封装集合) 208 8.12 Replace Record with Data Class(以数据类取代记录) 217 8.13 Replace Type Code with Class(以类取代类型码) 218 8.14 Replace Type Code with Subclasses(以子类取代类型码) 223 8.15 Replace Type Code with State/Strategy(以State/Strategy取代类型码) 227 8.16 Replace Subclass with Fields(以字段取代子类) 232 第9章 简化条件表达式 237 9.1 Decompose Conditional(分解条件表达式) 238 9.2 Consolidate Conditional Expression(合并条件表达式) 240 9.3 Consolidate Duplicate Conditional Fragments(合并重复的条件片段) 243 9.4 Remove Control Flag(移除控制标记) 245 9.5 Replace Nested Conditional with Guard Clauses(以卫语句取代嵌套条件表达式) 250 9.6 Replace Conditional with Polymorphism(以多态取代条件表达式) 255 9.7 Introduce Null Object(引入Null对象) 260 9.8 Introduce Assertion(引入断言) 267 第10章 简化函数调用 271 10.1 Rename Method(函数改名) 273 10.2 Add Parameter(添加参数) 275 10.3 Remove Parameter(移除参数) 277 10.4 Separate Query from Modifier(将查询函数和修改函数分离) 279 10.5 Parameterize Method(令函数携带参数) 283 10.6 Replace Parameter with Explicit Methods(以明确函数取代参数) 285 10.7 Preserve Whole Object(保持对象完整) 288 10.8 Replace Parameter with Methods(以函数取代参数) 292 10.9 Introduce Parameter Object(引入参数对象) 295 10.10 Remove Setting Method(移除设值函数) 300 10.11 Hide Method(隐藏函数) 303 10.12 Replace Constructor with Factory Method(以工厂函数取代构造函数) 304 10.13 Encapsulate Downcast(封装向下转型) 308 10.14 Replace Error Code with Exception(以异常取代错误码) 310 10.15 Replace Exception with Test(以测试取代异常) 315 第11章 处理概括关系 319 11.1 Pull Up Field(字段上移) 320 11.2 Pull Up Method(函数上移) 322 11.3 Pull Up Constructor Body(构造函数本体上移) 325 11.4 Push Down Method(函数下移) 328 11.5 Push Down Field(字段下移) 329 11.6 Extract Subclass(提炼子类) 330 11.7 Extract Superclass(提炼超类) 336 11.8 Extract Interface(提炼接口) 341 11.9 Collapse Hierarchy(折叠继承体系) 344 11.10 Form Tem Plate Method(塑造模板函数) 345 11.11 Replace Inheritance with Delegation(以委托取代继承) 352 11.12 Replace Delegation with Inheritance(以继承取代委托) 355 第12章 大型重构 359 12.1 Tease Apart Inheritance(梳理并分解继承体系) 362 12.2 Convert Procedural Design to Objects(将过程化设计转化为对象设计) 368 12.3 Separate Domain from Presentation(将领域和表述/显示分离) 370 12.4 Extract Hierarchy(提炼继承体系) 375 第13章 重构,复用与现实 379 13.1 现实的检验 380 13.2 为什么开发者不愿意重构他们的程序 381 13.3 再论现实的检验 394 13.4 重构的资源和参考资料 394 13.5 从重构联想到软件复用和技术传播 395 13.6 小结 397 13.7 参考文献 397 第14章 重构工具 401 14.1 使用工具进行重构 401 14.2 重构工具的技术标准 403 14.3 重构工具的实用标准 405 14.4 小结 407 第15章 总结 409
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值