LBS位置服务的SQL实现原理以及SQL代码

一、位置服务(搜索周边商家POI)

  1. 自身有海量的周边搜索数据,并且一定要有经纬度坐标;(数据来源可以从网上购买,如果要直接从地图API中获取,那是要付费的,而且费用不菲)
  2. 手机定位后将经纬度发送给服务器,服务器根据经纬度在数据库中匹配数据,并按照距离由近到远排序(此处省略一万字...)

SQL实现经纬度匹配以及距离排序代码

1、SQL计算经纬度距离的自定义函数

CREATE FUNCTION dbo.GetDistance  
(  
    @LatBegin REAL  
    , @LngBegin REAL  
    , @LatEnd REAL  
    , @LngEnd REAL  
)  
RETURNS FLOAT  
AS  
BEGIN  
    DECLARE @Distance REAL  
    DECLARE @EARTH_RADIUS REAL  
    SET @EARTH_RADIUS = 6378.137  
      
    DECLARE @RadLatBegin REAL, @RadLatEnd REAL, @RadLatDiff REAL, @RadLngDiff REAL  
    SET @RadLatBegin = @LatBegin * PI() / 180.0  
    SET @RadLatEnd = @LatEnd * PI() / 180.0  
    SET @RadLatDiff = @RadLatBegin - @RadLatEnd  
    SET @RadLngDiff = @LngBegin * PI() / 180.0 - @LngEnd * PI() / 180.0  
      
    SET @Distance = 2 * ASIN(SQRT(POWER(Sin(@RadLatDiff / 2), 2) + COS(@RadLatBegin) * COS(@RadLatEnd) * POWER(SIN(@RadLngDiff/2),2)))  
    SET @Distance = @Distance * @EARTH_RADIUS  
    --SET @Distance = Round(@Distance * 10000) / 10000  
      
    RETURN @Distance * 1000  
END  

2、获取周边数据的存储过程

CREATE PROCEDURE Up_Data_GetPOI 
	@lngbegin nvarchar(50), --当前经度
	@latbegin nvarchar(50), --当前纬度
	@distance int, --搜索半径,单位为米
	@WhereStr nvarchar(200) --附加搜索条件,如and 1=2 
	AS
	declare @SqlStr nvarchar(2000)
	
	set @SqlStr='select *, dbo.GetDistance('+@latbegin+','+@lngbegin+',纬度字段,经度字段) as distance from [存储商家的数据表名] where (纬度字段<>'' and 经度字段<>'') and dbo.GetDistance('+@latbegin+','+@lngbegin+',纬度字段,经度字段)<'+Convert(nvarchar(10),@distance)+' '+@WhereStr+' order by distance asc'
	
	--print @SqlStr
	Execute Sp_ExecuteSql @SqlStr
GO

二、位置交友

  1. 要有会员数据,同样也得有客户端是否在线以及经纬度的字段(不一定要获取到经纬度,默认为空)
  2. 手机客户端定位后,更新该会员的经纬度坐标以及客户端在线状态
  3. 通过上面的SQL代码即可实现数据列表,只要输出JSON/XML给手机客户端解析即可

店铺装修:淘宝/天猫店铺两侧雪花飘动代码

<div class="skin-box tb-module tshop-pbsm tshop-pbsm-shop-self-defined">
    <div class="skin-box-bd clear-fix">
        <div class="J_TWidget" style="position: relative;" data-widget-type="Tabs" data-widget-config="{'duration':0.1,'activeTriggerCls':'\u2003tm-isv-flower-mask','interval':0.1,'effect':'scrollx','activeIndex':1,'navCls':'lsm-shiyan','contentCls':'lsm-shiyan01','autoplay':true}">
            <div class="lsm-shiyan01" style="left: 0px; width: 999999px; display: none; position: absolute;">
                <span class="-_-switchable-panel-internal1229 ks-switchable-panel-internal526" role="tabpanel" aria-hidden="true" aria-labelledby="ks-switchable529" style="float: left; display: block;">&nbsp;</span>
            </div>
            <div>
                <div class="lsm-shiyan" role="tablist">
                    <div tabindex="-1" class="-_-isv-flower-mask ks-switchable-trigger-internal525 tm-isv-flower-mask" id="ks-switchable529" role="tab" style="left: 50%; top: 0px; width: 0px; height: 0px; right: auto;">
                        <div style="left: -950px; position: relative;">
                            <embed width="475" height="1000" src="http://img1.tbcdn.cn/tfscom/T1lmk2Fk8bXXXtxVjX.swf" type="application/x-shockwave-flash" wmode="transparent" flashvars="scene=taobao_shop" allowscriptaccess="never">
                        </div>
                    </div>
                </div>
            </div>
        </div>
        <div class="J_TWidget" style="position: relative;" data-widget-type="Tabs" data-widget-config="{'duration':0.1,'activeTriggerCls':'\u2003tm-isv-flower-mask','interval':0.1,'effect':'scrollx','activeIndex':1,'navCls':'lsm-shiyan','contentCls':'lsm-shiyan01','autoplay':true}">
            <div class="lsm-shiyan01" style="left: 0px; width: 999999px; display: none; position: absolute;">
                <span class="-_-switchable-panel-internal1229 ks-switchable-panel-internal639" role="tabpanel" aria-hidden="true" aria-labelledby="ks-switchable642" style="float: left; display: block;">&nbsp;</span>
            </div>
            <div>
                <div class="lsm-shiyan" role="tablist">
                    <div tabindex="-1" class="-_-isv-flower-mask ks-switchable-trigger-internal638 tm-isv-flower-mask" id="ks-switchable642" role="tab" style="left: 50%; top: 0px; width: 0px; height: 0px; right: auto;">
                        <div style="left: 475px; position: relative;">
                            <embed width="475" height="1000" src="http://img1.tbcdn.cn/tfscom/T1lmk2Fk8bXXXtxVjX.swf" type="application/x-shockwave-flash" wmode="transparent" flashvars="scene=taobao_shop" allowscriptaccess="never">
                        </div>
                    </div>
                </div>
            </div>
        </div>
    </div>
</div>

店铺装修:淘宝/天猫店通用左侧悬浮代码

最近有不少朋友问我这个代码,我就在这里发出来吧

<div class="J_TWidget" data-widget-type="Carousel" data-widget-config="{'duration':0.1,'activeTriggerCls':'.tshop-psm-shop-ww-hover','effect':'scrollx','activeIndex':1,'navCls':'user-bd11','contentCls':'user-hd11','interval':0.1,'autoplay':true}">
 <ul class="user-hd11" style="display:none;"><li></li></ul>
 <ul class="user-bd11">
  <li style="visibility: visible;left:0px;top:300px;">
   <div style="width:200px;height:30px; background:#000;color:#fff">
   	<!--这里放悬浮内容的代码-->
   </div>
  </li>
 </ul>
</div>

具体效果可以移步到此处查看

http://tjgemstone.tmall.com/

利用旧版Android漏洞的E-Z-2-Use攻击代码发布

  利用 Android 操作系统 WebView 编程接口漏洞的攻击代码已作为一个模块加入到开源漏洞利用框架 Metasploit 中。漏洞影响 Android 4.2 之前的版本,Google 在 Android 4.2 中修正了这个漏洞,但根据官方统计,超过五成用户仍然使用存在漏洞的旧版本。

  WebView 的漏洞允许攻击者在 Android 浏览器和其它应用中注入恶意 JavaScript 代码,获得与目标程序相同的访问权限,攻击者可以开启一个 shell 窗口访问受害者的文件系统、照相机,地理位置数据、SD 卡数据和地址簿。漏洞还可通过不安全网络的中间人攻击触发。漏洞存在于 Android 系统中,而不是私有的 GMS 应用平台,只有更新系统才能修正漏洞。安全研究人员 Tod Beardsley 希望,攻击代码的披露能迫使供应商尽快升级系统。

给C#开发者的代码审查清单

这是为C#开发者准备的通用性代码审查清单,可以当做开发过程中的参考。这是为了确保在编码过程中,大部分通用编码指导原则都能注意到。对于新手和缺乏经验(0到3年工作经验)的开发者,参考这份清单编码会很帮助。

清单

1. 确保没有任何警告(warnings)。

2.如果先执行Code Analysis(启用所有Microsoft Rules)再消除所有警告就更好了。

3. 去掉所有没有用到的usings。编码过程中去掉多余代码是个好习惯。(参考:msdn

4. 在合理的地方检查对象是否为’null’,避免运行的时候出现Null Reference Exception

5. 始终遵循命名规范。一般而言变量参数使用驼峰命名法,方法名和类名使用Pascal命名法。(参考:msdn

6. 请确保你了解SOLID原则。

根据维基百科定义:在程序设计领域,SOLID (单一功能、开闭原则、里氏替换、接口隔离以及依赖反转)是由罗伯特·C·马丁在21世纪早期引入的记忆术首字母缩略字,指代了面向对象编程和面向对象设计的五个基本原则。当这些原则被一起应用时,它们使得一个程序员开发一个容易进行软件维护和扩展的系统变得更加可能。SOLID所包含的原则是通过引发编程者进行软件源代码的代码重构进行软件的代码异味清扫,从而使得软件清晰可读以及可扩展时可以应用的指南。SOLID被典型的应用在测试驱动开发上,并且是敏捷开发以及自适应软件开发的基本原则的重要组成部分。参考:wiki/SOLID_(面向对象设计)

7. 代码可重用性:如果一块代码已经被使用超过一次,或者你希望将来使用它,请提取成一个方法。将重复的工作做成通用的方法放在相关的类中,这样一旦你完成别人就可以使用了。将常用功能开发成用户控件,这样可以跨项目重用它们。(参考:

8. 代码一致性:比方说,Int32写成int,String写成string,应该在代码里保持统一形式。不能一会二写成int一会儿写成Int32。

9. 代码可读性:代码应该是可维护的,便于其他开发者理解。(参考:msdn

10. 释放非托管资源,比如文件I/O,网络资源等。一旦使用结束就应该释放它们。如果你想一旦超出使用范围就自动释放对象,可以使用usings将非托管代码括起来。参考:msdn

11. 合理实现异常处理(try/catch和finally块)和异常记录。参考:msdn

12. 确保代码中方法的行数不要过多,不超过30到40行。

13. 及时用代码管理工具check-in/check-out代码。(比如TFS) 参考:codeproject.com

14. 相互审查代码:和你的同事交换代码,实现内部审查。

15. 单元测试:编写开发测试用例完成单元测试,确保代码被送到QA以前,基本测试完成。参考:msdn

16. 尽量避免for/foreach循环嵌套和if条件嵌套。

17. 如果代码只会使用一次,请使用匿名类型。参考:msdn

18. 尽量使用LINQ查询和Lambda表达式,增加可读性。参考:msdn

19. 合理使用var、object和dynamic关键字。由于很多开发者会感到困惑或者知道的很少,会觉得它们有些相似,故而交换使用,这是要避免的。参考:blogs.msdn

20. 使用访问限定符(private, public, protected, internal, protected internal)限定每个方法、类或变量的需要范围。比方说如果一个类只会在程序集内使用,那么定义成internal就足够了。参考:msdn

21. 在需要保持解耦的地方使用接口,有些设计模式的出现也是由于接口的使用。参考:msdn

22. 按照用法和需要将类定义为sealed、static或abstract。参考:msdn

23. 如果需要多次串联,请使用Stringbuilder代替string,这可以节省堆内存。

24. 检查是否有不可能执行的代码,如果有,请修改。

25. 在每个方法前注释,说明它的用法、输入类型和返回值类型信息。

26. 使用类似Silverlight Spy的工具,检查和操控Silverlight应用在运行时对XMAL的渲染,以此来改善效率。这可以在设计执行XAML时,节省大量退回和来回修改的时间。

27. 使用filddler工具通过检查HTTP/网络流量和带宽,来跟踪web应用和服务的性能。

28. 如果你想确认Visual Studio以外的方法,请使用WCFTestClient.exe工具,或者装载它的进程到Visual Studio来进行调试。

29. 在任何合理的地方使用constants和readonly。参考:/msdnmsdn

30. 尽量避免强制转换和类型转换,因为会造成性能损失。参考:msdn

31. 对于你想提供自定义信息的类,请重载ToString(来自Object类)。参考:msdn

32. 避免直接从其他代码中ctrl+c/ctrl+v。一直建议还是自己用手敲,即使你已经找到相关代码。这样可以锻炼自己写代码能力,还能正确理解那段代码的用法。最终你永远都不会忘记那段代码。

33. 保持阅读书籍和文章的良好习惯,遵循大神们的实践指导。(比如微软专家和一些著名的专家,Martin Fowler, Kent Beck, Jeffrey Ritcher, Ward Cunningham, Scott Hanselman, Scott Guthrie, Donald E Knuth.)

34. 确认代码是否有内存泄漏。如果有,请确保已修正。参考:blogs.msdn.com

35. 尽可能参加专家们组织的技术研讨会,可以接触到最新的软件趋势、技术和最佳实践

36. 要透彻理解OOP概念,并尽可能在代码里实现。

37. 知道项目设计架构,可以从整体上理解程序的执行流程。

38. 采取必要措施阻止避免任何交叉脚本攻击、SQL注入和其他安全漏洞。

39. 永远记得将保密和敏感信息加密(通过使用好的加密算法),比如保存到数据库的密码和保存在web.config文件中的连接字符,要避免被非认证的用户操纵。

40. 避免对已知类型(原始类型)使用默认关键字,比如int, decimal, bool等。多数情况下,如果不确定是值类型还是引用类型,就使用泛型类型(T)。参考:msdn

41. 微软(在代码分析条例和指导中)并不推荐使用’out’和’ref’,这些关键字是通过引用传参,请注意,’ref’参数在传入被调用方法之前,应当在调用方法中先初始化,但’out’参数就不是这样。参考:msdn

代码片段:通过Win32API监听和修改剪切板的内容

Imports System.Runtime.InteropServices

Public Class Form1
  '剪切板监控指针
  Dim ClipboardViewerNext As IntPtr
  '声明相关Win32API
  <DllImport("User32.dll", CharSet:=CharSet.Auto)> _
  Public Shared Function SetClipboardViewer(hWnd As IntPtr) As IntPtr
  End Function

  <DllImport("User32.dll", CharSet:=CharSet.Auto)> _
  Public Shared Function ChangeClipboardChain(hWndRemove As IntPtr, hWndNewNext As IntPtr) As Boolean
  End Function
  '注册事件
  Private Sub RegisterClipboardViewer()
    ClipboardViewerNext = SetClipboardViewer(Me.Handle)
  End Sub
  '注销事件
  Private Sub UnregisterClipboardViewer()
    ChangeClipboardChain(Me.Handle, ClipboardViewerNext)
  End Sub

  Protected Overrides Sub WndProc(ByRef m As Message)
    Select Case m.Msg
      '当剪切板内容发生变化时
      Case &H308
        '将当前焦点所在单元格内容复制到剪切板
        Clipboard.SetText(DataGridView1.CurrentCell.Value)
        Exit Select
      Case Else
        MyBase.WndProc(m)
        Exit Select
    End Select
  End Sub

  Private Sub Form1_Load(sender As System.Object, e As System.EventArgs) Handles MyBase.Load
    '添加测试数据
    DataGridView1.Rows.Add({"aaa", "bbb", "ccc"})
    DataGridView1.Rows.Add({"aaa", "bbb", "ccc"})
  End Sub

  Private Sub DataGridView1_CellClick(sender As System.Object, e As System.Windows.Forms.DataGridViewCellEventArgs) Handles DataGridView1.CellClick
    RegisterClipboardViewer()
  End Sub

  Private Sub DataGridView1_Leave(sender As System.Object, e As System.EventArgs) Handles DataGridView1.Leave
    UnregisterClipboardViewer()
  End Sub
End Class

代码片段:通过编程方式压缩.accdb文件

在VB.Net安装有VSTO的前提下,引用Microsoft.Office.Interop.Access.Dao

实现代码如下:

Imports Microsoft.Office.Interop.Access.Dao

Module Module1

  Sub Main()
    Dim db As New DBEngine
    db.CompactDatabase("H:\My Documents\算神.accdb", "H:\My Documents\算神1.accdb")
    Console.Read()
  End Sub

End Module

在WordPress中为代码自动显示滚动条

在写博客时,有时候添加的代码比较长,超过了一行所能显示的长度时,在WordPress的某些主题中会显示不全,只显示前半截出来,影响阅读。

解决这个问题方法其实很简单,只需要改一下主题的样式表就中可以了。

首先打开主题的样式表style.css,在样式表中搜索一下blockquote和pre。在inove主题里,它们使用相同的样式配置,在它们的样式里加上overflow-x: auto;,完整代码如下:

blockquote, pre {
	background:#F4F5F7 3px 3px no-repeat;
	border:1px dashed #CCC;
	padding:8px 8px 8px 8px;
	margin:5px 0;
	overflow-x:auto
} 

加了这一句话以后,如果在blockquote和pre块的语句超过一行的长度,那么浏览器就会自动显示水平滚动条(scrollbar),让读者滑过去看,很是方便。

如何写出无法维护的代码

这篇文章的原文在这里(http://mindprod.com/jgloss/unmain.html),我看完后我想说——
  1. 什么叫“创造力”,创造力就是——就算是要干一件烂事,都能干得那么漂亮,那么有创意的能力。
  2. 什么叫“抓狂”,抓狂就是——以一种沉着老练的不屈不挠的一本正经的精神,一点一点把你推向崩溃的边缘

  我把文章节选了一些,也并没有完全翻译,简译一下,也加入了一些自己的调侃。对于有下面这些编程习惯的朋友,请大家对号入座。另外,维护程序的朋友们,你们死定了!!

woodpeckerIf builders built buildings the way programmers write programs, then the first woodpecker that came along would destroy civilization. (如果建筑师盖房子就像程序员写程序一样,那么,第一只到来的啄木鸟就能毁掉我们的文明)

  ~ Gerald Weinberg (born: 1933-10-27 age: 77) Weinberg’s Second Law

  程序命名

  • 容易输入的名字。比如:Fred,asdf
  • 单字母的变量名。比如:a,b,c,x,y,z(陈皓注:如果不够用,可以考虑a1,a2,a3,a4,….)
  • 有创意地拼写错误。比如:SetPintleOpening, SetPintalClosing。这样可以让人很难搜索代码。
  • 抽象。比如:ProcessData, DoIt, GetData…抽象到就跟什么都没说一样。
  • 缩写。比如:WTF,RTFSC ……(陈皓注:使用拼音缩写也同样给力,比如: BT,TMD,TJJTDS)
  • 随机大写字母。比如:gEtnuMbER..
  • 重用命名。在内嵌的语句块中使用相同的变量名有奇效。
  • 使用重音字母。比如:int  ínt(注:第二个 ínt不是int)
  • 使用下划线。比如:_, __, ___。
  • 使用不同的语言。比如混用英语,德语,或是中文拼音。
  • 使用字符命名。比如:slash, asterix, comma…
  • 使用无关的单词。比如:god, superman, iloveu….
  • 混淆l和1。字母l和数字1有时候是看不出来的。

  伪装欺诈

  • 把注释和代码交织在一起

  for(j=0; j<array_len; j+ =8)<br />

  {<br />

      total += array[j+0 ];<br />

      total += array[j+1 ];<br />

      total += array[j+2 ]; /* Main body of<br />

      total += array[j+3]; * loop is unrolled<br />

      total += array[j+4]; * for greater speed.<br />

      total += array[j+5]; */<br />

      total += array[j+6 ];<br />

      total += array[j+7 ];<br />

  }

  • 隐藏宏定义。如:#define a=b a=0-b,当人们看到a=b时,谁也想不到那是一个宏。
  • 换行。如下所示,下面的示例使用搜索xy_z变得困难。

  #define local_var xy\<br />

  _z // local_var OK

  • 代码和显示不一致。比如,你的界面显示叫postal code,但是代码里确叫zipcode.
  • 隐藏全局变量。把使用全局变量以函数参数的方式传递给函数,这样可以让人觉得那个变量不是全局变量。
  • 使用同意词。如:

  #define xxx global_var // in file std.h&nbsp;<br />

  #define xy_zxxx// in file ..\other\substd.h&nbsp;<br />

  #define local_var xy_z// in file ..\codestd\inst.h

  • 使用相似的变量名。如:单词相似,swimmer 和 swimner,字母相似:ilI1 或 oO08。parselnt 和 parseInt, D0Calc 和 DOCalc。还有这一组:xy_Z,xy__z, _xy_z, _xyz, XY_Z,xY_z, Xy_z。
  • 重载函数。使用相同的函数名,但是其功能和具体实现完全没有关系。
  • 操作符重载。重载操作符可以让你的代码变得诡异,感谢CCTV,感谢C++。这个东西是可以把混乱代码提高到一种艺术的形式。比如:重载一个类的!操作符,但实际功能并不是取反,让其返回一个整数。于是,如果你使用!!操作符,那么,有意思的事就发生了——先是调用类的重载!操作符,然后把其返回的整数给!成了布尔变量,如果是!!!呢?呵呵。
  • #define。看过本站那些混乱代码的文章,你都会知道宏定义和预编译对于写出不可读的代码的重大意义。不过,一个具有想像力的东西是——在头文件中使用预编译来查看这个头文件被include了几次,而被include不同的次数时,其中的函数定义完全不一

  #ifndef DONE </p>

  <p>#ifdef TWICE </p>

  <p>// put stuff here to declare 3rd time around<br />

  void g(char* str);<br />

  #define DONE </p>

  <p>#else // TWICE<br />

  #ifdef ONCE </p>

  <p>// put stuff here to declare 2nd time around<br />

  void g(void* str);<br />

  #define TWICE </p>

  <p>#else // ONCE </p>

  <p>// put stuff here to declare 1st time around<br />

  void g(std::string str);<br />

  #define ONCE </p>

  <p>#endif // ONCE<br />

  #endif // TWICE<br />

  #endif // DONE

  文档和注释

  • 在注释中撒谎。你不用真的去撒谎,只需在改代码的时候不要更新注释就可以了。
  • 注释明显的东西。比如:/* add 1 to i */。(参看本站的“五种应该避免的注释”)
  • 只注释是什么,而不是为什么
  • 不要注释秘密。如果你开发一个航班系统,请你一定要保证每有一个新的航班被加入,就得要修改25个以上的位置的程序。千万别把这个事写在文档中。
  • 注重细节。当你设计一个很复杂的算法的时候,你一定要把所有的详细细设计都写下来,没有100页不能罢休,段落要有5级以上,段落编号要有500个以上,例如:1.2.4.6.3.13 – Display all impacts for activitywhere selected mitigations can apply(short pseudocode omitted). 这样,当你写代码的时候,你就可以让你的代码和文档一致,如:Act1_2_4_6_3_13()
  • 千万不要注释度衡单位。比如时间用的是秒还是毫秒,尺寸用的是像素还是英寸,大小是MB还是KB。等等。另外,在你的代码里,你可以混用不同的度衡单位,但也不要注释。
  • Gotchas。陷阱,千万不要注释代码中的陷阱。
  • 在注释和文档中发泄不满。(参看本站的“五种应该避免的注释”)

  程序设计

  • Java Casts。Java的类型转型是天赐之物。每一次当你从Collection里取到一个object的时候,你都需要把其转回原来的类型。因些,这些转型操作会出现在N多的地方。如果你改变了类型,那么你不一定能改变所有的地方。而编译器可能能检查到,也可能检查不到。
  • 利用Java的冗余。比如:Bubblegum b = new Bubblegom(); 和 swimmer = swimner + 1; 注意变量间的细微差别。
  • 从不验证。从不验证输入的数据,从不验证函数的返回值。这样做可以向大家展示你是多么的信任公司的设备和其它程序员。
  • 不要封装。调用者需要知道被调用的所有的细节。
  • 克隆和拷贝。为了效率,你要学会使用copy+ paste。你几乎都不用理解别人的代码,你就可以高效地编程了。(陈皓注:Copy+ Paste出来的代码bug多得不能再多)
  • 巨大的listener。写一个listener,然后让你的所有的button类都使用这个listener,这样你可以在这个listener中整出一大堆if…else…语句,相当的刺激。
  • 使用三维数组。如果你觉得三维还不足够,你可以试试四维。
  • 混用。同时使用类的get/set方法和直接访问那个public变量。这样做的好处是可以极大的挫败维护人员。
  • 包装,包装,包装。把你所有的API都包装上6到8遍,包装深度多达4层以上。然后包装出相似的功能。
  • 没有秘密。把所有的成员都声明成public的。这样,你以后就很难限制其被人使用,而且这样可以和别的代码造成更多的耦合度,可以让你的代码存活得更久。
  • 排列和阻碍。把drawRectangle(height,width)改成 drawRectangle(width, height),等release了几个版本后,再把其改回去。这样维护程序的程序员们将不能很快地明白哪一个是对的。
  • 把变量改在名字上。例如,把setAlignment(int alignment)改成,setLeftAlignment, setRightAlignment, setCenterAlignment。
  • Packratting。保留你所有的没有使用的和陈旧的变量,方法和代码。
  • That’s Final。Final你所有的子结点的类,这样,当你做完这个项目后,没有人可以通过继承来扩展你的类。java.lang.String不也是这样吗?
  • 避免使用接口。在java中,BS接口,在C++中BS使用虚函数。
  • 避免使用layout。这样就使得我们只能使用绝对坐标。如果你的老大强制你使用layout,你可以考虑使用GridBagLayout,然后把grid坐标hard code.
  • 环境变量。如果你的代码需要使用环境变量。(getenv()– C++ / System.getProperty()– Java ),那么,你应该把你的类的成员的初始化使用环境变量,而不是构造函数。
  • 使用Magic number。参看《Linux一个插曲》。
  • 使用全局变量。1)把全局变量的初始化放在不同的函数中,就算这个函数和这个变量没有任何关系,这样能够让我们的维护人员就像做侦探工作一样。2)使用全局变量可以让你的函数的参数变得少一些。
  • 配置文件。配置文件主要用于一些参数的初始化。在编程中,我们可以让配置文件中的参数名和实际程序中的名字不一样。
  • 膨胀你的类。让你的类尽可能地拥有各种臃肿和晦涩的方法。比如,你的类只实现一种可能性,但是你要提供所有可能性的方法。不要定义其它的类,把所有的功能都放在一个类中。
  • 使用子类。面向对象是写出无法维护代码的天赐之物。如果你有一个类有十个成为(变量和方法)你可以考虑写10个层次的继承,然后把这十个属性分别放在这十个层次中。如果可能的话,把这十个类分别放在十个不同的文件中。

  混乱你的代码

  • 使用XML。XML的强大是无人能及的。使用XML你可以把本来只要10行的代码变成100行。而且,还要逼着别人也有XML。(参看,信XML得永生信XML得自信
  • 混乱C代码。在《如何加密源代码》中已经说过一些方法了,这里再补充一些。
  • 使用不同的进制。比如:10 和010不是一样的。再比如:array = new int[]{   111,   120,   013,   121,};
  • 尽量使用void*。然后把其转成各种类型
  • 使用隐式的转型。C++的构造函数可以让你神不知鬼不觉得完成转型。
  • 分解条件表达式。如:把 a==100分解成,a>99 && a<101
  • 学会利用分号。如:if ( a );else;{   int d;   d = c;}
  • 间接转型。如:把double转string,写成new Double(d).toString()而不是 Double.toString(d)
  • 大量使用嵌套。一个NB的程序员可以在一行代码上使用超过10层的小括号(),或是在一个函数里使用超过20层的语句嵌套{},把嵌套的if else 转成 [? :] 也是一件很NB的事。
  • 使用C的变种数组。myArray[i] 可以变成*(myArray+ i)也可以变成*(i + myArray)其等价于 i[myArray]。再看一个函数调用的示例,函数声明:int myfunc(int q, int p){ return p%q; }函数调用myfunc(6291, 8)[Array];
  • 长代码行。一行的代码越长越好。这样别人阅读时就需要来来回回的
  • 不要较早的return。不要使用goto,不要使用break,这样,你就需要至少5层以上的if-else来处理错误。
  • 不要使用{}。不要在if else使用{},尤其是在你重量地使用if-else嵌套时,你甚至可以在其中乱缩进代码,这样一来,就算是最有经验的程序员也会踩上陷阱。
  • 使用宏定义。宏定义绝对是混乱C/C++代码的最佳利器。参看 老手是这样教新手编程的
  • 琐碎的封装。比较封装一个bool类,类里面什么都做,就是一个bool.
  • 循环。千万不可用for(int i=0; i<n; i++)使用while代替for,交换n和i,把<改成<=,使用 i–调整步伐。

  测试

  • 从不测试。千万不要测试任何的出错处理,从来也不检测系统调用的返回值。
  • 永远不做性能测试。如果不够快就告诉用户换一个更快的机器。如果你一做测试,那么就可能会要改你的算法,甚至重设计,重新架构。
  • 不要写测试案例。不要做什么代码覆盖率测试,自动化测试。
  • 测试是懦夫行为。一个勇敢的程序员是根本不需要这一步的。太多的程序太害怕他们的老板,害怕失去工作,害怕用户抱怨,甚至被起诉。这种担心害怕直接影响了生产力。如果你对你的代码有强大的信心,那还要什么测试呢?真正的程序员是不需要测试自己的代码的。

  其它

  • 你的老板什么都知道。无论你的老板有多SB,你都要严格地遵照他的旨意办事,这样一样,你会学到更多的知识如何写出无法维护的代码来的。
  • 颠覆Help Desk。你要确保你那满是bug的程序永远不要被维护团队知道。当用户打电话和写邮件给你的时候,你就不要理会,就算要理会,让用户重做系统或是告诉用户其帐号有问题,是标准的回答。
  • 闭嘴。对于一些像y2k这样的大bug,你要学会守口如瓶,不要告诉任何人,包括你的亲人好友以及公司的同事和管理层,
  • 忽悠。你会学会忽悠,就算你的代码写得很烂,你也要为其挂上GoF设计模式的标签,就算你的项目做得再烂,你也要为其挂上敏捷的标签,只有学会像中国Thoughtworks的咨询师那样去忽悠,你才能学会更炫更酷的方法,让整个团队和公司,甚至整个业界都开始躁动,这样才能真正为难维护的代码铺平道路。

  这个文档中还有很多很多,实在是太TMD强大了,大家自己去看看吧。有精力有能力的朋友不妨把其翻译成中文。

  总之,我们的口号是——

  Write Everywhere, Read Nowhere