## 7.11 子模板
子模块允许将一个 Git 仓库作为另一个 Git 仓库的子目录。 它能将另一个仓库克隆到自己的项目中,同时还保持提交的独立。
### 7.11.1 **开始使用子模块**
将一个已存在的 Git 仓库添加为正在工作的仓库的子模块:
```
$ git submodule add [url]
```
此时版本库中会有一个新的`.gitmodules`文件。 该配置文件保存了项目 URL 与已经拉取的本地目录之间的映射。如果有多个子模块,该文件中就会有多条记录。 要重点注意的是,该文件也像`.gitignore`文件一样受到(通过)版本控制。而在`git status`输出中列出的另一个是项目文件夹记录。
### 7.11.2 **克隆含有子模块的项目**
如果克隆一个含有子模板的项目,默认会包含该子模块目录,但其中还没有任何文件。此时需要运行两个命令:
```
$ git submodule init
```
用来初始化本地配置文件,和
```
$ git submodule update
```
从该项目中抓取所有数据并检出父项目中列出的合适的提交。
或者在克隆时,直接使用 `--recursive` ,Git 会自动初始化并更新仓库中的每一个子模块。
### 7.11.3 在包含子模块的项目上工作
**拉取上游修改**
如果想要在子模块中查看新工作,可以进入到目录中运行`git fetch`与`git merge`,合并上游分支来更新本地代码。
在版本库的根目录下运行以下命令查看子模板具体的修改。
```
$ git diff --submodule
```
如果不想在子目录中手动抓取与合并,那么还有种更容易的方式。 运行命令让Git 自动进入子模块然后抓取并更新。
```
$ git submodule update --remote
```
此命令默认更新并检出子模块仓库的`master`分支,可以通过修改 `.gitmodules` 来更改默认的分支:
```
$ git config -f .gitmodules submodule.[submodule].branch [branch]
```
**在子模块上工作**
如果本地的子模板中有修改没有提交,运行:
```
$ git submodule update --remote
```
不会覆盖,它只会将远程仓库中的修改抓取到本地,类似于`git fetch`。如果想要抓取的同时进行合并,可以使用 `--merge` 选项:
```
$ git submodule update --remote --merge
```
**发布子模块改动**
`git push`命令接受可以设置为 `check` 或 `on-demand` 的`--recurse-submodules`参数。
* 如果任何提交的子模块改动没有推送那么 `check` 选项会直接使`push`操作失败
* 而 `on-demand` 选项会尝试一并推送子模板的改动,但是如果子模板的推送失败,整个 `push` 操作失败
### 7.11.4 子模块技巧
**子模块遍历**
如果项目中包含了大量子模块,可以使用`foreach`子模块命令,它能在每一个子模块中运行任意命令:
```
$ git submodule foreach 'git stash'
```
**有用的别名**
可以通过设置别名来替代一些较长的子模板命令。例如:
```
$ git config alias.sdiff '!'"git diff && git submodule foreach 'git diff'"
$ git config alias.spush 'push --recurse-submodules=on-demand'
$ git config alias.supdate 'submodule update --remote --merge'
```
### 7.11.5 子模块的问题
在有子模块的项目中切换分支可能会造成麻烦。 如果在创建一个新分支后在其中添加一个子模块,之后切换到没有该子模块的分支上时,仍然会有一个还未跟踪的子模块目录。
- 介绍
- 第一章 起步
- 1.1 关于版本控制
- 1.2 Git 简史
- 1.3 Git 基础
- 1.4 命令行
- 1.5 安装 Git
- 1.6 初次运行 Git 前的配置
- 1.7 获得帮助
- 第二章 基础
- 2.1 获取仓库
- 2.2 记录每次更新到仓库
- 2.3 查看提交历史
- 2.4 撤销操作
- 2.5 远程仓库的使用
- 2.6 打标签
- 2.7 Git 别名
- 第三章 分支
- 3.1 分支简介
- 3.2 分支的新建与合并
- 3.3 分支管理
- 3.4 分支开发工作流
- 3.5 远程分支
- 3.6 变基
- 第四章 服务器上的 Git
- 4.1 协议
- 4.2 在服务器上搭建 Git
- 4.3 生成 SSH 公钥
- 4.4 配置服务器
- 4.5 Git 守护进程
- 4.6 Smart HTTP
- 4.7 GitWeb
- 4.8 GitLab
- 4.9 第三方托管的选择
- 第五章 分布式 Git
- 5.1 分布式工作流程
- 5.2 向一个项目贡献
- 5.3 维护项目
- 第六章 GitHub
- 6.1 账户的创建和配置
- 6.2 对项目做出贡献
- 6.3 维护项目
- 6.4 管理组织
- 6.5 脚本 GitHub
- 第七章 Git 工具
- 7.1 选择修订版本
- 7.2 交互式暂存
- 7.3 储藏与清理
- 7.4 签署工作
- 7.5 搜索
- 7.6 重写历史
- 7.7 重置揭密
- 7.8 高级合并
- 7.9 Rerere
- 7.10 使用Git调试
- 7.11 子模板
- 7.12 打包
- 7.13 替换
- 7.14 凭证存储
- 第八章 自定义 Git
- 8.1 配置 Git
- 8.2 Git 属性
- 8.3 Git 钩子
- 8.4 使用强制策略的一个例子
- 第九章 Git 与其他系统
- 9.1 作为客户端的 Git
- 9.2 迁移到 Git
- 第十章 Git 内部原理
- 10.1 底层命令和高层命令
- 10.2 Git 对象
- 10.3 Git 引用
- 10.4 包文件
- 10.5 引用规格
- 10.6 传输协议
- 10.7 维护与数据恢复
- 10.8 环境变量
