iOS学习iOSIOS 超级无敌

iOS团队编程规范

2017-07-25  本文已影响903人  檀羽冲

前 言

一、命名规范

1、统一要求

含义清楚,尽量做到不需要注释也能了解其作用,若做不到,就加注释,使用全称,不使用缩写。

2、类名

大驼峰式命名:每个单词的首字母都采用大写字母

==例:== MFHomePageViewController

3、私有变量

==例:== NSString *_somePrivateVariable

4、property变量

==例:==

typedef void (^ErrorCodeBlock) (id errorCode,NSString *message);

@property (nonatomic, readwrite, strong) UIView *headerView;    //注释

@property (nonatomic, readwrite, copy) ErrorCodeBlock errorBlock;   //将block拷贝到堆中
@property (nonatomic, readwrite, copy) NSString *userName;    

5、宏和常量命名

==例:==

//宏定义的常量
#define ANIMATION_DURATION    0.3
#define MY_MIN(A, B)  ((A)>(B)?(B):(A))

//局部类型常量
static const NSTimeInterval kAnimationDuration = 0.3;

//外部可见类型常量
//EOCViewClass.h
extern const NSTimeInterval EOCViewClassAnimationDuration;
extern NSString *const EOCViewClassStringConstant;  //字符串类型

//EOCViewClass.m
const NSTimeInterval EOCViewClassAnimationDuration = 0.3;
NSString *const EOCViewClassStringConstant = @"EOCStringConstant";

6、Enum

==例:==

typedef NS_ENUM(NSInteger, UIViewAnimationTransition) {
    UIViewAnimationTransitionNone,
    UIViewAnimationTransitionFlipFromLeft,
    UIViewAnimationTransitionFlipFromRight,
    UIViewAnimationTransitionCurlUp,
    UIViewAnimationTransitionCurlDown,
};

typedef NS_OPTIONS(NSUInteger, UIControlState) {
    UIControlStateNormal       = 0,
    UIControlStateHighlighted  = 1 << 0,
    UIControlStateDisabled     = 1 << 1,
};

7、Delegate

==例:==

@protocol UITableViewDataSource<NSObject>

@required

//回调方法存在两个以上参数
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section;

@optional

//回调方法的参数只有类自己
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView;              // Default is 1 if not implemented
@protocol UITableViewDelegate<NSObject, UIScrollViewDelegate>

@optional

//使用`did`和`will`通知`Delegate`
- (nullable NSIndexPath *)tableView:(UITableView *)tableView willSelectRowAtIndexPath:(NSIndexPath *)indexPath;
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath;

8、方法

==例:==

//不要使用 and 来连接属性参数
- (int)runModalForDirectory:(NSString *)path file:(NSString *)name types:(NSArray *)fileTypes;    //推荐
- (int)runModalForDirectory:(NSString *)path andFile:(NSString *)name andTypes:(NSArray *)fileTypes;    //反对

//表示对象行为的方法、执行性的方法
- (void)insertModel:(id)model atIndex:(NSUInteger)atIndex;
- (void)selectTabViewItem:(NSTableViewItem *)tableViewItem

//返回性的方法
- (instancetype)arrayWithArray:(NSArray *)array;

//参数过长的情况
- (void)longMethodWith:(NSString *)theFoo
                  rect:(CGRect)theRect
              interval:(CGFloat)theInterval
{
   //Implementation
}

//不要加get
- (NSSize) cellSize;  //推荐
- (NSSize) getCellSize;  //反对

//使用情态动词,不要使用do或does
- (BOOL)canHide;  //推荐
- (BOOL)shouldCloseDocument;  //推荐
- (BOOL)doesAcceptGlyphInfo;  //反对

二、代码注释规范

优秀的代码大部分是可以自描述的,我们完全可以用代码本身来表达它到底在干什么,而不需要注释的辅助。

但并不是说一定不能写注释,有以下三种情况比较适合写注释:

除了上述这三种情况,如果别人只能依靠注释才能读懂你的代码的时候,就要反思代码出现了什么问题。

最后,对于注释的内容,相对于“做了什么”,更应该说明“为什么这么做”

1、import注释

如果有一个以上的import语句,就对这些语句进行分组,每个分组的注释是可选的。

// Frameworks
#import <QuartzCore>;

// Models
#import "NYTUser.h"

// Views
#import "NYTButton.h"
#import "NYTUserView.h"

2、属性注释

写在属性之后,用两个空格隔开
==例:==

@property (nonatomic, readwrite, strong) UIView *headerView;  //注释    

3、方法声明注释:

一个函数(方法)必须有一个字符串文档来解释,除非它:

而其余的,包括公开接口,重要的方法,分类,以及协议,都应该伴随文档(注释):

建议这样写:

/This comment serves to demonstrate the format of a doc string.

Note that the summary line is always at most one line long, and after the opening block comment,
and each line of text is preceded by a single space.
*/

方法的注释使用Xcode自带注释快捷键:Commond+option+/
==例:==

/**
 <#Description#>

 @param tableView <#tableView description#>
 @param section <#section description#>
 @return <#return value description#>
 */
- (CGFloat)tableView:(UITableView *)tableView heightForHeaderInSection:(NSInteger)section
{
    //...
}

4、代码块注释

单行的用//+空格开头,多行的采用/* */注释

5、TODO

使用//TODO:说明 标记一些未完成的或完成的不尽如人意的地方

==例:==

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
    //TODO:增加初始化
    return YES;
}

三、代码格式化规范

1、指针*位置

定义一个对象时,指针*靠近变量

==例:== NSString *userName;

2、方法的声明和定义

- 、+和 返回值之间留一个空格,方法名和第一个参数之间不留空格

==例:==

- (void)insertSubview:(UIView *)view atIndex:(NSInteger)index;

3、代码缩进

==例:==

!bValue
fLength = fWidth * 2;

- (void)sampleMethod1;

- (void)sampleMethod2;

4、对method进行分组

使用#pragma mark -method进行分组

#pragma mark - Life Cycle Methods
- (instancetype)init
- (void)dealloc

- (void)viewWillAppear:(BOOL)animated
- (void)viewDidAppear:(BOOL)animated
- (void)viewWillDisappear:(BOOL)animated
- (void)viewDidDisappear:(BOOL)animated

#pragma mark - Override Methods

#pragma mark - Intial Methods

#pragma mark - Network Methods

#pragma mark - Target Methods

#pragma mark - Public Methods

#pragma mark - Private Methods

#pragma mark - UITableViewDataSource  
#pragma mark - UITableViewDelegate  

#pragma mark - Lazy Loads

#pragma mark - NSCopying  

#pragma mark - NSObject  Methods

5、大括号写法

==例:==

- (void)sampleMethod
{
    BOOL someCondition = YES;
    if(someCondition) {
        // do something here
    }
}

6、property变量

==例:==

@property (nonatomic, readwrite, strong) UIView *headerView;    //注释    

四、编码规范

1、if语句

①、须列出所有分支(穷举所有的情况),而且每个分支都须给出明确的结果。

==推荐这样写:==

var hintStr;
if (count < 3) {
  hintStr = "Good";
} else {
  hintStr = "";
}

==不推荐这样写:==

var hintStr;
if (count < 3) {
 hintStr = "Good";
}

②、不要使用过多的分支,要善于使用return来提前返回错误的情况,把最正确的情况放到最后返回。

==推荐这样写:==

if (!user.UserName) return NO;
if (!user.Password) return NO;
if (!user.Email) return NO;

return YES;

==不推荐这样写:==

BOOL isValid = NO;
if (user.UserName)
{
    if (user.Password)
    {
        if (user.Email) isValid = YES;
    }
}
return isValid;

③、条件过多,过长的时候应该换行。条件表达式如果很长,则需要将他们提取出来赋给一个BOOL值,或者抽取出一个方法

==推荐这样写:==

if (condition1 && 
    condition2 && 
    condition3 && 
    condition4) {
  // Do something
}
BOOL finalCondition = condition1 && condition2 && condition3 && condition4
if (finalCondition) {
  // Do something
}
if ([self canDelete]){
  // Do something
}

- (BOOL)canDelete
{
    BOOL finalCondition1 = condition1 && condition2
    BOOL finalCondition2 =  condition3 && condition4

    return condition1 && condition2;
}

==不推荐这样写:==

if (condition1 && condition2 && condition3 && condition4) {
  // Do something
}

④、条件语句的判断应该是变量在右,常量在左。

==推荐:==

if (6 == count) {
}

if (nil == object) {
}

if (!object) {
}

==不推荐:==

if (count == 6) {
}

if (object == nil) {
}

if (object == nil)容易误写成赋值语句,if (!object)写法很简洁

⑤、每个分支的实现代码都须被大括号包围

==推荐:==

if (!error) {
  return success;
}

==不推荐:==

if (!error)
    return success;

可以如下这样写:

if (!error) return success;

2、for语句

①、不可在for循环内修改循环变量,防止for循环失去控制。

for (int index = 0; index < 10; index++){
   ...
   logicToChange(index)
}

②、避免使用continue和break。

continuebreak所描述的是“什么时候不做什么”,所以为了读懂二者所在的代码,我们需要在头脑里将他们取反。

其实最好不要让这两个东西出现,因为我们的代码只要体现出“什么时候做什么”就好了,而且通过适当的方法,是可以将这两个东西消灭掉的:

var filteredProducts = Array<String>()
for level in products {
    if level.hasPrefix("bad") {
        continue
    }
    filteredProducts.append(level)
}

我们可以看到,通过判断字符串里是否含有“bad”这个prefix来过滤掉一些值。其实我们是可以通过取反,来避免使用continue的:

for level in products {
    if !level.hasPrefix("bad") {
      filteredProducts.append(level)
    }
}

while里的break其实就相当于“不存在”,既然是不存在的东西就完全可以在最开始的条件语句中将其排除。

while里的break:

while (condition1) {
  ...
  if (condition2) {
    break;
  }
}

取反并合并到主条件:

while (condition1 && !condition2) {
  ...
}

有人喜欢这样做:在有返回值的方法里break之后,再返回某个值。其实完全可以在break的那一行直接返回。

func hasBadProductIn(products: Array<String>) -> Bool {

    var result = false    
    for level in products {
        if level.hasPrefix("bad") {
            result = true
            break
        }
    }
   return result
}

遇到错误条件直接返回:

func hasBadProductIn(products: Array<String>) -> Bool {
    for level in products {
        if level.hasPrefix("bad") {
            return true
        }
    }
   return false
}

这样写的话不用特意声明一个变量来特意保存需要返回的值,看起来非常简洁,可读性高。

3、Switch语句

①、每个分支都必须用大括号括起来

推荐这样写:

switch (integer) {  
  case 1:  {
    // ...  
   }
    break;  
  case 2: {  
    // ...  
    break;  
  }  
  default:{
    // ...  
    break; 
  }
}

②、使用枚举类型时,不能有default分支, 除了使用枚举类型以外,都必须有default分支

RWTLeftMenuTopItemType menuType = RWTLeftMenuTopItemMain;  
switch (menuType) {  
  case RWTLeftMenuTopItemMain: {
    // ...  
    break; 
   }
  case RWTLeftMenuTopItemShows: {
    // ...  
    break; 
  }
  case RWTLeftMenuTopItemSchedule: {
    // ...  
    break; 
  }
}

Switch语句使用枚举类型的时候,如果使用了default分支,在将来就无法通过编译器来检查新增的枚举类型了。

4、函数

①、一个函数只做一件事(单一原则)

每个函数的职责都应该划分的很明确(就像类一样)。

==推荐:==

dataConfiguration()
viewConfiguration()

==不推荐:==

void dataConfiguration()
{   
   ...
   viewConfiguration()
}

②、对于有返回值的函数(方法),每一个分支都必须有返回值

==推荐:==

int function()
{
    if(condition1){
        return count1
    }else if(condition2){
        return count2
    }else{
       return defaultCount
    } 
}

==不推荐:==

int function()
{
    if(condition1){
        return count1
    }else if(condition2){
        return count2
    }
}

③、对输入参数的正确性和有效性进行检查,参数错误立即返回

==推荐:==

void function(param1,param2)
{
      if(param1 is unavailable){
           return;
      }

      if(param2 is unavailable){
           return;
      }

     //Do some right thing
}

④、如果在不同的函数内部有相同的功能,应该把相同的功能抽取出来单独作为另一个函数

原来的调用:

void logic() {
  a();
  b();
  if (logic1 condition) {
    c();
  } else {
    d();
  }
}

将a,b函数抽取出来作为单独的函数

void basicConfig() {
  a();
  b();
}

void logic1() {
  basicConfig();
  c();
}

void logic2() {
  basicConfig();
  d();
}

⑤、将函数内部比较复杂的逻辑提取出来作为单独的函数

一个函数内的不清晰(逻辑判断比较多,行数较多)的那片代码,往往可以被提取出去,构成一个新的函数,然后在原来的地方调用它这样你就可以使用有意义的函数名来代替注释,增加程序的可读性。

举一个发送邮件的例子:

openEmailSite();
login();

writeTitle(title);
writeContent(content);
writeReceiver(receiver);
addAttachment(attachment);

send();

中间的部分稍微长一些,我们可以将它们提取出来:

void writeEmail(title, content,receiver,attachment)
{
  writeTitle(title);
  writeContent(content);
  writeReceiver(receiver);
  addAttachment(attachment); 
}

然后再看一下原来的代码:

openEmailSite();
login();
writeEmail(title, content,receiver,attachment)
send();

参考资料:
iOS 代码规范
iOS开发总结之代码规范
iOS开发代码规范(通用)
Objective-C开发编码规范
【iOS】命名规范
Ios Code Specification
Apple Coding Guidelines for Cocoa
Google Objective-C Style Guide

上一篇 下一篇

猜你喜欢

热点阅读