Git基础应用

2018-12-19  本文已影响0人  Net夜风

linus研发,分布式版本控制工具;


git是通过记录快照的方式来实现文件内容保留的,它不是记录差异;但是,git在仓库中保留的每一个文件进行追踪时,采用独特的方式,没一个文件的名称在仓库中,这个仓库只是文件追踪系统,它只追踪内容,意味着仅保留文件内容的变化,至于文件叫什么名,文娱哪个目录下,仓库在文件内部是不加任何保留的;
git仓库是一个对象存储系统,所谓对象存储,每一个文件的所有数据,无论是数据还是元数据都是自包含的,往存储空间里存放的每一个文件,这个文件的所有属性都是自包含;
所以每一个文件都是由着自己对自我结构描述属性等信息的内容,所有称作是完成而独立的对象;
引用每一个对象时,只需要引用对象的一个符号链接或一个名称就行;git版本仓库里的内容就是对象存储系统;
例如,有一个文件需要保存到仓库中,将来必要时能在仓库中检索出来原来的某个版本;这个文件在工作区中看是文件,而且是保存在文件系统上的文件,但是一旦把它纳入到版本库中去,就以为这它要被版本库生成一个文件对象而保存在版本库中;

git对每一个文件对象的名称追踪时,就需要对文件另起一个名字,这个名字很独特是文件内容的哈希值;注意:文件名是文件内容的哈希值;可以使用MD5或sha取得文件的哈希码,只要文件内容变化,它的哈希一定会改变;本来git的仓库就是通过保存文件快照进行的,而它的快照保存机制就是只要文件内容变了就把这个文件复制一份新版本出来,重新创建个对象;
如果在工作区中有两个文件不在同一个目录下,但内容一摸一样,在对象库中起始只保存一个版本

在开发应用程序时,通常要创建个目录,里面存放程序的代码:
[root@node1 ~]# mkdir testproj
[root@node1 ~]# cd testproj/

为了能够使这个代码目录下,将来把所有版本内容的变化可以基于版本进行追偿,可以随时回到过去的某个版本,通常就需要在这个目录下创建一个隐藏目录,里面记录这个工作目录当中的每一个文件在需要保存状态时的状态变化;
所以,对应testproj就叫做工作目录,在工作目录下就会有版本库自动生成版本仓库;


git的安装基本使用

git在本地工作的模式

初始化以后,创建了好多文件:
branches:分支
config:当前版本仓库的配置;对于git配置有三个等级:

HEAD:指向最新的树对象
objects:对象库目录

演示:
    此时没有index目录,因为还没有添加文件,创建文件后自动生成
    
    [root@node1 testproj]# cp /etc/passwd ./
    [root@node1 testproj]# vim README
        Project: test project

    [root@node1 testproj]# ls
    passwd  README
    [root@node1 testproj]# git add .    #暂存工作目录下所有文件,也可以使用git add -a
    [root@node1 testproj]# tree .git
    .git
    ├── branches
    ├── config
    ├── description
    ├── HEAD
    ├── hooks
    │   ├── applypatch-msg.sample
    │   ├── commit-msg.sample
    │   ├── post-update.sample
    │   ├── pre-applypatch.sample
    │   ├── pre-commit.sample
    │   ├── prepare-commit-msg.sample
    │   ├── pre-push.sample
    │   ├── pre-rebase.sample
    │   └── update.sample
    ├── index
    ├── info
    │   └── exclude
    ├── objects
    │   ├── 63
    │   │   └── 452156647080f0f17e0d0cb14d7c1f1f90e22e
    │   ├── 78
    │   │   └── 270f5908030219758738a85d499b8bfe6baa2b
    │   ├── info
    │   └── pack
    └── refs
        ├── heads
        └── tags
    
    11 directories, 16 files
    此时自动创建了index目录,用来存放暂存区的数据
    [root@node1 testproj]# file .git/index 
    .git/index: Git index, version 2, 2 entries
    [root@node1 testproj]# vim README 
    [root@node1 testproj]# git add README
    Project: test project
    version1.1.0

    [root@node1 testproj]# tree .git
    .git
    ├── branches
    ├── config
    ├── description
    ├── HEAD
    ├── hooks
    │   ├── applypatch-msg.sample
    │   ├── commit-msg.sample
    │   ├── post-update.sample
    │   ├── pre-applypatch.sample
    │   ├── pre-commit.sample
    │   ├── prepare-commit-msg.sample
    │   ├── pre-push.sample
    │   ├── pre-rebase.sample
    │   └── update.sample
    ├── index
    ├── info
    │   └── exclude
    ├── objects
    │   ├── 63
    │   │   └── 452156647080f0f17e0d0cb14d7c1f1f90e22e
    │   ├── 78
    │   │   └── 270f5908030219758738a85d499b8bfe6baa2b
    │   ├── 7a
    │   │   └── 7f3b22a21932aefe2b204fe528475ca358d422
    │   ├── info
    │   └── pack
    └── refs
        ├── heads
        └── tags
    
    12 directories, 17 files
    此时修改了README文件内容后重新暂存后objects目录下多了一个目录
    [root@node1 testproj]# mkdir docs
    [root@node1 testproj]# touch docs/changelog
    [root@node1 testproj]# git add docs
    [root@node1 testproj]# tree .git  
    .git
    ├── branches
    ├── config
    ├── description
    ├── HEAD
    ├── hooks
    │   ├── applypatch-msg.sample
    │   ├── commit-msg.sample
    │   ├── post-update.sample
    │   ├── pre-applypatch.sample
    │   ├── pre-commit.sample
    │   ├── prepare-commit-msg.sample
    │   ├── pre-push.sample
    │   ├── pre-rebase.sample
    │   └── update.sample
    ├── index
    ├── info
    │   └── exclude
    ├── objects
    │   ├── 63
    │   │   └── 452156647080f0f17e0d0cb14d7c1f1f90e22e
    │   ├── 78
    │   │   └── 270f5908030219758738a85d499b8bfe6baa2b
    │   ├── 7a
    │   │   └── 7f3b22a21932aefe2b204fe528475ca358d422
    │   ├── info
    │   └── pack
    └── refs
        ├── heads
        └── tags
    
    12 directories, 17 files
    此时暂存区未发生改变,因为git add 不追踪空目录
    [root@node1 testproj]# vim docs/changelog
    auth: magedu
    [root@node1 testproj]# git add docs/
    [root@node1 testproj]# tree .git
    .git
    ├── branches
    ├── config
    ├── description
    ├── HEAD
    ├── hooks
    │   ├── applypatch-msg.sample
    │   ├── commit-msg.sample
    │   ├── post-update.sample
    │   ├── pre-applypatch.sample
    │   ├── pre-commit.sample
    │   ├── prepare-commit-msg.sample
    │   ├── pre-push.sample
    │   ├── pre-rebase.sample
    │   └── update.sample
    ├── index
    ├── info
    │   └── exclude
    ├── objects
    │   ├── 63
    │   │   └── 452156647080f0f17e0d0cb14d7c1f1f90e22e
    │   ├── 78
    │   │   └── 270f5908030219758738a85d499b8bfe6baa2b
    │   ├── 7a
    │   │   └── 7f3b22a21932aefe2b204fe528475ca358d422
    │   ├── 7e
    │   │   └── 184b2a7d9bd622e74df67fbc6a9f75345090db
    │   ├── info
    │   └── pack
    └── refs
        ├── heads
        └── tags
    
    13 directories, 18 files
    此时在objects目录下多了一个目录,有内容的东西才能作为对象存储
    [root@node1 testproj]# git status
    # On branch master
    #
    # Initial commit
    #
    # Changes to be committed:
    #   (use "git rm --cached <file>..." to unstage)
    #
    #   new file:   README
    #   new file:   docs/changelog
    #   new file:   passwd
    #
    [root@node1 testproj]# git rm --cached passwd
    rm 'passwd'
    [root@node1 testproj]# ls
    docs  passwd  README
    [root@node1 testproj]# git status
    # On branch master
    #
    # Initial commit
    #
    # Changes to be committed:
    #   (use "git rm --cached <file>..." to unstage)
    #
    #   new file:   README
    #   new file:   docs/changelog
    #
    --cached是指删除缓存内的文件,不删除工作目录下的文件
git配置文件分为三种:

git 提交时必须要指明提交的用户名和email信息,因此,在提交前需要先手动配置

[root@node1 testproj]# git config -l   #列出当前版本库的配置,此命令要在版本库下执行或直接查看.git/config
core.repositoryformatversion=0
core.filemode=true
core.bare=false
core.logallrefupdates=true
[root@node1 testproj]# cat .git/config
[core]
    repositoryformatversion = 0
    filemode = true
    bare = false
    logallrefupdates = true

user.email:指明用户的邮箱;
user.name:指明用户的用户名
[root@node1 testproj]# git config --global user.name "MageEdu"
[root@node1 testproj]# git config --global user.email "magedu@magedu.com"
[root@node1 testproj]# git config -l
user.name=MageEdu
user.email=magedu@magedu.com
core.repositoryformatversion=0
core.filemode=true
core.bare=false
core.logallrefupdates=true

因为使用的是全局级别,在用户家目录下会生成.gitconfig文件;
[root@node1 testproj]# ls -a /root/.gitconfig 
/root/.gitconfig
[root@node1 testproj]# cat /root/.gitconfig
[user]
    name = MageEdu
    email = magedu@magedu.com
如果没指明则是仓库级别的:
[root@node1 testproj]# cat .git/config
[core]
    repositoryformatversion = 0
    filemode = true
    bare = false
    logallrefupdates = true
git提交

每一次提交要指明变化说明,会自动打一个文件名;
如果不想使用文本编辑的交互式进行说明变化信息,也可以是用-m <msg>

[root@node1 testproj]# git commit -m "v0.0.1"
[master (root-commit) 7adf5fc] v0.0.1
 3 files changed, 28 insertions(+)
 create mode 100644 README
 create mode 100644 docs/changelog
 create mode 100644 passwd
 [root@node1 testproj]# cd .git/objects/
 [root@node1 objects]# ls  其中的两数字的目录为分层管理众多的文件对象;
 47  5d  63  78  7a  7e  89  be  fc  info  pack
 [root@node1 objects]# ll 
 total 0
 drwxr-xr-x. 2 root root 52 Dec 19 23:10 47
 drwxr-xr-x. 2 root root 52 Dec 19 23:11 5d
 drwxr-xr-x. 2 root root 52 Dec 19 22:31 63
 drwxr-xr-x. 2 root root 52 Dec 19 22:31 78
 drwxr-xr-x. 2 root root 98 Dec 19 23:06 7a
 drwxr-xr-x. 2 root root 52 Dec 19 22:43 7e
 drwxr-xr-x. 2 root root 52 Dec 19 23:10 89
 drwxr-xr-x. 2 root root 52 Dec 19 23:06 be
 drwxr-xr-x. 2 root root 52 Dec 19 23:06 fc
 drwxr-xr-x. 2 root root  6 Dec 19 20:35 info
 drwxr-xr-x. 2 root root  6 Dec 19 20:35 pack
 [root@node1 objects]# ll 89   #文件名就是文件内容的哈希码;
 total 4
 -r--r--r--. 1 root root 111 Dec 19 23:10  6049330e920c53a53017258b8334e75b41c779
git-clone

这就叫远程仓库,把远程仓库克隆到本地;这也是生成仓库的第二种办法;
所以,生成仓库有两种办法:第一,创建新目录,里面创建文件,添加、提交就生成仓库了;第二,把别人的仓库克隆到本地;

[root@node1 testproj]# vim INSTALL
[root@node1 testproj]# ls
docs  INSTALL  passwd  README
[root@node1 testproj]# git add INSTALL 
[root@node1 testproj]# git status
# On branch master
# Changes to be committed:
#   (use "git reset HEAD <file>..." to unstage)
#
#   new file:   INSTALL
#
[root@node1 testproj]# git rm --cached INSTALL
rm 'INSTALL'
[root@node1 testproj]# git status
# On branch master
# Untracked files:
#   (use "git add <file>..." to include in what will be committed)
#
#   INSTALL
nothing added to commit but untracked files present (use "git add" to track)
[root@node1 testproj]# git ls-files
README
docs/changelog
passwd
[root@node1 testproj]# git add INSTALL 
[root@node1 testproj]# git ls-files   #显示目录树种的内容
INSTALL
README
docs/changelog
passwd
    
[root@node1 testproj]# tree .git
.git
├── branches
├── COMMIT_EDITMSG
├── config
├── description
├── HEAD
├── hooks
│   ├── applypatch-msg.sample
│   ├── commit-msg.sample
│   ├── post-update.sample
│   ├── pre-applypatch.sample
│   ├── pre-commit.sample
│   ├── prepare-commit-msg.sample
│   ├── pre-push.sample
│   ├── pre-rebase.sample
│   └── update.sample
├── index
├── info
│   └── exclude
├── logs
│   ├── HEAD
│   └── refs
│       └── heads
│           └── master
├── objects
│   ├── 09
│   │   └── 5e68cb0e446afdccc74ca4f6e08add47106ab8
│   ├── 47
│   │   └── 9abf23993860872cded7d081e4a8622900dbca
│   ├── 5d
│   │   └── 6ed1a098f588ab6526fd3b53a1331d298d44af
│   ├── 63
│   │   └── 452156647080f0f17e0d0cb14d7c1f1f90e22e
│   ├── 78
│   │   └── 270f5908030219758738a85d499b8bfe6baa2b
│   ├── 7a
│   │   ├── 7f3b22a21932aefe2b204fe528475ca358d422
│   │   └── df5fc60c3e8829843257e47d22f30fdfb78d42
│   ├── 7e
│   │   └── 184b2a7d9bd622e74df67fbc6a9f75345090db
│   ├── 89
│   │   └── 6049330e920c53a53017258b8334e75b41c779
│   ├── be
│   │   └── 0e31953e749a2d94f7386c508c2b7900112b50
│   ├── fc
│   │   └── bc6dac21cf3af7e7acf87c602a7d96fa836a6a
│   ├── info
│   └── pack
└── refs
    ├── heads
│   └── master
└── tags

22 directories, 29 files
git ls-files:列出文件;
git cat-fles:查看文件;
git hash-object:计算文件的hash码;
git write-tree:根据当前索引中的内容创建树对象;

简单总结git版本库:

git的版本库就是存储了所有用来维护和管理当前项目的修订版本和历史信息数据的一个数据库;
所以版本库就是数据库里面记录的大量的数据,存储了版本库当中所有文件的完整副本,甚至包括版本库自身的副本都有;
除此之外,每个版本库还保存了跟版本库相关的一组元数据,就是版本库提交者的用户信息邮箱地址等;
克隆还可以复制仓库内容,但只复制数据不复制元数据;

版本库中有非常重要的两个区域:缓存区和对象库;
对象库中有四类对象:块、树、提交、标签;
而索引是个临时的动态的二进制文件,用于描述某个时刻,整个工作目录当中或整个版本库当中某一个版本的状态信息;

因为索引里面反应了要么是当前目录里add进去的文件,要么是某一版本库当中已经被提交的文件的映射信息;索引内容的随时能变化的;
如果有多个版本都提交了, 如果此时整个工作目录反应的是第一个版本,那么索引当中显示的信息都是第一个版本中所有文件的映射信息;
如果此时让工作目录反应了第二个版本,索引中的文件立即会改变成第二个版本中所有文件映射信息了;所以说索引里面是临时的、动态的信息;
如果当前没有提交任何版本,只是add了一些数据,那么索引里反应的都是add进去文件的映射路径;

上一篇 下一篇

猜你喜欢

热点阅读