Discuz! Board

 找回密码
 立即注册
搜索
热搜: 活动 交友 discuz
查看: 2918|回复: 3
打印 上一主题 下一主题

Git的submodule功能详解

[复制链接]

257

主题

354

帖子

1677

积分

金牌会员

Rank: 6Rank: 6

积分
1677
跳转到指定楼层
楼主
发表于 2016-12-13 20:03:18 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
http://www.360doc.com/content/15/0704/14/7811581_482561114.shtml

git的submodule功能详解
1. 前言
项目的版本库在某些情况下需要引用其他版本库中的文件,例如有一套公用的代码库,可以被多个项目调用,这个公用代码库能直接放在某个项目的代码中,而是要独立为一个代码库,那么其他要调用公用的代码库该如何处理?分别把公用的代码库拷贝到各自的项目中会造成冗余,丢弃了公共代码库的维护历史,这些显示不是好的办法,现在要了解的git子模组(git submodule)就解决了这个问题。
Git 子模块功能允许你将一个Git仓库当作另外一个Git仓库的子目录。这允许你克隆另外一个仓库到你的项目中并且保持你的提交相对独立。
2. 准备工作
首先需要两个版本库例如:
1) 一个公共的版本库(例如:libA.git)
git@gitlab.szreach.com:fengyang/liba.git
2) 一个引用公共版本库的主版本库(例如:super.git)
git@gitlab.szreach.com:fengyang/super.git
3. TortoiseGit上git submodule功能实现1. 添加子模组的步骤
第一步:使用submodule add...添加子模组
在需要作添加子模组的Git版本库中右击,选择“TortoiseGit->Submodule Add...”,在“Repository:”里面输入需要添加子模组的版本库路径,在“Path:”
中输入添加子模组存放的目录路径。如下图所示:

此时查看需要添加子模组的Git工作区的目录结构。在根目录下多了一个.gitmodules文件,并且公共代码库被克隆到lib/lib_a目录下。


第二步:查看.gitmodules的内容
.gitmodules的内容记录着含子模组存放的目录路径及子模组版本库的路径

注:此时工作区尚未提交,完成提交后,子模组才算正式完成在需要作添加子模组的Git版本库中创立。
注:此主版本库(super.git)就变成了一个包含子模组的版本库。
2. 克隆带子模组的版本库的步骤
克隆带子模组的git库,并不能自动将子模组的版本库克隆出来,对于只关心项目本身的数据,而不关心项目引用的外部项目数据的用户,这个功能非常好,数据没有冗余而且克隆的速度也很快。

第一步:克隆super主版本库
在工作区中克隆主主版本库后,会发现子模组的版本库并没有克隆,只有将存放子模组版本库的目录克隆下来了。


----:存放子模组版本库的目录路径

第二步:克隆子模组
如果需要克隆出子模组形式引用的外部库,首先需要执行Submodule Update操作。
在需要作克隆子模组的Git工作区中右击,选择“TortoiseGit->Submodule Update...”,点击“OK”即可。


执行Submodule Update...操作后就会把子模组的版本库克隆下来。




3. 对主Git库工作区作commit、checkout、revert、pull、push等更改Git工作区内容的操作时,查看Submodule目录下内容的效果
1) 对主Git库工作区作commit操作
当在主Git库工作区中变更到达一个适当状态时,我们需要将这些变更提交到Git库,作版本备份和跟踪。

方法:
在工作区内右击,选择“ Commit-> "当前分支名称" ”,在Commit窗口中,选择需要提交的变更path,填写提交说明,点击“Ok”。



结论:在主git工作区作commit操作,只是将主Git工作区的变更历史提交到主Git本地库中,对Submodule没有任何影响。

2) 对主Git库工作区作checkout操作
当在主Git库工作区中想要切换到其他某个分支、某个tag、某个commit历史记录。
我们这里举个特殊的例子:
切换到某个commit历史记录(如在主版本库中提交子版本库的历史记录)
方法:
在Git工作区内的某目录上右击,选择“Swicht/Checkout...”在Swicht/Checkout...Checkout窗口中,在Switch to Commmit中列出需要切换的commit历史记录,在此列表中选择需要切换到的commit历史记录,以后点击“Ok”按钮。






注:由于切换到的是在主版本库中提交子版本库的那条历史记录,因此子模组中状态也会改换,也需要在子模组中执行切换。

执行Submodule Update操作即可。




3) 对主Git库工作区作revert操作
当在主Git库工作区中这次变更有错误时,就可以使用revert操作来撤销这次次操作。

方法:
在工作区内右击,选择“Revert”,在Revert窗口中,选择需要撤销的变更path,点击“Ok”。






结论:在主git工作区作revert操作,只是将主Git工作区某次的变更操作给撤销,此次操作之前的commit都会被保留,且对Submodule没有任何影响。


4) 对主Git库工作区作push操作
当在主Git库工作区中将主Git本地库的数据推送到主Git远程库中。

方法:在工作区的目录上(不要在文件上)右击,选择“Git Sync...”
"Local Branch","Remote Branch",“Remote URL”这3栏必须正确才能“推”数据。


结论:在主git工作区作push操作,只是将主Git本地库的变更历史推送到到主Git远程库中,对Submodule没有任何影响。


5) 对主Git库工作区作pull操作
当在主Git库工作区中将主Git远程库的拉取到主Git本地库中。

方法:在工作区的目录上(不要在文件上)右击,选择“Git Sync...”
"Local Branch","Remote Branch",“Remote URL”这3栏必须正确才能“拉”数据。


结论:在主git工作区作pull操作,只是将主Git远程库的变更历史拉取到到主Git本地库中,对Submodule没有任何影响。
4. 对submodule下的内容进行修改后commit、push,对submodule目录作pull
在执行Submodule Update...操作更新出子模组后,都以某个具体的提交版本进行检出,进入子模组目录,会发现其处于非跟踪状态。

显然这种情况下,如果修改lib/lib_a下的文件,提交就会丢失。下面介绍如何在检查的子模组中修改,以及如何更新子模组。
在子模组中切换到master分支(或者其他想要修改的分支)后在进行修改。
l 切换到master分支,然后在工作区做一些改动。


l 执行commit后,并且推送到子模组库中后。
l 回到主版本库中。
在主版本库中查看状态,在主版本库中可以看到子模组已修改,包含了更新的提交。
l 需要将修改提交在主版本库中进行推送。

对submodule目录作pull
当子模组版本库中有新的提交历史记录,就需要做pull操作来更新。


注:对子模组库中做了pull操作,同时需要对调用子模组库的主版本库中也要做Pull操作。
如下图所示:




注意:进入子模组目录,会发现其处于非跟踪状态。显然这种情况下,如果修改lib/lib_a下的文件,提交就会丢失。提示如下:


5. 移除Submodule
第一步:用Delete(keep local) 将目录或文件命令从git的索引库中移除子模块目录
方法:
在工作区内目录或文件上右击,选择“Delete(keep local)”,在Delete窗口中,点击“Remove”。

Delete(keep local)把目录lib/从git的索引库中移除,但是对目录lib/本身并不进行任何操作。
因此需要彻底删除。

第二步:删除物理文件
第三步:删除.gitmodule文件
第四步:删除.git/config的submodule配置源文件
删除如下图所示红框的部门


第五步:删除后,提交更改

http://www.kafeitu.me/git/2012/03/27/git-submodule.html
回复

使用道具 举报

1228

主题

1997

帖子

7586

积分

认证用户组

Rank: 5Rank: 5

积分
7586
沙发
发表于 2018-2-12 19:02:58 | 只看该作者
本帖最后由 Qter 于 2020-2-4 12:00 编辑

git submodule 增加 删除及更新URL
都是基于git根路径的


增加一个submodule
git submodule add https://github.com/cisco/openh264 open_source_code/openh264再如:
git submodule add https://github.com/hechengjin/mimetic.git src/firemail_solution/Ext/mimetic
切记不能直接修改.gitmodules


修改后,先再子模块提交,然后主模块也能检查出子模块的修改,主模块也要提交一下(但提交的内容并不是真实修改的东西,应该是对应的分支进展情况信息)

通过界面如下:
git根目录下 右键 -> TortoiseGit -> 添加子模块




删除一个submodule

1.删除 .gitsubmodule中对应submodule的条目

2.删除 .git/config 中对应submodule的条目

3.执行 git rm --cached {submodule_path}。注意,路径不要加后面的“/”。例如:你的submodule保存在 supports/libs/websocket/ 目录。执行命令为: git rm --cached supports/libs/websocket

更新submodule的URL

1.更新 .gitsubmodule中对应submodule的条目URL

2.更新 .git/config 中对应submodule的条目的URL

3.执行 git submodule sync

通过界面如下:
git根目录下 右键 -> TortoiseGit -> 更新子模块




回复 支持 反对

使用道具 举报

1228

主题

1997

帖子

7586

积分

认证用户组

Rank: 5Rank: 5

积分
7586
板凳
发表于 2020-1-12 13:00:01 | 只看该作者
背景

项目中经常使用别人维护的模块,在git中使用子模块的功能能够大大提高开发效率。

使用子模块后,不必负责子模块的维护,只需要在必要的时候同步更新子模块即可。

本文主要讲解子模块相关的基础命令,详细使用请参考man page。

子模块的添加

添加子模块非常简单,命令如下:

git submodule add <url> <path>

其中,url为子模块的路径,path为该子模块存储的目录路径。

执行成功后,git status会看到项目中修改了.gitmodules,并增加了一个新文件(为刚刚添加的路径)

git diff --cached查看修改内容可以看到增加了子模块,并且新文件下为子模块的提交hash摘要

git commit提交即完成子模块的添加

子模块的使用

克隆项目后,默认子模块目录下无任何内容。需要在项目根目录执行如下命令完成子模块的下载:

git submodule init
git submodule update
1
2
或:

git submodule update --init --recursive
1
执行后,子模块目录下就有了源码,再执行相应的makefile即可。

子模块的更新

子模块的维护者提交了更新后,使用子模块的项目必须手动更新才能包含最新的提交。

在项目中,进入到子模块目录下,执行 git pull更新,查看git log查看相应提交。

完成后返回到项目目录,可以看到子模块有待提交的更新,使用git add,提交即可。

删除子模块

有时子模块的项目维护地址发生了变化,或者需要替换子模块,就需要删除原有的子模块。

删除子模块较复杂,步骤如下:

rm -rf 子模块目录 删除子模块目录及源码
vi .gitmodules 删除项目目录下.gitmodules文件中子模块相关条目
vi .git/config 删除配置项中子模块相关条目
rm .git/module/* 删除模块下的子模块目录,每个子模块对应一个目录,注意只删除对应的子模块目录即可
执行完成后,再执行添加子模块命令即可,如果仍然报错,执行如下:

git rm --cached 子模块名称

完成删除后,提交到仓库即可。
————————————————
版权声明:本文为CSDN博主「guotianqing」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/guotianqing/article/details/82391665

回复 支持 反对

使用道具 举报

1228

主题

1997

帖子

7586

积分

认证用户组

Rank: 5Rank: 5

积分
7586
地板
发表于 2020-1-27 10:03:36 | 只看该作者
https://www.jianshu.com/p/f8a55b972972
更新如果在本地修改子仓库,在主仓库 git status会显示子仓库有修改。
  1. $git status
  2. On branch master
  3. Your branch is up-to-date with 'origin/master'.
  4. Changes not staged for commit:
  5.   (use "git add <file>..." to update what will be committed)
  6.   (use "git checkout -- <file>..." to discard changes in working directory)
  7.   (commit or discard the untracked or modified content in submodules)

  8.     modified:   lib (modified content)

  9. no changes added to commit (use "git add" and/or "git commit -a")
复制代码
需要现在子仓库提交,然后再到主仓库提交代码。

回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|Archiver|手机版|小黑屋|firemail ( 粤ICP备15085507号-1 )

GMT+8, 2024-5-9 06:14 , Processed in 0.063329 second(s), 18 queries .

Powered by Discuz! X3

© 2001-2013 Comsenz Inc.

快速回复 返回顶部 返回列表