Git基础知识与简单用法

本文包括git的简介、基本使用方法、部分原理
这么好用的工具必须安利大家使用,真的是越使用越觉得其强大的功能。
用过图形界面和命令行,还是命令行下面使用的得心用手,推荐大家了解原理后使用命令行。

相关文档


安装


Windows

下载相应包安装即可

Linux

  $ sudo yum install git #yum软件包管理工具
  $ sudo apt-get install git #基于Debian的发行版

Mac OS X

Terminal 里尝试首次运行 git 命令即可,会提示你安装

源码安装

命令校验是否安装成功

$ git --version

概述


概念

  • 直接记录快照,而非差异比较
  • 近乎所有操作都是本地执行
  • Git 保证完整性
  • Git 一般只添加数据
  • 三种状态
    • 已提交(committed) 对应 .git directory(Repository) 工作区域
    • 已修改(modified) 对应 Working Directory工作区域
    • 已暂存(staged/Index) 对应 Index/Staging Area工作区域

配置

  • 系统级别:/etc/gitconfig,通过git config --system命令来配置
  • 用户级别:~/.gitconfig或者~/.config/git/config,通过git config --global来配置
  • 仓库级别:.git/config
# 设置用户名
$ git config --global user.name "xxx"
# 设置email
$ git config --global user.email "xxx@xx.com"
# 设置默认文本编辑器,当Git需要你输入信息的时候会调用它
$ git config --global core.editor emacs
# 检查配置信息
$ git config --list
# 设置credential cache,省得多次输入密码
$ git config --global credential.helper cache

获取帮助

# 有关命令的更多信息
$ git <verb> --help
$ git help <verb>
$ man git-<verb>

基础文件操作


获取Git仓库

  • git init将现有目录和文件初始化一个Git仓库,会生成一个.git文件夹(这个文件夹就是本地Git仓库)
  • git clone [url]来克隆现有的仓库
$ git clone https://github.com/catchance/git-learning.git git-learning

检查当前文件状态

$ git status

git status命令的功能

  1. 查看当前工作区的状态

忽略文件

创建.gitignore文件来来将某些文件不纳入Git管理

# 忽略所有以 .o 或 .a 结尾的文件
*.[oa]
# Git 忽略所有以波浪符(~)结尾的文件
*~
  • 匹配模式可以以/开头防止递归
  • 匹配模式可以以/结尾指定目录
  • 所有空行或者以 # 开头的行都会被 Git 忽略。
  • 可以使用标准的 glob 模式匹配(指 shell 所使用的简化了的正则表达式。)
    1. 星号(*)匹配零个或多个任意字符;
    2. [abc] 匹配任何一个列在方括号中的字符
    3. 问号(?)只匹配一个任意字符
    4. 使用两个星号(*) 表示匹配任意中间目录a/**/z
    5. 方括号中使用短划线分隔两个字符,表示所有在这两个字符范围内的都可以匹配[0-9]

跟踪文件,将文件添加到暂存区,文件处于暂存状态。

# 跟踪文件,文件处于暂存状态 默认是忽略溢出的操作
$ git add <file>...
# 跟踪文件,文件处于暂存状态 同时对于删除的操作进行记录
$ git add --all <file>...
# 将文件撤出未跟踪的状态
$ git reset HEAD <file>...
# 将已经修改的文件还原成HEAD索引指向的版本
$ git checkout -- <file>...
  • HEAD指向当前所在分支,将HEAD想象为当前分支的别名。
  • git add命令的功能:
    1. 可以用它开始跟踪新文件
    2. 把已跟踪的文件放到暂存区
    3. 合并时把有冲突的文件标记为已解决
  • git checkout --命令
    1. 撤销对工作区修改,这个命令是以最新的存储时间节点(add和commit)为参照
    2. 覆盖工作区对应文件file;这个命令改变的是工作区
  • git reset命令
    1. 清空add命令向暂存区提交的关于file文件的修改(Ustage)
    2. 这个命令仅改变暂存区,并不改变工作区

对比文件信息

# 对比工作目录中的文件和暂存区的区别
$ git diff
# 对比暂存区的文件和上一次提交的不同
$ git diff --cached
# 效果等同于 --cached命令
$ git diff --staged
# 可以使用diff插件来分析结果
$ git difftool --tool-help

git show命令

  • 显示具体的代码改动情况
    # 查看某次commit的修改内容
    $ git show <commit-hash-id>
    

提交更新

# 提交更新
$ git commit -m "message"
# 跳过暂存区直接提交
$ git commit -a -m "message"
  • 提交时记录的是放在暂存区域的快照
  • 每一次运行提交操作,都是对你项目作一次快照,以后可以回到这个状态,或者进行比较。

移除文件

# 将删除的文件提交 rm <file>
$ git add --all <file>... 提交工作空间中删除的文件
# 同 git add 命令一样的
$ git rm <file>
# 如果工作空间不想删除,而只是想删除Git跟踪的文件的话
$ git rm --cached <file>...

移动文件

# Git 并不显式跟踪文件移动操作
# 如果在 Git 中重命名了某个文件,仓库中存储的元数据并不会体现出这是一次改名操作
# git mv命令相当于 mv xx xxx git rm xx git add xxx
$ git mv <file> <file>

查看提交历史

git log命令的输出格式多种多样,自行参考帮助文档。

# 查询提交历史
$ git log
# 常用提交信息
$ git log --oneline --graph --all --decorate
# -2 用来显示最近两次提交,-p用来显示每次提交的内容差异
$ git log -p -2
# 查看某个文件的修改历史
$ git log -p <filename>
# 除了显示基本信息之外,还附带了每次 commit 的变化
$ git log --stat
# 可以指定使用不同于默认格式的方式展示提交历史
$ git log --pretty=oneline
# 添加了一些ASCII字符串来形象地展示你的分支、合并历史
$ git log --graph
# 显示某个路径或者文件的提交历史
$ git log -- <path>

查询引用变更的记录

# 查询操作的相关记录
$ git reflog

撤销操作

# 重新提交命令
# 这个命令会将暂存区中的文件提交,如果快照会保持不变,而你所修改的只是提交信息。
$ git commit --amend
# 撤销暂存区的记录,工作区的文件不变。
$ git reset HEAD <path>
# 重置工作区中的文件,从暂存区或者上次提交中重置文件。
$ git checkout -- <path>

远程仓库


查看远程仓库信息

# 列出你指定的每一个远程服务器的简写
$ git remote  
#会显示需要读写远程仓库使用的 Git 保存的简写与其对应的 URL。
$ git remote -v
# 要查看某一个远程仓库的更多信息
$ git remote show origin

添加远程仓库

# 添加一个新的远程 Git 仓库,同时指定一个你可以轻松引用的简写
$ git remote add <shortname> <url>

从远程仓库中抓取与拉取

# 从远程仓库中获得数据,它并不会自动合并或修改你当前的工作
$ git fetch [remote-name]
# 通常会从最初克隆的服务器上抓取数据并自动尝试合并到当前所在的分支。
$ git pull [remote-name]

推送到远程仓库

# 要将 master 分支推送到 origin 服务器时 `git push [remote-name] [branch-name]`
$ git push origin master

远程仓库的移除与重命名

# 修改你的远程分支名字
$ git remote rename pb paul
$ git remote
# 移除一个远程仓库
$ git remote rm paul
$ git remote

打标签

# 过滤显示定义的标签
$ git tag -l 'v1.8.5*'
# 创建标签
$ git tag -a v1.4 -m 'my version 1.4'
# 轻量标签 轻量标签本质上是将提交校验和存储到一个文件中 - 没有保存任何其他信息。
$ git tag v1.4-lw
# 后期打标签 9fceb2每次提交的校验和
$ git tag -a v1.2 9fceb2  
# 共享标签 默认情况下,git push 命令并不会传送标签到远程仓库服务器上。
# 在创建完标签后你必须显式地推送标签到共享服务器上。
$ git push origin [tagname]
# 检出标签 `git checkout -b [branchname] [tagname]`
$ git checkout -b version2 v2.0.0

Git别名

# 取消暂存文件别名
git config --global alias.unstage 'reset HEAD --'
# 轻松地看到最后一次提交
git config --global alias.last 'log -1 HEAD'

Git分支


创建分支

创建分支,就相当于在当前分支上面创建一个指针。
Git有一个HEAD的特殊指针,指向当前所在的本地分支。

# 创建分支
$ git branch testing

stash的常用操作

$ git stash  #保存当前的修改
$ git stash pop #弹出stash中最顶端的即stash<0>的内容
$ git stash list #查看stash中所有的改动
$ git stash apply @stash<id> #弹出stash<id>的修改
$ git stash pop @stash<id> #弹出stash<id>的修改
$ git stash clear  #清除所有的stash
$ git stash drop @stash<id>  #移除stash<id>

切换分支

切换分支的时候通过保存进度(stashing) 和 修补提交(commit amending)
保持好一个干净的状态。

$ git checkout testing
$ git checkout master
# 创建分支并且切换分支
$ git checkout -b <branchname>

删除分支

# 删除分支
$ git branch -d <branchname>

合并分支

fast-forward分支所指向的提交是你当前提交的直接上游

# 合并test分支到当前分支
$ git merge test

冲突的解决

分支管理(开发分支流)

  • 长期分支 用来维护不同层次的稳定性
    1. master分支 只保留完全稳定的代码
    2. develop/next分支 后续开发和测试稳定性
    3. pu(proposed updates) 建议更新分支,不成熟的内容不适合放入master和develop的
    4. test 测试部门使用的需要测试的分支
  • 短期分支,特性分支
    1. issue分支 特性分支
    2. hotfix分支 紧急修复补丁分支

远程分支

$ git remote show <remote>
$ git ls-remote <remote>
  • 推送
      $ git push <remote> <branch>
      # 将本地的 localbranch 分支推送到远程仓库上的 serverfix 分支
      $ git push origin localbranch:serverfix
      # 运行 git merge origin/serverfix 将这些工作合并到当前所在的分支
      $ git merge origin/serverfix
    
  • 跟踪分支
      # 查看设置的所有跟踪分支
      $ git branch -vv
      # 在自己的 serverfix 分支上工作,可以将其建立在远程跟踪分支之上
      $ git checkout -b serverfix origin/serverfix
      # 上面命令的快捷方式
      $ git checkout --track origin/serverfix
      # 设置已有的本地分支跟踪一个刚刚拉取下来的远程分支
      $ git branch -u origin/serverfix
    
  • 拉取
      # 抓取本地没有的数据时,它并不会修改工作目录中的内容
      $ git fetch
      # 相当于fetch然后紧接着一个merge命令
      $ git pull
    
  • 删除远程分支
      # 删除分支
      $ git push origin --delete serverfix
    

变基rebase

整合分支的两种方式:rebase和merge
merge:它会把两个分支的最新快照以及二者最近的共同祖先进行三方合并,合并的结果是生成一个新的快照并提交
rebase:提取在分支中所做的补丁和修改,然后在需要合并的分支上再应用一次。用于把一个分支的修改合并到当前分支。rebase用来修改本地私有提交历史的。

  # git rebase [basebranch] [topicbranch]
  # 在git rebase后面加上参数<branch>,那么会先执行git checkout到这个分支,如果没有加分支则表示对当前分支进行操作
  $ git rebase master server
  # git add 解决冲突后继续rebase
  $ git rebase --continue  
  # --abort参数来终止rebase的行动,并且"mywork" 分支会回到rebase开始前的状态。
  $ git rebase --abort
# 想维持树的整洁
$ git fetch origin master
$ git rebase origin/master
$ git push

不要对在你的仓库外有副本的分支执行rebase操作

Git Revert原理:根据你要回退的提交所做的改动做相反的改动,然后重新提交代码,使代码达到没有这些旧提交所能达到的状态。

使用git reset是不影响远程分支的,一切都在本地发生。如果回退需要很快影响远程分支的,应该使用git revert

规范


  • 如果某些文件已经被跟踪了, 再放入到.gitinore可能会失效, 用以下命令来忽略

    # 忽略文件
    $ git update-index --assume-unchanged filename
    # 撤销用:
    $ git update-index --no-assume-unchanged filename
    
  • git diff --check 找出可能的空白错误并将它们列出来

  • 本地分支和远程分支的绑定(tracking),加上 rebase 策略

    [branch "master"]
        remote = origin
        merge = refs/heads/master
        rebase = true
    
  • 解决多个问题,最好不要一次提交,根据任务分开提交。

  • 每次提交信息的模板

    修改的摘要(50 个字符或更少)

    如果必要的话,加入更详细的解释文字。在
    大概 72 个字符的时候换行。在某些情形下,
    第一行被当作一封电子邮件的标题,剩下的
    文本作为正文。分隔摘要与正文的空行是
    必须的(除非你完全省略正文);如果你将
    两者混在一起,那么类似变基等工具无法
    正常工作。

    空行接着更进一步的段落。

    • 句号也是可以的。

    • 项目符号可以使用典型的连字符或星号
      前面一个空格,之间用空行隔开,
      但是可以依据不同的惯例有所不同。

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

推荐阅读更多精彩内容