Git

基本命令

# 安装Git
$ sudo apt install git

# 配置个人信息
$ git config --global user.name "Your Name"
$ git config --global user.email "email@example.com"

# 切换目录初始化
$ git init

# 文件添加到仓库
$ git add -p <file>

# 把文件提交到仓库
$ git commit -m "add LICENSE"

# 查看仓库当前状态
$ git status

# 查看difference
$ git diff

# 显示从最近到最远的提交日志
$ git log --pretty=oneline # 格式化输出信息

# 版本退回
$ git reset --hard HEAD^ # 当前版本HEAD,上一个版本HEAD^,上上个版本HEAD^^
$ git reset --hard 130f10a # 或HEAD~100

# 查看命令记录
$ git reflog

# 丢弃工作区的修改,回到最近一次git commit或git add时的状态:
$ git checkout -- README.md

# 把暂存区的修改撤销掉(unstage)
$ git reset HEAD READER.md

# 从版本库中删除该文件
$ git rm README.md
$ git commit -m "remove READER.md"

# 把误删的文件恢复到最新版本,checkout其实用版本库里的版本替换工作区的版本
$ git checkout -- README.md

远程仓库

$ ssh-keygen -t rsa -C "youremail@example.com"
# 测试是否成功
$ ssh -T git@github.com

# 把一个已有的本地仓库与之关联
$ git remote add origin git@github.com:Windrivder/Windrivder.git

# 把本地库的所有内容推送到远程库上(推送master分支的内容)
$ git push -u origin master

# 向远程库推送更新
$ git push origin master

# 从远程库克隆
$ git clone git@github.com:michaelliao/gitskills.git

分支管理

# 创建+切换dev分支
$ git checkout -b dev

# 相当于
$ git branch dev # 创建分支
$ git checkout dev

# 查看当前分支,当前分支前面标有×号
$ git branch

# 切换回master分支
$ git checkout master

# 合并指定分支到当前分支
$ git merge dev

# 删除dev分支
$ git branch -d dev

# 查看分支合并情况
$ git log --graph --pretty=oneline --abbrev-commit
*   59bc1cb conflict fixed
|\
| * 75a857c AND simple
* | 400b400 & simple
|/
* fec145a branch test

# 删除feature1分支
$ git branch -d feature1

# 创建并切换dev分支
$ git checkout -b dev

# 修改readme.txt文件,并提交一个新的commit
$ git add readme.txt
$ git commit -m "add merge"

# 切换回master
$ git checkout master

# 合并dev分支,请注意--no-ff参数,表示禁用Fast forward
$ git merge --no-ff -m "merge with no-ff" dev

# 看看分支历史
$ git log --graph --pretty=oneline --abbrev-commit
*   7825a50 merge with no-ff
|\
| * 6224937 add merge
|/
*   59bc1cb conflict fixed

# 如果需要临时修复Bug,可以把当前工作现场“储藏”起来,等Bug修复后恢复现场后继续工作
$ git stash

# 此时查看工作区是干净
# 切换到需要修复Bug的分支,创建临时分支来修复
$ git checkout master
$ git checkout -b issue-101

# 修复完成后切换到master分支,完成合并,删除临时分支
$ git checkout master
$ git merge --no-ff -m "merged bug fix 101" issue-101
$ git branch -d issue-101

# Bug修复后,切换回dev分支继续干活
$ git checkout dev

# 查看工作现场列表
$ git stash list

# 恢复工作现场
$ git stash pop # 恢复的同时把stash内容也删了
$ git stash apply # 恢复,不删除stash的内容,使用git stash drop

# 再次查看工作现场列表,干净
$ git stash list

# 可以多次stash,恢复时指定恢复
$ git stash apply stash@{0}

# 强行删除一个没有合并过的分支
$ git branch -D <name>

# 要查看远程库的信息
$ git remote
$ git remote -v

# 推送其他分支
$ git push origin dev

# 从远程库clone,默认情况只能看到master分支,需要在dev分支,必须创建远程origin的dev分支到本地
$ git checkout -b dev origin/dev
$ git checkout -b branch-name origin/branch-name
$ git branch --set-upstream branch-name origin/branch-name # 关联

# 向远程库推送dev有冲突
$ git pull # 抓取到本地合并解决冲突,再向远程推送
$ git push origin dev

标签管理

# 切换到需要打标签的分支
$ git branch
$ git checkout master

# 创建标签
$ git tag v1.0

# 查看所有标签
$ git tag

# 给历史提高的commit id打标签
$ git log --pretty=oneline --abbrev-commit # 查看commit id
$ git tag v0.9 6224937

# 查看标签信息
$ git show v0.9

# 创建带有说明的标签
$ git tag -a v0.1 -m "version 0.1 released" 3628164

# 用PGP签名标签
$ git tag -s <tagname> -m "blablabla..."

# 推送某个标签到远程
$ git push origin v1.0

# 一次性推送全部尚未推送到远程的本地标签
$ git push origin --tags

# 删除远程标签
$ git tag -d v0.9 # 删除本地
$ git push origin :refs/tags/v0.9 # 删除远程

自定义 Git

# 显示颜色,会让命令输出看起来更醒目
$ git config --global color.ui true

# 忽略某些文件时,需要编写.gitignore,然后将.gitignore放到版本库中
# st就表示status
$ git config --global alias.st status

# 配置一个unstage别名
$ git config --global alias.unstage 'reset HEAD'
$ git unstage test.py # 等价于
$ git reset HEAD test.py

# 显示最后一次提交信息
$ git config --global alias.last 'log -1'

# log
git config --global alias.lg "log --color --graph --pretty=format:'%Cred%h%Creset -%C(yellow)%d%Creset %s %Cgreen(%cr) %C(bold blue)<%an>%Creset' --abbrev-commit"
# 每个仓库的配置文件放在.git/config
# 当前用户的配置文件放在用户主目录下的一个隐藏文件.gitconfig中

搭建 Git 服务器

1. 安装 git:

$ sudo apt-get install git

2. 创建一个 git 用户,用来运行 git 服务:

$ sudo adduser git

3. 创建证书登录:收集所有需要登录的用户的公钥,就是他们自己的 id_rsa.pub 文件,把所有公钥导入到 /home/git/.ssh/authorized_keys 文件里,一行一个

4. 初始化 Git 仓库:

# 选定一个目录作为 Git 仓库,假定是 /srv/sample.git,在 /srv 目录下输入命令

$ sudo git init --bare sample.git
# 把 owner 改为 git
$ sudo chown -R git:git sample.git

5. 禁用 shell 登录:

# 编辑/etc/passwd文件
git:x:1001:1001:,,,:/home/git:/bin/bash # 修改成下面的内容
git:x:1001:1001:,,,:/home/git:/usr/bin/git-shell

6. 克隆远程仓库,在各自的电脑上运行:

$ git clone git@server:/srv/sample.git

MRC和ARC混编

ARC 下使用MRC

在targets的build phases选项下Compile Sources下选择要不使用arc编译的文件,双击它,输入 -fno-objc-arc 即可

MRC 下使用ARC

在targets的build phases选项下Compile Sources下选择要使用arc编译的文件,双击它,输入 -fobjc-arc 即可

时光

时光你是多么可怕,将你带到这世间,又把你身边的人一个个带着离开这世间。最后也将把你带走。渺沧海之一粟,生命的意义又在哪里。

NSUserDefaults 简介,使用 NSUserDefaults 存储自定义对象

一、了解NSUserDefaults以及它可以直接存储的类型

NSUserDefaults是一个单例,在整个程序中只有一个实例对象,他可以用于数据的永久保存,而且简单实用,这是它可以让数据自由传递的一个前提,也是大家喜欢用它保存简单数据的一个主要原因。

使用 NSUserDefaults 存储自定义对象的最初,我们必须认识NSUserDefaults可以存储哪一些类型的数据,下面一一列出:

NSUserDefaults支持的数据类型有:NSNumber(NSInteger、float、double),NSString,NSDate,NSArray,NSDictionary,BOOL.

如果想要将上述数据类型的数据永久保存到NSUserDefaults中去,只需要简单的操作(一个Value 一个Key ),例如,想要保存一个NSString的对象,代码实现为:

    //将NSString 对象存储到 NSUserDefaults 中
    NSString *passWord = @"1234567";
    NSUserDefaults *user = [NSUserDefaults standardUserDefaults];
    [user setObject:passWord forKey:@"userPassWord"];

将数据取出也很简单,只需要取出key 对应的值就好了,代码如下:

    NSUserDefaults *user = [NSUserDefaults standardUserDefaults];
    NSString *passWord = [ user objectForKey:@"userPassWord"];

注意:对相同的Key赋值约等于一次覆盖,要保证每一个Key的唯一性

值得注意的是:

NSUserDefaults 存储的对象全是不可变的(这一点非常关键,弄错的话程序会出bug),例如,如果我想要存储一个 NSMutableArray 对象,我必须先创建一个不可变数组(NSArray)再将它存入NSUserDefaults中去,代码如下:

    NSMutableArray *mutableArray = [NSMutableArray arrayWithObjects:@"123",@"234"nil];
    NSArray * array = [NSArray arrayWithArray:mutableArray];
    
    NSUserDefaults *user = [NSUserDefaults standardUserDefaults];
    [user setObject:array forKey:@"记住存放的一定是不可变的"];

取出数据是一样的,想要用NSUserDefaults中的数据给可变数组赋值

先给出一个错误的写法:

    /*-------------------------错误的赋值方法-------------------*/
    NSUserDefaults *user = [NSUserDefaults standardUserDefaults];
    
    //这样写后,mutableArray 就变成了不可变数组了,如果你要在数组中添加或删除数据就会出现bug
    NSMutableArray *mutableArray = [user objectForKey:@"记住存放的一定是不可变的"];

正确的写法:

    /*-------------------------正确的赋值方法-------------------*/
    NSUserDefaults *user = [NSUserDefaults standardUserDefaults];
    
    //可以用alloc 方法代替
    NSMutableArray *mutableArray = [NSMutableArray arrayWithArray:[user objectForKey:@"记住存放的一定是不可变的"]];

二、使用 NSUserDefaults 存储自定义对象

1、将自定义类型转换为NSData类型

当数据重复而且多的时候(例如想存储全班同学的学号,姓名,性别(这个数据量可能太大了 )),如果不用SQLite 存储 (多数据最好还是用这个),你可以选择使用归档,再将文件写入本地,但是这种方式和 NSUserDefaults 比起来麻烦多了(因为NSFileManage 本来就挺复杂) ,但是问题是,NSUserDefaults 本身不支持自定义对象的存储,不过它支持NSData的类型,下面举一个例子来介绍。

我们先建立一个叫Student 的类,这个类里有三个属性(学号,姓名,性别),如图:

iOS5.png

我们要做的就是将Student类型变成NSData类型 ,那么就必须实现归档:

这里要实现 在.h 文件中申明 NSCoding 协议,再 在 .m 中实现 encodeWithCoder 方法 和

initWithCoder 方法就可以了 :

.h 中修改文件如图 :

114831_yrSH_1245365

.m中加入代码 :

115119_XzVi_1245365.png

这样做就可以将自定义类型转变为NSData类型了

2、将自定义类型数据存入 NSUserDefaults 中

如果要存储全班同学的信息,我们可以建一个NSMutableArray 来存放全班同学的信息(里面存储的全是NSData对象)在需要存储的地方加入代码:

//首先,要建立一个可变数组来存储 NSDate对象

     Student *student = [[Student alloc] ini];
     
    //下面进行的是对student对象的 name , studentNumber ,sex 的赋值
    student.name = @"lady-奕奕";
    student.studentNumber = @"3100104006";
    student.sex = @"女";
    
    //这是一个存放全班同学的数组
    NSMutableArray * dataArray = [NSMutableArray arrayWithCapacity:50];
    
    //将student类型变为NSData类型
    NSData *data = [NSKeyedArchiver archivedDataWithRootObject:student];
    
    //存放数据的数组将data加入进去
    [dataArray addObject:data];

如果你只想存一个人的信息,你可以直接将NSData存入NSUserDefaults中 :

    NSData *data = [NSKeyedArchiver archivedDataWithRootObject:student];   
    
    NSUserDefaults *user = [NSUserDefaults standardUserDefaults];
    [user setObject:data forKey:@"oneStudent"];

如果你想存储全班同学的信息,你还要用一个for循环将data 放入 dataArray中,这里具体的操作就不实现了,只给出存放的代码:

   //记住要转换成不可变数组类型
    NSArray * array [NSArray arrayWithArray:dataArray];
    
    NSUserDefaults *user = [NSUserDefaults standardUserDefaults];
    [user setObject:array forKey:@"allStudent"];

从NSUserDefaults中取出数据在还原也很简单

例如还原一个学生的数据:

NSUserDefaults *user = [NSUserDefaults standardUserDefaults];
 
 NSdData *data = [user objectForKey:@"oneStudent"];
    
 Student *student = [NSKeyedUnarchiver unarchiveObjectWithData:data];

总之,NSUserDefaults 在我们编写代码中是最常用的一个永久保存数据的方法,也是最简单的。

iOS清空NSUserDefaults下的内容
//方法一

NSString*appDomain = [[NSBundlemainBundle]bundleIdentifier];

[[NSUserDefaultsstandardUserDefaults]removePersistentDomainForName:appDomain];

//方法二

– (void)resetDefaults {

NSUserDefaults* defs = [NSUserDefaultsstandardUserDefaults];

NSDictionary* dict = [defsdictionaryRepresentation];

for(idkeyindict) {

[defsremoveObjectForKey:key];

}

[defssynchronize];

}

iOS第三方库 管理工具 CocoaPods

CocoaPods(第三方类库管理工具)

一, CocoaPods是什么?

每种语言发展到一个阶段,就会出现相应的依赖管理工具。随着 iOS 开发者的增多,业界出现了为 iOS 程序提供依赖管理的工具,它的名字叫做:CocoaPods.

CocoaPods现在已经成为 iOS 开发的依赖管理标准工具。开发 iOS 项目不可避免地要使用第三方库,CocoaPods 的出现使得我们可以节省设置和更新第三方开源库的时间。

在没有使用 CocoaPods 以前,我们需要:

  1. 把这些第三方开源库的源代码文件复制到项目中,或者设置成 git 的 submodule。
  2. 对于这些开源库通常需要依赖系统的一些framework,我需要手工地将这些 framework 分别增加到项目依赖中,比如通常情况下,一个网络库ASIHttpRequest(AFNetworking)就需要增加以下 framework: CFNetwork, SystemConfiguration, MobileCoreServices, CoreGraphics, zlib。
  3. 对于某些开源库,我还需要设置-licucore或者 -fno-objc-arc等编译参数
  4. 管理这些第三方库的更新。

这些体力活虽然简单,但毫无技术含量并且浪费时间。在使用 CocoaPods 之后,我只需要将用到的第三方开源库放到一个名为 Podfile 的文件中,然后执行pod install。 CocoaPods 就会自动将这些第三方开源库的源码下载下来,并且为我的工程设置好相应的系统依赖和编译参数

二, CocoaPods下载和安装?

安装方式异常简单, 在安装CocoaPods之前,首先要在本地安装好Ruby环境。 Mac 下都自带 ruby,使用 ruby 的 gem 命令即可下载安装:

$sudo gem install cocoapods

如果你的 gem 太老,可能也会有问题,可以尝试用如下命令升级 gem:

$sudo gem update —-system

另外,ruby 的软件源 https://rubygems.org 因为使用的是亚马逊的云服务,所以被墙了,需要更新一下 ruby 的源,使用如下代码将官方的 ruby 源替换成国内淘宝的源:

$gem sources —-remove https://rubygems.org/

等有反应之后再敲入以下命令:

$gem sources —a https://ruby.taobao.org/

为了验证你的Ruby镜像是并且仅是taobao,可以用以下命令查看:

$gem sources -l

上面所有的命令完成这时候,你再次在终端中运行:

$sudo gem install cocoapods

等上一分钟(根据网速),

最后再执行下面的命令:

$pod setup

成功后,  CocoaPods就可以在你本地下载并且安装好了。

还有一点需要注意,$pod setup在执行时,会输出Setting up CocoaPods master repo,但是会等待比较久的时间。这步其实是 Cocoapods 在将它的信息下载到~/.cocoapods目录下,如果你等太久,可以试着 cd 到那个目录,用du -sh *来查看下载进度。

你也可以参考接下来的使用 CocoaPods 的镜像索引的内容来提高下载速度。

如果上面的pod setup执行很慢,  我们可以用以下方式更换下载路径: (可选)

<如果半个小时内可以下载完成, 则忽略本步骤>

使用 CocoaPods 的镜像索引

所有的项目的 Podspec 文件都托管在https://github.com/CocoaPods/Specs。第一次执行pod setup时,CocoaPods 会将这些podspec索引文件更新到本地的 ~/.cocoapods/目录下,这个索引文件比较大。所以第一次更新时非常慢,更新了将近 1 个小时才完成。

一个叫 akinliu 的朋友在 gitcafe 和 oschina 上建立了 CocoaPods 索引库的镜像,因为 gitcafe 和 oschina 都是国内的服务器,所以在执行索引更新操作时,会快很多。如下操作可以将 CocoaPods 设置成使用 gitcafe 镜像:

$pod repo remove master

$pod repo add master https://gitcafe.com/akuandev/Specs.git

$pod repo update

如果要使用 oschina 上的镜像:  则将以上代码中的 https://gitcafe.com/akuandev/Specs.git 替换成 http://git.oschina.net/akuandev/Specs.git 即可.

三, CocoaPods的使用?

好了,安装好CocoaPods之后,接下来就是使用它。所幸,使用CocoaPods和安装它一样简单,也是通过一两行命令就可以搞定。

1, 利用CocoaPods,在项目中导入AFNetworking类库

AFNetworking类库在GitHub地址是:https://github.com/AFNetworking/AFNetworking

为了确定AFNetworking是否支持CocoaPods,可以用CocoaPods的搜索功能验证一下。在终端中输入:

$pod search AFNetworking

过几十秒钟之后,你会在终端中看到关于AFNetworking类库的一些信息。

接下来创建Xcode工程, 我们在项目中加入CocoaPods的支持;

CocoaPods将会自动下载AFNetworking并加入到工程当中呢,那如何来做呢?  你创建一个Podfile文件(注意,一定得是这个文件名,而且没有后缀),然后在里面添加你需要下载的类库,也就是告诉CocoaPods,“某某和某某和某某某,快到碗里来!”。每个项目只需要一个Podfile文件。

好吧,废话少说,我们先创建这个神奇的Podfile。在终端中进入你项目所在目录(cd命令)

$cd /Users/ijeff/Desktop/cocoaPodsDemo1

然后在当前目录下,利用vim创建Podfile,运行:

$vim Podfile

然后在Podfile文件中输入以下文字:

platform :ios, ‘7.0’

pod “AFNetworking”, “~> 2.0”

这两句文字的意思是,当前AFNetworking支持的平台: ios,

要下载的AFNetworking版本是2.0。

然后保存退出。vim环境下,保存退出命令是:

$:wq

这时候,你会发现你的项目目录中,出现一个名字为Podfile的文件,而且文件内容就是你刚刚输入的内容。

(注意,Podfile文件要在项目文件夹的根目录下。)

这时候,你就可以利用CocoaPods下载AFNetworking类库了。还是在终端中的当前项目目录下,运行以下命令:(–verbose –no-repo-update)

$pod install

运行完$pod install命令后会产生几个新文件。你会发现多了两个文件 “.xcworkspace”, “Podfile.lock”和一个文件夹“Pods”。

注意:  以后打开项目就用 .xcworkspace 打开,而不是之前的.xcodeproj文件。

至此,CocoaPods已经在你的项目中植入,管理了一个第三方库AFNetworking。上面写了这么多,其实过程是十分简单的。总结一下就是:

  1. 先在项目中创建Podfile,Podfile的内容是你想导入的类库。一般类库的原作者会告诉你导入该类库应该如何写Podfile;
  2. 运行命令:$pod install

使用$pod install进行安装时如果出现如下错误:

$ pod install

Analyzing dependencies

[!] Unable to find a specification for `AFNetworking (~> 2.0)`

尝试解决办法:

把当前Pod的目录清理一下就行了。在终端执行以下命令:
pod repo remove master
pod setup
setup成功后执行install

 2, 正确编译运行一个包含CocoaPods类库的项目

在网上下载的项目如果使用了CocoaPods, 则有时候会出现很多错误, 基本都是说你编译的这份代码找不到某某头文件,这就意味着你要成功编译就必须先导入一些第三方类库。同时你会发现在项目的根目录文件夹下面有三个跟CocoaPods相关的文件(文件夹):Podfile,Podfile.lock和Pods

这时候,打开终端,进入项目所在的根目录,也就是和Podfile在同一目录下,输入以下命令(由于已经有Podfile,所以不需要再创建Podfile):

$pod update

执行完后,再回到工程根目录文件夹看一看,会看到多了一个文件.xcworkspace, 最后就点击该文件打开项目即可;