Git学习

git

详细教程来源:

安装初始化

  • git config --global user.name "github用户名"
  • git config --global user.email "邮箱"
    • --global 更改的配置文件位于用户目录下的,即该用户下所有项目默认使用该配置,文件保存在~/.gitconfig文件
    • --system 系统中对所有用户普遍适用的配置
    • 去掉–global,重 新配置设定保存在.get/config文件里
  • 使用git config --list 查看已有配置信息

创建版本库

  • mkdir learngit创建一个新目录
  • git init 初始化一个本地git仓储

添加文件及提交

  • git status 查看本地仓储的变更状态

  • git add . /git add --all添加所有文件如$ git add file2.txt file3.txt

  • git add 文件名 添加单个文件或多个

  • .gitignore文件 添加本地git 忽略清单文件 window文件新建.开头文件夹 要用命令mkdir .gitignore 新建文件 echo '' >> .aaa 或sublime新建

  • git commit -m "message..."提交被托管的文件变化到本地仓储

    • git commit 注释多的时候,不加-m ,打开vim写注释
    • git commit -a -m "message..." -a表示跳过使用暂存区提交, 不用add,已经跟踪过的才可-a

版本回退

  • git status 查看本地仓储的变更状态
  • git diff
    • git diff 查看哪些更新还没暂存
    • git diff --cachedgit diff --staged 哪些更新已暂存还未提交
  • git log 查看提交日志,最近的一次是append GPL,上一次是add distributed,最早的一次是wrote a readme file
    • git log --pretty=oneline 显示一行 哈希值全显
    • git log --oneline 显示一行 哈希值简显
    • git log --oneline --decorate --graph --all 全部分支
  • git reset --soft HEAD~ 只改head回到上个版本,工作区暂存区不变
  • git reset [--mixed] HEAD~ 改head和暂存区,不改工作区
  • git reset --hard 2d2250 前六位版本强制回归版本
  • git reset --hard HEAD^ head最新版本,回到上个版本,git reset --hard HEAD^^回到上上个版本,上100个版本HEAD~100
  • git reflog查看命令历史回新版本

工作区和暂存区

工作区就是learngit ,工作区有一个隐藏目录.git,是Git的版本库。Git的版本库里有stage(或者叫index)的暂存区,还有自动创建的第一个分支master,以及指向master的一个指针叫HEAD。

stage示意图

  • git add把文件添加进去,是把文件修改添加到暂存区;
    暂存区状态
  • git commit提交更改,是把暂存区的所有内容提交到当前分支。
    新建license.txt 修改readme.txt 再分别git add 使用git status查看stage:
    提交后状态

撤销修改

  • 场景1:当改了工作区某个文件的内容,撤销修改用命令git checkout -- file。后推荐使用 git restore filename
  • 场景2:当改了工作区某个文件的内容,还添加到了暂存区时,想丢弃修改,分两步,第一步用命令git reset HEAD file,就回到了场景1,第二步按场景1操作。
  • 场景3:已经提交了不合适的修改到版本库时,想要撤销本次提交,参考版本回退一节,不过前提是没有推送到远程库。
    • 修改commit 重新注释: git commit –amend

删除文件,重命名文件

  • rm file 查看git status
  • 确认删除则git rm file ,git commit -m “”
  • 误删则 git checkout – file
  • 暂存区回退到工作区,本地文件不删除:git rm –cached file
  • 已提交删除文件,本地文件删除(自动添加到暂存区): git rm file
  • 已提交重命名(自动添加到暂存区):git mv old new

远程仓库

  • 创建SSH Key。
    ssh-keygen -t rsa -C "youremail@example.com"

  • 登陆GitHub,打开“Account settings”,“SSH Keys”页面:

ssh keys

  • 点“Add SSH Key”,填上任意Title,在Key文本框里粘贴id_rsa.pub文件的内容。

添加远程库和删除

  • 登陆GitHub,创建一个新的仓库learngit。
  • 在本地learngit仓库下运行命令:
    git remote add origin git@github.com:yourusername/xxx.git

yourusername为github账户名,远程库的名字就是origin。

  • git remote rm origin删除远程 Git 仓库
  • git remote -v 远程 Git 仓库地址

    推送到远程库

  • git push -u origin master把当前分支master推送到远程。

第一次推送加上了-u参数,把本地的master分支和远程的master分支关联起来,在以后的推送或者拉取时就可以简化命令git push origin master

从远程库克隆

  • git clone git@github.com:yourusername/xxx.git默认的git://使用ssh
    还可以用https://github.com/用户名/库名.git使用https协议每次推送都必须输入口令

创建删除与合并分支

  • git checkout -b <name>-b参数表示创建并切换
  • git branch <name>创建<name>分支
  • git checkout <name>切换到<name>分支
    • 切换分支前保证当前分支已提交,以免污染其他分支。未提交切换,切换后的分支存在未提交的文件,切换回原分支 提交,再切换到要切换的分支,该文件不存在了。已提交后再修改未提交,切换分支 不给切换。
  • git branch命令查看显示分支,当前分支前面会标一个*
  • git branch -v 查看最后一次提交
  • git branch -vv 查看设置的所有跟踪分支
  • git merge <name>合并某分支到当前分支
  • git branch -d <name> 删除分支,切换到别的分支再删该分支
  • git branch -D <name> 强制删除,有内容时
  • git branch name <commithash> 新建分支回到该哈希版本
  • git push --delete origin <name>删除远程分支
    • git remote prune origin --dry-run 列出仍在远程跟踪但是远程已被删除的无用分支
    • git remote prune origin清除上面命令列出来的远程跟踪
  • 新建分支gh-pages 固定名字,https://用户名.github.io/库名/即可访问html

解决冲突

  • 新建分支git checkout -b feature1,修改readme.txt改成Creating a new branch is quick AND simple.,在feature1分支上提交:
  • 切换到master分支$ git checkout master在master分支上修改readme.txt,改成Creating a new branch is quick & simple.`提交
  • git merge feature1,Git无法执行“快速合并”,只能试图把各自的修改合并起来,
    需手动解决冲突后再提交,修改成Creating a new branch is quick and simple.再git,commit
  • 用带参数的 git log --graph --pretty=oneline --abbrev-commit可以看到分支的合并情况:

分支管理策略

合并分支时,git merge --no-ff -m "merge with no-ff" dev加上–no-ff参数就可以用普通模式合并,合并后的历史有分支,能看出来曾经做过合并,而fast forward合并就看不出来曾经做过合并。

Bug分支

修复bug时,我们会通过创建新的bug分支进行修复,然后合并,最后删除;
当手头工作没有完成时,先把工作现场git stash一下,然后去修复bug,修复后,再git stash pop,回到工作现场。

  • git stash
  • git checkout master
  • git checkout -b issue-101
  • git add readme.txt
  • git commit -m "fix bug 101"
  • git checkout master
  • git merge --no-ff -m "merged bug fix 101" issue-101
  • git branch -d issue-101
  • git checkout dev
  • git status
  • git stash list

需要恢复一下,有两个办法:

  • git stash apply恢复,但是恢复后,stash内容并不删除,你需要用git stash drop来删除;

  • git stash pop,恢复的同时把stash内容也删了:

  • git stash list查看,就看不到任何stash内容了:

  • $ git stash list

可多次stash,恢复的时候,先用git stash list查看,然后恢复指定的stash,用命令:

$ git stash apply stash@{0}

Feature分支

添加一个新功能,最好新建一个feature分支,在上面开发,完成后,合并,最后,删除该feature分支:

  • git checkout -b feature-vulcan
  • git add vulcan.py
  • git status
  • git commit -m "add feature vulcan"
  • git checkout dev

若新功能必须取消,

  • git branch -d feature-vulcan

强行删除:

  • git branch -D feature-vulcan

多人协作

  • 在另一台电脑(注意要把SSH Key添加到GitHub)或者同一台电脑的另一个目录下克隆:

    git clone git@github.com:michaelliao/learngit.git

  • 查看远程库信息,使用git remote -v
  • 默认情况下,只能看到本地的master分支,用git branch命令看看
  • 本地新建的分支如果不推送到远程,对其他人就是不可见的;

  • 从本地推送分支,使用git push origin branch-name,如果推送失败,先用git pull抓取远程的新提交;

  • 在本地创建和远程分支对应的分支,使用git checkout -b branch-name origin/branch-name,本地和远程分支的名称最好一致;

  • 建立本地分支和远程分支的关联,使用git branch --set-upstream branch-name origin/branch-name

  • 从远程抓取分支,使用git pull,如果有冲突,要先处理冲突。

  • 修改后git addgit commit -m ""git push origin branch-name

创建标签

  • git tag <name>打一个新标签,默认标签是打在最新提交的commit上的,若要找之前的版本打标签,方法是找到历史提交的commit id,然后打上如 git tag <name> commit_id
  • git tag查看所有标签:
  • git show <tagname>查看标签信息,标签不是按时间顺序列出,而是按字母排序的
    git show v0.9
  • git tag -a <tagname> -m "blablabla..."可以指定标签信息;-a指定标签名,-m指定说明文字:
  • git tag -s <tagname> -m "blablabla..."可以用PGP签名标签,首先安装gpg(GnuPG),如报错要配置Key

操作标签

  • 命令git push origin <tagname>可以推送一个本地标签;

  • 命令git push origin --tags可以推送全部未推送过的本地标签;

  • 命令git tag -d <tagname>可以删除一个本地标签;

  • 命令git push origin :refs/tags/<tagname>可以删除一个远程标签。

忽略特殊文件

在Git工作区的根目录下创建.gitignore文件,把要忽略的文件名填进去,Git就会自动忽略这些文件。所有配置文件.gitignore文件

忽略文件的原则是:

  • 忽略操作系统自动生成的文件,比如缩略图等;
  • 忽略编译生成的中间文件、可执行文件等,也就是如果一个文件是通过另一个文件自动生成的,那自动生成的文件就没必要放进版本库,比如Java编译产生的.class文件;
  • 忽略你自己的带有敏感信息的配置文件,比如存放口令的配置文件。

要添加的文件被.gitignore忽略了,用-f强制添加:
git add -f App.class

用git check-ignore命令检查:

1
2
$ git check-ignore -v App.class
.gitignore:3:*.class App.class

配置别名

1
2
3
4
$ git config --global alias.st status
$ git config --global alias.co checkout
$ git config --global alias.ci commit
$ git config --global alias.br branch

命令git reset HEAD file可以把暂存区的修改撤销掉(unstage),重新放回工作区,配置一个unstage别名:
$git config --global alias.unstage 'reset HEAD'

配置一个git last,让其显示最后一次提交信息:
git config --global alias.last 'log -1'

配置lg:

1
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 lg的效果:

lg的效果图

配置文件

  • 配置Git的时候,加上–global是针对当前用户起作用的,如果不加,那只针对当前的仓库起作用。每个仓库的Git配置文件都放在.git/config文件中.
  • 别名就在[alias]后面,要删除别名,直接把对应的行删掉即可。
  • 当前用户的Git配置文件放在用户主目录下的一个隐藏文件.gitconfig

    git 内部原理–底层

.git 文件夹

  • config 文件包含项目特有的配置选项
  • description 描述
  • HEAD 文件指向目前被检出的分支
  • hooks/ 目录包含客户端或服务端的钩子脚本(hook scripts)
  • info/ 目录包含一个全局性排除(global exclude)文件, 用以放置那些不希望被记录在 .gitignore 文件中的忽略模式(ignored patterns)。
  • objects/ 目录存储所有数据内容
  • refs/ 目录存储指向数据(分支、远程仓库和标签等)的提交对象的指针
  • index git init时无该文件,文件保存暂存区信息。

    git 底层命令

  • git hash-object -w --stdin git hash-object 会接受你传给它的东西,而它只会返回可以存储在 Git 仓库中的唯一键。 -w 选项会指示该命令不要只返回键,还要将该对象写入数据库中。 最后,–stdin 选项则指示该命令从标准输入读取内容;若不指定此选项,则须在命令尾部给出待存储文件的路径。git hash-object -w test.txt
  • find .git/objects -type f 查看对象数据库记录文件,Git 存储内容的方式——一个文件对应一条内容, 以该内容加上特定头部信息一起的 SHA-1 校验和为文件命名
  • git cat-file -p 哈希值 查看内容。git cat-file -p 99f1a6d12cb4b6f19c8655fca46c3ecf317074e0
  • git cat-file -t 哈希值 查看对象类型,git对象为blob类型,树对象为tree类型。 git cat-file -t 1f7a7a472abf3dd9643fd615f6da379c4acb3e3a
  • 创建一个树对象
    • git update-index命令创建一个暂存区。git update-index --add --cacheinfo 100644 \ 83baae61804e65cc73a7201a7252750c76066a30 test.txt 让git对象对应上文件名
    • git write-tree 命令将暂存区内容写入一个树对象。git write-tree d8329fc1cc938780ffdd9f94e0d364e0ea74f579
  • 提交对象echo 'xxcomment' | git commit-tree d8329f
  • git ls-files -s 查看暂存区显示有关索引和工作树中文件的信息
-------------本文结束 感谢您的阅读-------------
显示 Gitment 评论