思维导图
介绍
前几篇系列文章,我比较关注的是<PHP 杂谈《重构-改善既有代码的设计》之一 重新组织你的函数>,但是我觉得我还是没有说清楚,我自己也有很多不理解的地方,而且这篇是我的第一篇这方面的文章,有很多的纰漏,所以我会经常性的去做修改,如果大家有好的意见不妨告知一、二。
今天谈得是“接口”,此接口非“Interface”,而是一个统称。我们一般可以把供别人使用的函数或者url(一般是用于提供数据)叫接口。——可能还有别的意思,毕竟我现在还属于“菜鸟”,如果有理解上的错误,请指正。
我们知道“容易被理解和被使用的接口”,是开发良好面向对象软件的关键。——本文将介绍“使接口变得更简洁易用”的重构手法。
题外话:
如果大家觉得我这篇文章太长,看起来麻烦的话,建议大家”就看图片和粗体的文字“。
昨天,“old“博友给我留言,我以前也没仔细考虑过,这次我也想了想。留言内容是:
data:image/s3,"s3://crabby-images/24d1c/24d1cda406268369c10adc31ee6238c5bf9e72ab" alt=""
我个人觉得,很多事情只有我们去关注过,才能知道它的价值。
至于简单,重构的目地也是为了简单和易理解性。
至于执着,我觉得在技术上,我们很多时候需要这种执着,即使你过后觉得你错了,但是我们在这之间还是会有所收获。我们只有经历过很多次的磨合(这种磨合有正确的也有错误的),我们才能知道它的价值,我们才能收获到我们需要的东西。
至于利益,”Old“是不是指公司利益,恩,确实是,很多时候我们在编码的过程中,需要赶进度,还有我们在重构中也会有一些错误出来,所以我的建议是,在开发之初,你就要在设计和重构中,不断进行磨合,不要觉得浪费时间,很多时候,好的结构能加速你的开发。
专业术语
data:image/s3,"s3://crabby-images/b61aa/b61aa9a7b7fe007d571d5bbbc4669d2505dca500" alt=""
data:image/s3,"s3://crabby-images/b1ab0/b1ab054764c5deabf12725883c339f8ecbabd646" alt=""
data:image/s3,"s3://crabby-images/64935/6493575e95821f4c33dd0f0ed54b82828bd7eebb" alt=""
data:image/s3,"s3://crabby-images/69b86/69b86537b7d9474ba188366c1ac3338b5557e1d4" alt=""
data:image/s3,"s3://crabby-images/9ae2b/9ae2b5135cd0f1859c87d6a8bbf1ca0fb76bdfff" alt=""
data:image/s3,"s3://crabby-images/86391/86391ecc8caac867ce28591485be848062791142" alt=""
data:image/s3,"s3://crabby-images/21306/2130691835b48b2ac7e322516512edaf5df50fdf" alt=""
Rename Method 状况:如果函数的名称未能揭示函数的用途,那么修改函数名称。
data:image/s3,"s3://crabby-images/6b6e2/6b6e23beb2e14be6eb68437a68281c9b1671886a" alt=""
动机:
我极力提倡的一种编程风格就是将复杂的处理过程分解成小函数。但是如果小函数的命名不好,这会使你费劲周折却弄不清楚这些小函数各自的用途。
给函数命名的一个好办法:考虑应该给这个函数写上一句怎样的注释 -——> 想办法将注释变成函数的名称。
起一个好名称并不容易,需要经验。——要想成为一个真正的编程高手,“起名称”的水平至关重要。
如果你看到一个函数名称不能很好的表达它的用途,应该马上加以修改。
Example:
data:image/s3,"s3://crabby-images/7e8f0/7e8f0d22e45f4a600abafd4a2f712ab0300bbd83" alt=""
data:image/s3,"s3://crabby-images/78449/78449c43c60692cb2a3f4537a3e91e315fe3e33f" alt=""
data:image/s3,"s3://crabby-images/36faf/36fafcda38781e09a8d9f7897eb9ccb966c7d673" alt=""
data:image/s3,"s3://crabby-images/22b07/22b07d22d23f7635803b7908788197e67c8a48ab" alt=""
data:image/s3,"s3://crabby-images/31259/312598b28f3b3f41d15e16fd34800bf0eb06b8b2" alt=""
data:image/s3,"s3://crabby-images/5c6e5/5c6e5c8e942922d6021c0ac6e002bb86113a443c" alt=""
Add Parameter
状况:某个函数需要从调用端得到更多的信息,那么为此函数添加一个参数,让该参数带进函数所需信息。
动机:
1、Add Parameter 是一个很常用的重构手法。
2、修改过的函数需要一些过去没有的信息,因此你需要给函数添加一个参数。
3、除了Add Parameter外,只要有可能,其他选择都比“Add Parameter”要好,因为有可能其他选择不会增加参数列的长度。——过长的参数列会使程序员记不住那么多参数。
data:image/s3,"s3://crabby-images/a749f/a749fad90bf22eca10a9b6aa12806b8b949e6668" alt=""
Remove Parameter
状况:函数本体不再需要某个参数,那么将该参数去除。
动机:
1、参数指出函数信息,不同参数代表不同意义。函数调用这必须为每一个参数操心该传什么东西进去。——如果不去掉参数,那就为每一次调用多费一份心。
2、如果你发现有很多调用者,那么为了不让调用者操心,你可以这样做,把要移除的参数设置为某个默认值(如null),这样调用者只传那些没有默认值的参数。
data:image/s3,"s3://crabby-images/5229d/5229dc7d05b3eddbb0be47c4159dfa099a4b9f65" alt=""
Separate Query from Modifier
状况:如果某个函数既返回对象的状态值,又修改(副作用)对象状态(state),那么
建立两个不同的函数,其中一个负责查询,另一个负责修改。
data:image/s3,"s3://crabby-images/2df12/2df12d0dabd8b0bb3369ca1e6d52246b53259146" alt=""
Example:
data:image/s3,"s3://crabby-images/80b30/80b300b3bb326b662ded0521e139a6842f0f5abe" alt=""
data:image/s3,"s3://crabby-images/9e182/9e18237dbaf69a676dc2a01f4dda757cce145687" alt=""
data:image/s3,"s3://crabby-images/1c0ef/1c0ef93b6f9937bf3888426da5db3e6603772285" alt=""
Parameterize Method
状况:如果若干函数做了类似的工作,但在函数本体中包含了不同的值,那么
建立单一函数,以参数表达那些不同的值。
动机:
1、一般是因为有少数几个值不同,所以建立了几个相似的函数。
2、分离的函数替换为一个统一的函数,通过参数来处理那些变化情况,以简化问题。
3、去除重复的代码,提高灵活性。——可以使用这个参数处理其他变化情况。
data:image/s3,"s3://crabby-images/d7774/d7774a2d3d49c087632482ece572a5a16ee4b62b" alt=""
Example:
data:image/s3,"s3://crabby-images/c30c6/c30c6cb2300f4986819259432c541b573d04e0a4" alt=""
data:image/s3,"s3://crabby-images/94084/94084745868fb15cacecc2ccebecbc5afb6fe113" alt=""
data:image/s3,"s3://crabby-images/8c996/8c9960b9fce680f79f796572757cf8e0eea73a97" alt=""
Replace Parameter with Explicit Methods
状况:你有一个函数,其内完全取决于参数值而采取不同的反应,那么
针对该参数的每个值,建立一个独立的函数。
动机:
1、如果某个参数有离散值,而函数内又以条件式检查这些参数值,并根据不同的参数值做出不同的反应,那么就应该使用本次重构。
2、可以获得好处:“编译期代码检查”,“接口更清楚”(如果用参数值决定函数行为,那么函数用户不但需要观察该函数,而且还要判断参数是否“合法化”。——而合法的参数,很少在文档中提到,必须通过上下文,才能判断)
3、不考虑“编译期检验”的好处,为了获取一个清晰的接口,我们也值得这么做。
data:image/s3,"s3://crabby-images/88ec2/88ec2afdd4beed685c0a692132eeb16b6bc5ced3" alt=""
Example:
data:image/s3,"s3://crabby-images/3d344/3d34464b2e47d757697bcfa8e608bab1f76b56f3" alt=""
data:image/s3,"s3://crabby-images/7573b/7573b1f5ed1d2f24d50612c1c13640028e158689" alt=""
data:image/s3,"s3://crabby-images/326a2/326a230d9b53e01d0cf344f77845b0b48e392e91" alt=""
data:image/s3,"s3://crabby-images/7573b/7573b1f5ed1d2f24d50612c1c13640028e158689" alt=""
data:image/s3,"s3://crabby-images/c09f0/c09f05ba84994d5eada9627a535b77eca32c8dec" alt=""
Preserve Whole Object
状况:如果你从某个对象中取出若干值,将它们作为某一次函数调用中的参数,那么
改使用(传递)整个对象。
动机:
1、参数列更稳固;
2、提高代码的可读性;——过长的参数列很难使用,因为调用者和被调用者都必须记住这些参数的用途。
data:image/s3,"s3://crabby-images/92769/9276940a338e3b21e5d83069a441eabe1801ca69" alt=""
Example:
data:image/s3,"s3://crabby-images/b578f/b578f8e39518bb46332e659b26073b28d73200aa" alt=""
data:image/s3,"s3://crabby-images/7573b/7573b1f5ed1d2f24d50612c1c13640028e158689" alt=""
data:image/s3,"s3://crabby-images/0a5df/0a5dffda66d51080df21ae3335e87e47cff24096" alt=""
Replace Parameter with Methods
状况:如果对象调用某个函数,并将所得结果做为参数,传递给另一个函数(接受参数的函数也有调用前一个函数的能力),那么
让参数接受者去除该项参数,并直接调用前一个函数。
动机:
1、如果函数通过其他途径获得参数值,那么它就不应该通过参数取得该值。
2、过长的参数列会增加程序阅读者的理解难度,因此我们应该尽可能的缩短参数列的长度。
3、方法:看看“参数接受端”是否可以通过“与调用端相同的计算”来取得参数携带值。
4、如果函数调用端通过对象内部的另一个函数来计算参数,并在计算过程中“未曾引用调用端的其他参数”,那么就可以将这个计算过程转移到被调用端内,从而去除该项参数。
Example:
data:image/s3,"s3://crabby-images/2e4d9/2e4d9c4761dd4e56a2a5814638db31305424ab46" alt=""
data:image/s3,"s3://crabby-images/7573b/7573b1f5ed1d2f24d50612c1c13640028e158689" alt=""
data:image/s3,"s3://crabby-images/0f582/0f582c591cab1f966fae44b18df9d4c4255df5e9" alt=""
Introduce Parameter Object
状况:某些参数总是很自然地同时出现,那么
以一个对象取代这些参数。
动机:
data:image/s3,"s3://crabby-images/46fec/46fecfcab7635976aa093a05bdc295f588c5efcc" alt=""
1、一组参数可能有几个函数同时使用,这些函数可能隶属于同一个class,也可能隶属于不同的classes。——这样的一组参数就是所谓的Data Clump(数据泥团)。
2、我们可以运用一个对象包装所有这些数据,再以对象取代Data Clump。——目地:哪怕只是为了把这些数据组织在一起,这样做也是值得的。
3、本项重构的价值在于“缩短了参数列的长度”。此外,新对象所定义的访问函数(accessors)还可以使代码更具一致性。——这又进一步降低了代码的理解难度和修改难度。
4、本项重构还可以带给你更多好处。——当你把这些参数组织到一起之后,往往很快可以发现“可被移植新建class“的行为。——减少重复代码。
Example:
data:image/s3,"s3://crabby-images/97f1d/97f1dde1ef786a78806dad7c7caa83fdccd2ad8e" alt=""
data:image/s3,"s3://crabby-images/32dc1/32dc1768d28c5a7be2b3f1e6f163eb9bfcd57dde" alt=""
data:image/s3,"s3://crabby-images/9660c/9660cf21cf46e5c9a33a2ab6d13948384ca6c604" alt=""
Remove Setting Method
状况:你的class中的某个值域,应该在对象初创时被设置,然后就不再改变,那么
去掉该值域的所有设置函数(setter)。
data:image/s3,"s3://crabby-images/5f0a3/5f0a32234dd38c71054952700e54fb43a9d01995" alt=""
动机:
1、如果你为某个值域提供了设置函数(setter),这就暗示了这个值域可以被改变。
2、如果你不希望在对象初创之后,此值域还有机会改变,那就不要为它提供设置函数。——这样你的意图会更加清晰,并且可以排除其值被修改的可能性。
Example:
data:image/s3,"s3://crabby-images/ce4f6/ce4f6ca0d3de4c93c70e341e1fc6661be1b634c2" alt=""
data:image/s3,"s3://crabby-images/e450e/e450e3cd58e15f66825081297f729f37381f84fd" alt=""
data:image/s3,"s3://crabby-images/b867b/b867bb65cf9d19df778d47029a6f08e099ecd335" alt=""
data:image/s3,"s3://crabby-images/5ccfb/5ccfbdc7267869d1b793732935bcfced3393abe6" alt=""
data:image/s3,"s3://crabby-images/5b4c1/5b4c16bb9d09d553b31edbd05865210c2fe5eb09" alt=""
data:image/s3,"s3://crabby-images/69c77/69c777df09e414a4e124d56d486a489bdbecdc34" alt=""
data:image/s3,"s3://crabby-images/2e4f3/2e4f34e097ea77315d833c6dcb99217476804ed7" alt=""
Hide Method
状况:如有有一个函数,从来没有被其他class用到,那么
将这个函数设置为private。
动机:
data:image/s3,"s3://crabby-images/f3def/f3defbc1dd42459c68d387f9ab91c669648eebb7" alt=""
1、重构往往促使你修改“函数的可见度“。——时刻检查可被隐藏的函数。
2、经常检查有没有可能降低某个函数的可见度(使它私有化)。
——>当你在另一个class中移除对某个函数的调用时,就应该检查。
——>特别对setter函数进行上述的检查。
Replace Constructor with Factory Method
状况:如果你希望在创建对象时不仅仅是对它做简单的构件动作,那么将__construct(构造函数)替换为factory method。
动机:
在subclass过程中以factory method取代type code。——你可能常常需要type code创建相应的对象。
Example:
data:image/s3,"s3://crabby-images/a8ae0/a8ae01376896a2479bc4019ce885b3752ac92cd1" alt=""
data:image/s3,"s3://crabby-images/5ba18/5ba18246984baa35b7d144311d758b1d77c96c93" alt=""
data:image/s3,"s3://crabby-images/f3de4/f3de466371086722d197ccb8d9bf7bf7bb6736f2" alt=""
data:image/s3,"s3://crabby-images/d7f8e/d7f8e34b8adcaf508eedbed9273d4dbaa1a249fe" alt=""
data:image/s3,"s3://crabby-images/361e8/361e8fd4cfc4a8fe1cfc1a9869f7331149b41cab" alt=""
data:image/s3,"s3://crabby-images/563ae/563ae3565469dd75106761e7d69f4bc593379f74" alt=""
data:image/s3,"s3://crabby-images/b0cdd/b0cddaf7aba5de37b6e4fdab513ec4b2d451e701" alt=""
接着来:
data:image/s3,"s3://crabby-images/d6f64/d6f64b7836ce3e77bed919b035f99fe905838e62" alt=""
data:image/s3,"s3://crabby-images/db54c/db54c202e338d900497f2e6d85350b4cbc67b123" alt=""
Replace Error Code with Exception
状况:如果某个函数返回一个特定的代码(special code),用以表示某种错误情况,那么
改用异常(Exception)。
动机:
清楚的将”普通程序“和”错误处理“分开,这使的程序更容易”理解“。
data:image/s3,"s3://crabby-images/b0d8c/b0d8cd89054645aad9dc40de32bb35be60918a27" alt=""
Example:
data:image/s3,"s3://crabby-images/bea21/bea21895b5e3333b931d24b6643a3ea1a82db49e" alt=""
data:image/s3,"s3://crabby-images/af02d/af02d2703e6faeffaea46dacb822df84c2c8d087" alt=""
data:image/s3,"s3://crabby-images/2e415/2e415f83aaa72f167da71afaf73c48669969eca5" alt=""
data:image/s3,"s3://crabby-images/84ae3/84ae3ff0ba97fb06cc8605a4d936ec685b1bd2ae" alt=""
conclusion
把我每一次的收获与大家分享,如果大家有那么一丁点的收获,也让我高兴不已。还有如在文章中有错误,望请指点一、二。
我不知道是不是找错地方了,有博友留言说“博客园里主要盛行C#”,看得人是不是主要以PHP程序员为主?还有很少有人给我留言,也很少有人指出我文章中的错误(难道我的文章中真的没有错误吗?),昨天”@四眼蒙面侠“给我留了言,我在与他的交谈中收获甚多,也感谢的他的批评指正,也希望能跟大家多交流。