转载自https://github.com/xirong/my-git/blob/master/useful-git-command.md
Repository
- 检出(clone)仓库代码:
git clone repository-url
/git clone repository-url local-directoryname
- 例如,clone jquery 仓库到本地:
git clone git://github.com/jquery/jquery.git
- clone jquery 仓库到本地,并且重命名为 my-jquery :
git clone git://github.com/jquery/jquery.git my-jquery
- 例如,clone jquery 仓库到本地:
- 查看远程仓库:
git remote -v
- 添加远程仓库:
git remote add [name] [repository-url]
- 删除远程仓库:
git remote rm [name]
- 修改远程仓库地址:
git remote set-url origin new-repository-url
- 拉取远程仓库:
git pull [remoteName] [localBranchName]
- 推送远程仓库:
git push [remoteName] [localBranchName]
例:git push -u orgin master
将当前分支推送到远端master分支 - 将本地 test 分支提交到远程 master 分支:
git push origin test:master
(把本地的某个分支 test 提交到远程仓库,并作为远程仓库的 master 分支) 提交本地 test 分支作为远程的 test 分支 :git push origin test:test
Git Config
Git 配置文件分为三级,系统级(–system)、用户级(–global)和目录级(–local),三者的使用优先级以离目录 (repository)最近为原则,如果三者的配置不一样,则生效优先级目录级>用户级>系统级,可以通过 git config --help
查看更多内容。
- 系统级配置存储在
/etc/gitconfig
文件中,可以使用git config --system user.name "jim"
,git config --sytem user.email "jim.jim@gmail.com"
来进行配置,该配置对系统上所有用户及他们所拥有的仓库都生效的配置值。 - 用户级存储在每个用户的
~/.gitconfig
中,可以使用git config --global user.name "jim"
,git config --global user.email "jim.jim@gmail.com"
来进行配置,该配置对当前用户上所有的仓库有效。 - 目录级存储在每个仓库下的
.git/config
中,可以使用git config --local user.name "jim"
,git config --local user.email "jim.jim@gmail.com"
来进行配置,只对当前仓库生效。
Basic Usage
- 添加文件到暂存区(staged):
git add filename
/git stage filename
- 将所有修改文件添加到暂存区(staged):
git add --all
/git add -A
- 提交修改到暂存区(staged):
git commit -m 'commit message'
/git commit -a -m 'commit message'
注意理解 -a 参数的意义 - 从Git仓库中删除文件:
git rm filename
- 从Git仓库中删除文件,但本地文件保留:
git rm --cached filename
- 重命名某个文件:
git mv filename newfilename
或者直接修改完毕文件名 ,进行git add -A && git commit -m 'commit message'
Git会自动识别是重命名了文件 - 获取远程最新代码到本地:
git pull (origin branchname)
可以指定分支名,也可以忽略。pull 命令自动 fetch 远程代码并且 merge,如果有冲突,会显示在状态栏,需要手动处理。更推荐使用:git fetch
之后git merge --no-ff origin branchname
拉取最新的代码到本地仓库,并手动 merge 。
Checkout
checkout命令用于从历史提交(或者暂存区域)中拷贝文件到工作目录,也可用于切换分支。
![](./_image/2016-07-14 21-26-37.jpg?r=49) ![](./_image/2016-07-14 21-15-47.jpg?r=49&f=2)
匿名分支:如果既没有指定文件名,也没有指定分支名,而是一个标签、远程分支、SHA-1值或者是像 master~3 类似的东西,就得到一个匿名分支,称作 detached HEAD(被分离的 HEAD 标识)。
![](./_image/2016-07-14 21-44-06.jpg?r=56)
当HEAD处于分离状态(不依附于任一分支)时,提交操作可以正常进行,但是不会更新任何已命名的分支。(你可以认为这是在更新一个匿名分支。)一旦此后你切换到别的分支,比如说 master,那么这个提交节点(可能)再也不会被引用到,然后就会被丢弃掉了。注意这个命令之后就不会有东西引用 2eecb。详细查看:visual-git-guide#detached
但是,如果你想保存这个状态,可以用命令 git checkout -b name
来创建一个新的分支。
![](./_image/2016-07-14 21-45-50.jpg?r=56)
Log
- 查看日志:
git log
- 查看日志,并查看每次的修改内容:
git log -p
- 查看日志,并查看每次文件的简单修改状态:
git log --stat
- 一行显示日志:
git log --pretty=oneline
/git log --pretty='format:"%h - %an, %ar : %s'
- 查看日志范围:
- 查看最近10条日志:
git log -10
- 查看2周前:
git log --until=2week
或者指定2周的明确日期,比如:git log --until=2015-08-12
- 查看最近2周内:
git log --since=2week
或者指定2周明确日志,比如:git log --since=2015-08-12
- 只查看某个用户的提交:
git log --committer=user.name
/git log --author=user.name
- 只查看提交msg中包含某个信息的历史,比如包含’测试’两个字的:
git log --grep '测试'
- 试试这个 :
git log --graph --pretty=format:'%Cred%h%Creset -%C(yellow)%d%Creset %s %Cgreen(%cr) %C(bold blue)<%an>%Creset' --abbrev-commit
感觉好用就加成 alias ,方便日后用,方法:git config --global alias.aliasname 'alias-content'
- 更多用法:Viewing the History – 《Pro Git2》
- 查看最近10条日志:
Undo things
- 上次提交 msg 错误/有未提交的文件应该同上一次一起提交,需要重新提交备注:
git commit --amend -m 'new msg'
- 一次
git add -A
后,需要将某个文件撤回到工作区,即:某个文件不应该在本次commit中:git reset HEAD filename
- 撤销某些文件的修改内容:
git checkout -- filename
注意:一旦执行,所有的改动都没有了,谨慎!谨慎!谨慎! - 将工作区内容回退到远端的某个版本:
git reset --hard <sha1-of-commit>
Reset
reset命令把当前分支指向另一个位置,并且有选择的变动工作目录和索引,也用来在从历史仓库中复制文件到索引,而不动工作目录。
![](./_image/2016-07-14 20-31-39.png?r=64&f=1)
将工作区内容回退到远端的某个版本:git reset --hard <sha1-of-commit>
git reset --hard HEAD^
reset index and working directory ,以来所有的变更全部丢弃,并将 HEAD 指向
git reset --soft HEAD^
nothing changed to index and working directory ,仅仅将 HEAD 指向,所有变更显示在 “changed to be committed”中 git reset --mixed HEAD^
default,reset index ,nothing to working directory 默认选项,工作区代码不改动,添加变更到index区
Revert
git revert
will create a new commit that’s the opposite (or inverse) of the given SHA. If the old commit is “matter”, the new commit is “anti-matter”—anything removed in the old commit will be added in the new commit and anything added in the old commit will be removed in the new commit.
This is Git’s safest, most basic “undo” scenario, because it doesn’t alter history—so you can now git push the new “inverse” commit to undo your mistaken commit.
git revert [--[no-]edit] [-n] [-m parent-number] [-s] [-S[<keyid>]] <commit>…
git revert --continue
git revert --quit
git revert --abort
Diff
- 查看工作区(working directory)和暂存区(staged)之间差异:
git diff
- 查看工作区(working directory)与当前仓库版本(repository)HEAD版本差异:
git diff HEAD
- 查看暂存区(staged)与当前仓库版本(repository)差异:
git diff --cached
/git diff --staged
Merge
![](./_image/2016-07-14 20-53-25.jpg?r=80)
- 解决冲突后/获取远程最新代码后合并代码:
git merge branchname
,将 branchname 分支上面的代码合并到当前分支 - 保留该存在版本合并log:
git merge --no-ff branchname
参数--no-ff
防止 fast-forward 的提交,详情参考:the difference,fast-forward:分支内容一致,指针直接移动,并未能看出分支信息
Rebase
Rebase 同 Merge 的结果是一样的,就是合并本地、远程的改动,但过程中还有区别。
git checkout mywork
git rebase origin
这些命令会把你的”mywork”分支里的每个提交(commit)取消掉,并且把它们临时 保存为补丁(patch)(这些补丁 放到”.git/rebase”目录中),然后把”mywork”分支更新 到最新的”origin”分支,最后把保存的这些补丁应用 到”mywork”分支上。
一张图分清 rebase 和 merge 的区别
![](./_image/2016-07-19 20-35-30.jpg?r=56)
在rebase的过程中,也许会出现冲突(conflict). 在这种情况,Git会停止rebase并会让你去解决冲突;在解决完冲突后,用 git-add
命令去更新这些内容的索引(index), 然后,你无需执行 git-commit,只要执行: git rebase --continue
这样git会继续应用(apply)余下的补丁。在任何时候,你可以用 –abort 参数来终止rebase的行动,并且”mywork” 分支会回到rebase开始前的状态。 git rebase --abort
Cherry Pick
cherry-pick命令”复制”一个提交节点并在当前分支做一次完全一样的新提交。
![](./_image/2016-07-14 20-57-04.jpg?r=65)
Branch
Aone2 Git 分支开发部署模型详细解读 http://docs.alibaba-inc.com:8090/pages/viewpage.action?pageId=194872297
- master:master永远是线上代码,最稳定的分支,存放的是随时可供在生产环境中部署的代码,当开发活动告一段落,产生了一份新的可供部署的代码时,发布成功之后,代码才会由 aone2 提交到 master,master 分支上的代码会被更新。应用上 aone2 后禁掉所有人的 master的写权限
- develop:保存当前最新开发成果的分支。通常这个分支上的代码也是可进行每日夜间发布的代码,只对开发负责人开放develop权限。
- feature: 功能特性分支,每个功能特性一个 feature/ 分支,开发完成自测通过后合并入 develop 分支。可以从 master 或者develop 中拉出来。
- hotfix: 紧急bug分支修复分支。修复上线后,可以直接合并入master。
![](./_image/2016-07-19 19-58-15.jpg?r=51)
Git-Develop 分支模式是基于 Git 代码库设计的一种需要严格控制发布质量和发布节奏的开发模式。develop 作为固定的持续集成和发布分支,并且分支上的代码必须经过 CodeReview 后才可以提交到 Develop 分支。它的基本流程如下:
- 每一个需求/变更都单独从Master上创建一条Branch分支;
- 用户在这个Branch分支上进行Codeing活动;
- 代码达到发布准入条件后aone上提交Codereview,Codereview通过后代码自动合并到Develop分支;
- 待所有计划发布的变更分支代码都合并到Develop后,系统再rebase master代码到Develop分支,然后自行构建,打包,部署等动作。
- 应用发布成功后Aone会基于Develop分支的发布版本打一个“当前线上版本Tag”基线;
- 应用发布成功后Aone会自动把Develop分支的发布版本合并回master;
常用命令:
- 查看分支:
git branch
、git branch -v
、git branch -vv
(查看当前分支 tracking 哪个远端分支)、git branch --merged
、git branch --no-merged
- 创建分支:
git branch branchname
- 例: 基于 master 分支新建 dev 分支 :
git branch dev
- 例: 基于 master 分支新建 dev 分支 :
- 基于之前的某个 Commit 新开分支:
git branch branchname <sha1-of-commit>
- 例: 基于上线的的提交 a207a38d634cc10441636bc4359cd8a18c502dea 创建 hotfix 分支 :
git branch hotfix a207a38
- 例: 基于 remoteBranch、localBranch、commitId、tag 创建分支均可以
git checkout -b newbranch localBranch/remoteBranch/commitId/tag
- 例: 基于上线的的提交 a207a38d634cc10441636bc4359cd8a18c502dea 创建 hotfix 分支 :
- 切换分支:
git checkout branchname
- 例: 由分支 master 切换到 dev 分支:
git checkout dev
- 例: 由分支 master 切换到 dev 分支:
- 创建新分支并切换到下面:
git checkout -b branchname
或者git branch branchname && git checkout branchname
- 例:基于 master 分支新建 dev 分支,并切换到 dev 分支上:
git checkout -b dev
或git branch dev && git checkout dev
- 例:基于 master 分支新建 dev 分支,并切换到 dev 分支上:
- 查看分支代码不同:
git diff branchname
比较 branchname 分支与当前分支的差异点 - 合并分支:
git merge branchname
将 branchname 分支代码合并到当前分支 - 删除分支:
git branch -d branchname
强制删除未合并过的分支:git branch -D branchname
- 查看远程分支:
git branch -r
或git branch -r -v
- 获取远程分支到本地:
git checkout -b local-branchname origin/remote-branchname
- 推送本地分支到远程:
git push origin remote-branchname
或git push origin local-branchname:remote-branchname
- 将本地 dev 代码推送到远程 dev 分支:
git push (-u) origin dev
或git push origin dev:dev
- (技巧)将本地 dev 分支代码推送到远程 master 分支:
git push origin dev:master
- 将本地 dev 代码推送到远程 dev 分支:
- 删除远程分支:
git push origin :remote-branchname
或git push origin --delete remote-branchname
- TrackingBranch,可以通过
git branch -vv
来查看当前 track 的分支情况。新建立分支时会自动 track 相应远程分支,git checkout -b sf origin/serverfix
(Branch sf set up to track remote branch serverfix from origin. Switched to a new branch ‘sf’). 也可以手动 track:git branch -u origin/serverfix
(Branch serverfix set up to track remote branch serverfix from origin). 等同于命令git checkout --track origin/serverfix
“Checking out a local branch from a remote branch automatically creates what is called a “tracking branch” (or sometimes an “upstream branch”). Tracking branches are local branches that have a direct relationship to a remote branch. If you’re on a tracking branch and type git pull, Git automatically knows which server to fetch from and branch to merge into.
When you clone a repository, it generally automatically creates a master branch that tracks origin/master. However, you can set up other tracking branches if you wish – ones that track branches on other remotes, or don’t track the master branch. The simple case is the example you just saw, running git checkout -b [branch] [remotename]/[branch]. This is a common enough operation that git provides the –track shorthand:”
Tag
- 查看 tag:
git tag
- 查找指定 tag,比如查找 V1.0.* :
git tag -l 'V1.0.*'
会列出匹配到的,比如 V1.0.1,V1.0.1.1,V1.0.2 等 - 创建轻量级 tag(lightweight tags):
git tag tag-name
,例如:git tag v1.0
- 创建 tag(annotated tags):
git tag -a tag-name -v 'msg'
,例如:git tag -a v1.0.0 -m '1.0.0版本上线完毕打tag'
- annotated tags VS lightweight tags 可以通过命令真实查看下:
git show v1.0
/git show v1.0.0
- “A lightweight tag is very much like a branch that doesn’t change – it’s just a pointer to a specific commit.
- annotated tags VS lightweight tags 可以通过命令真实查看下:
Annotated tags, however, are stored as full objects in the Git database. They’re checksummed; contain the tagger name, e-mail, and date; have a tagging message; and can be signed and verified with GNU Privacy Guard (GPG). ”
- 查看指定 tag 信息:
git show tag-name
- 基于历史某次提交(commit)创建 tag :
git tag -a tagname <sha1-of-commit>
- 例:基于上线时的提交 a207a38d634cc10441636bc4359cd8a18c502dea 创建tag:
git tag -a v1.0.0 a207a38
- 例:基于上线时的提交 a207a38d634cc10441636bc4359cd8a18c502dea 创建tag:
- 删除 tag :
git tag -d tagname
- 拉取远程 tag 到本地:
git pull remotename --tags
例如:git pull origin --tags
- 推送 tag 到远程服务器:
git push remotename tagname
例如:git push origin v1.0.0
- 将本地所有 tag 推送到远程:
git push remotename --tags
例如:git push origin --tags
- 删除远程 tag :
git push origin :tagname
或者git push origin --delete tagname
Submodule
添加子模块:$ git submodule add [url] [path]
如:$ git submodule add git://github.com/soberh/ui-libs.git src/main/webapp/ui-libs
初始化子模块:$ git submodule init —-只在首次检出仓库时运行一次就行
更新子模块:$ git submodule update —-每次更新或切换分支后都需要运行一下
删除子模块:(分4步走哦)
1) $ git rm –cached [path]
2) 编辑“.gitmodules”文件,将子模块的相关配置节点删除掉
3) 编辑“ .git/config”文件,将子模块的相关配置节点删除掉
4) 手动删除子模块残留的目录
Stash
![](./_image/2016-07-16 21-30-01.jpg?r=64)