Git

日期:08/03/2017 - 08/07/2017


一、基本概念

版本控制系统:记录某项(常为多人)工作的进度历史,enable项目版本查看和回溯,并且能够便捷合并多人的同时性工作的系统。分布式版本控制系统的代表是Git。集中式版本控制系统的代表有SVN、CVS。
集中式版本控制系统:拥有中央服务器,过往版本、当前版本都存放在中央服务器上,成为系统的瓶颈,并且不能脱机使用。
分布式版本控制系统:没有中央服务器或中央服务器仅起辅助作用,个人端存放完整的版本库。
2005年,Linus为管理开源的Linux代码库而用两周时间编写了Git。


二、本机操作

由于Git原本运行于Linux,所安装的windows版本中(似乎)集成了Linux的shell,可以运用Linux的命令行指令进行操作。Git提交的事实上是对文件的修改(而非文件本身的内容)。若在文件末尾不换行,则提交的修改中总是包括原文档最后一行的删除修改和其增加修改。
Git在逻辑上分为工作区(Working Directory),暂存区(stage/index)和版本库区。本机文件状态(实则其修改情况)为工作区。工作区工作完成后,需将工作区文件提交到暂存区。暂存区确认无误后,将暂存区内容提交到版本库,即实现版本更新。

>> git init

指令将当前所在目录建成一个git版本库(repository)。
Git仅能记录纯文本文件版本。编写时使用支持多国语言的UTF-8编码可以规避大量编码问题。Windows自带的记事本在记录UTF-8编码时的多余前缀导致其不能被正常识别,不应用来编辑Git管理的文本文件。不应使用包含中文的文件名/路径。

>> git add <filename1> <filename2> ...

将(目录下的)文件添加到暂存区。

>> git rm <filename>

提交一个修改到暂存区,此修改是删除特定文件。

>> git commit -m "<note>"

提交整个版本库。

>> git status

查看版本库的状态。changes not staged for commit: 被修改的文件尚未add; changes to be committed: 已经add,等待commit。

>> git diff

查看已提交的版本库与现有版本库(本机文件)之间的区别。

>> git log (--pretty=oneline)

查看版本库的往昔所有commit记录(显示版本号和相应的注释)。

>> git reset --hard <version>

将版本库(本机文件)回退为指定版本。版本指代格式:HEAD(当前版本),HEAD^^(上上个版本),HEAD~100(往前一百个版本),版本号(可仅输入前x位,一般七位足够)。

>> git reflog

显示你对每个版本库执行过的操作,包括添加版本库、重置版本库。可以通过此指令找到被回退的版本库号。

>> git checkout -- <filename>

用版本库或暂存区的文件替换本机文件。这意味着,若文件尚未被add,则会回退到当前版本库状态;若文件已被add,其会回退到add时的状态(当add后执行修改时,这些修改会被抹除)。

>> git reset HEAD <filename>

将暂存区的指定文件扔掉。即撤销add指令。


三、远程仓库

SSH:一种网络协议,用于计算机之间的加密登录。1995年由芬兰学者Tatu Ylonen设计,目前成为Linux计算机的标配加密方式。Putty是Windows操作系统上的一种提供SSH加密协议的软件。SSH采用的是公钥加密方法,即用户向远程主机发送登录请求,远程主机回复一个公钥,用户用公钥将账号密码加密后发送给远程主机。公钥加密很容易受到中间人攻击(Man-in-the-middle attack),即有人在中间截获用户的请求,伪造公钥与用户通信从而获取密码。但SSH协议中在初次采用公钥之前会向用户确认是否采用特定指纹的公钥。要登录的远程主机可以将公钥指纹(长公钥的一个缩短化处理)告知用户,使用户得以检查。通过用户手动信任某些特定公钥,SSH协议保护了用户不受中间人攻击。
公钥与私钥:不对称加密方式。用公钥加密的内容只能用私钥解密,用私钥加密的内容只能用公钥解密[3]。若用他人公钥加密后发送信息,则可确保收信人身份;若用自己私钥加密后发送信息,则可证明发信人身份。

>> ssh-keygen -t rsa -C "youremail@example.com"

采用RSA算法创建本机的的SSH密钥对(公钥和私钥)。RSA算法下的密钥长达1024位,存储在.ssh/id_rsa和.ssh/id_rsa.pub。Git软件会自动识别文件夹中的公钥(pub)并将其添加到信任列表中。只有信任列表中的公钥有权对远程版本库进行修改。

>> git remote add origin git@github.com:用户名/版本库名.git

将当前库与GitHub上的某个库关联起来。关联后,远程库的名字默认叫做origin。

>> git push -u origin master

将当前库推送到其叫做origin的远程库。-u参数在第一次推送时使用,可以将当前库的master与远程库的master分支关联(???)。再次推送时可省略此参数。
初次连接GitHub时要求信任GitHub的公钥指纹
向GitHub推送修改需要权限。将SSH公钥添加到GitHub账户设置中的允许访问列表中即可。此时我遇到了第一个真正的bug。虽然已经添加了公钥,但我还是无法将本机内容推送上去。这可能是在关联两个库的时候尚未添加公钥导致的。

Git-Bug-1

>> git remote rm origin

删除远程库origin。之后重新添加再add,显示似乎有未尽工作分支,所以推送失败。
Git-Bug-2
>> git pull

拉取远程库状态。
Git-Bug-3

这显然是我使用旧的远程库与本机版本库关联的结果,尽管旧库已经空了。Git的分支合并真差劲,差评!重新创建新库后一切顺利。

>> git clone git@github.com:<username>/<repositoryname>.git

将远程库克隆到本机。可以视为即在本机创建一个空库、将两个库关联、并将远程库的内容拉取到本机文件。

四、多人协作

Git通过分支来实现多人协作。分支的实质是树形结构+指针。HEAD是一个指向当前分支的指针的指针,当前分支的指针指向分支下最近的一次修改(或回退后的某个修改)。master是默认的分支名称。
基于Git的强大分支功能,可以设计多种项目开发结构。安全的主分支(稳定版本)、便捷的个人开发分支、灵活的小分支、语法清晰的注解是好的项目开发系统的必要特征。

一种项目开发结构示意图[1]
>> git remote

查看远程版本库信息。
>> git branch
查看当前分支列表。

>> git checkout -b <branchname>
//or
>> git branch <branchname>
>> git checkout <branchname>

创建并切换到特定分支。新创建的分支默认与当前分支指向同一修改节点。切换分支后,本机文件也被更改为当前分支的状态。

>> git merge <branchname>

将指定分支合并到当前分支。

>> git branch -d <branchname>

删除指定分支。不能删除当前所在的分支。删除master时候会被警告(???)。

>> git log --graph --pretty=oneline --abbrev-commit

查看当前分支状态(带树形图)。

>> git merge --no-ff -m "note" <branchname>

合并分支但保留被合并的分支结构,方便回顾分支开发过程和继续开发分支。需要为此操作提交注解。

>> git tag <tagname> (<versionname>)
>> git tag -a <tagname> -m "<note>"

在当前分支上打一个标签(实际是一个有名字的固定指针)。若不提供版本号,则打在当前最新commit的版本上。

>> git tag

查看所有标签。

>> git show <tagname>

查看标签所在版本的详细信息。

>> git tag -d <tagname>

删除标签。

>> git push <remotebranchname> <tagname>

将本地标签推送到远程版本库。

>> git push <remotebranchname> --tags

将全部本地标签推送到远程版本库。

>> git push origin :refs/tags/<tagname>

删除远程标签。

>> git stash

将工作区和暂存区的内容(修改)保存起来,清空这两个区域。配合分支切换,这一操作适用于临时一次性执行其他优先任务,如其他分支上的bug修改。

>> git stash list

查看本机上的stash列表。

>> git stash pop   //only for the first stash
//or
>> git stash apply <stashindex>
>> git stash drop <stashindex>

恢复stash并删除。

如需更改远程库,需使用push指令,将本机分支推送到远程分支上。

六、开源项目

fork:将他人的远程版本库复制到自己的账号下(值传递)。
pull:向他人推送(xiao)自己修改后的版本库。若他人采纳,即参与开源完成。


七、参考文献

[1] Git教程
[2] SSH与远程登录
[3] 公钥和私钥
[4] GitHub常见错误
[5] Markdown语法

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,569评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,499评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,271评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,087评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,474评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,670评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,911评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,636评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,397评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,607评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,093评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,418评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,074评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,092评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,865评论 0 196
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,726评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,627评论 2 270

推荐阅读更多精彩内容