千家信息网

linux程序包管理初步

发表于:2025-01-21 作者:千家信息网编辑
千家信息网最后更新 2025年01月21日,Linux程序包管理我们linux操作系统从底层到高层的层次结构:最底层首先是硬件,计算机的计算能力都是在硬件逻辑上设计实现的,而不同厂商生产的芯片哪怕是同一厂商生产的同一型号的芯片,他们给我们提供的
千家信息网最后更新 2025年01月21日linux程序包管理初步

Linux程序包管理

我们linux操作系统从底层到高层的层次结构:

最底层首先是硬件,计算机的计算能力都是在硬件逻辑上设计实现的,而不同厂商生产的芯片哪怕是同一厂商生产的同一型号的芯片,他们给我们提供的计算接口都不一样,我们基本都同说过嵌入式开发,那么这种开发如果没有系统可以使用的话,主要是面向硬件,来写程序的,而这种程序只能适应一种或者几种有限的芯片,单片机尤其如此,现在很多嵌入式开发过程中,也已经有了一种通用层来实现,但是像单片机这种还是面向硬件编程的,

然后在底层之上就是将硬件接口封装起来的操作系统层,系统层将底层的硬件接口封装起来以后,通过系统调用(system call),向上输出,但是系统调用仍然很底层,为了加速开发操作,在操作系统层之上又加了半层叫"库(lib call",注意这里不是全层,而是半层,因为对于程序员来讲,可以面向半层"库"来写程序,也可以面向我们的操作系统来写程序,然后在库之上就是应用程序了,应用程序是给我们真正带来一些生产力的,我们的内核是不能完成任何具体的操作的,在这众多的应用类型程序中,我们知道有一类比较独特的,他主要是让用户跟主机交互的,shell任何利用操作系统完成任务的操作,都是通过应用程序完成的,所以我们说将来我们真正去实现运维操作的过程中,除了系统管理,库调用管理之外,其实作为运维人员来讲,就是不停的在我们的操作系统上安装配置某一应用程序包,然后让这个应用程序包运行起来,并提供服务,或者利用此工具完成某种具体相应操作的过程。那对于我们运维人员来讲,安装卸载管理程序包,是实现很多后期工作的最基础,最根本的任务,所以首先我们要学会灵活实现对于程序包的安装管理操作。

任何程序的运行,他有可能在程序包上提供两种格式的程序包,我们称之为源码包或者叫二进制包,那么我们将到这就不得不回顾一下以前讲到的APIABIAPI叫应用程序开发接口,ABI叫应用程序二进制接口。我们说过,API层次兼容未必ABI层次兼容。因为有些操作系统他们执行程序的格式,或者对二进制的是识别格式,并非是相同的,像linux跟类unixUnix-like)他们的操作系统,一般而言是相同的,他们的ABI是相同的,但是与windows相差很远,比方说windows系统上的可执行程序都是exe或者是msi的,而linux上的是elf格式的。所以他们在ABI层次不兼容,所以我们说即使我们使用高级编程语言,去编写程序,他们即便是在源码格式兼容的,但是一旦编译成功以后,本来是linux上面的格式的,而跑到windows上一样依然无法运行。所以有一个程序包,在linux上编译好了,他是二进制格式的,我们是不能拿来装到windows之上的。反过来如果一个程序包在windows上编译好了,是一个exe格式的文件,能不能在linux系统上运行起来呢?也很难。我们可以借助虚拟化将二者的差异将其磨平了,比方说现在的各种各样的应用程序,几乎都是针对库调用来开发的,很少说是直接通过系统调用来实现。即使是通过系统调用来实现,那也没问题,现在在windows上有很多程序模拟linux的库,在linux上也有很多模拟windows库的程序。

比方说现在这里有一个linux操作系统,没关系我们装一个WinE,这个wine就能在linux上模拟出windows库来。所以我们就可以借助wine来运行windows应用程序。比方说希望在linux上玩的各种各样的游戏,甚至还能给我们提供一个安装的路径,模拟出windows上的CDE盘。但这只是一个库虚拟层。同样在windows上我们也能提供linux运行环境。叫cywin,在windows之上运行一个cywin,他能够提供运行模拟出linux的运行环境来。所以说有一些程序只能在linux上运行的话,我们也能在windows上基于cywin运行起来,比如说像docker。这都不是正常的方案,正常的方案就是,由于abf库是不兼容的,至少在二进制层次上他们是不兼容的。所以他们没有办法安装块系统来实现系统调用。

库级别的虚拟化:我们可以借助库级别的虚拟化来抹平他们的不同。比如在linux上我们可以借助于wine来提供windows库从而能够运行windows应用程序。而在Windoes上我们借助于Cywin这个程序包,来虚拟模拟出linux的运行环境。从而使得那些二进制程序也能跨系统来运行了。

系统级开发:

C/C++:服务及应用程序:httpdvsftpdnginx

go

应用级开发

java/Python/perl/ruby/php

javahadoophbase,(他们的运行需要jvm就是java虚拟机)

Pythonopenstackopenstack是一个云操作系统)

运行Python程序需要用到pvmPython虚拟机)

运行perl:(依赖于perl

运行ruby:(依赖于ruby解释器)

运行php:(依赖于php解释器)

C/C++程序格式:

源代码:文本格式的程序代码;

二进制格式:文本格式的程序代码---->编译器------>二进制格式(二进制程序,库文件,配置文件,帮助文件)

(二进制格式我们仅仅是指他的陈序和库是二进制的,他的配置文件和帮助文件当然是文本的。我们知道Linux重要的哲学思想之一就是:用文本文件保存配置信息,这样带来的好处就在于,我们将来配置任何应用程序时,只需要依赖一款文本编辑器就可以。)

所以我们将来要想安装应用程序,那么针对我们两种格式的内容,他的安装方式就显然不一样,我们知道源代码不能运行,所以我们必须将其编译以后将其安装运行。对于源代码而言我们要手动把"文本格式的程序代码---->编译器------>二进制格式"这个过程手动完成。如果是别人给我们提供的是二进制格式的文件,上面的过程就不需要我们自己做了。因为我们拿到的直接是可运行格式的。万一很不幸的是对方仅提供了源代码怎么办呢?那么我们就只能自己编译了。而要想能够自己编译的话,那就依赖于编译开发环境。因为编译要依赖于编译器,依赖于头文件,依赖于开发库。我们的库也是源代码格式的,所以库也有两种文件。后来我们把它编译成了二进制格式的。

所以我们的编译开发环境依赖于:编译器,头文件,开发库。

如果我们拿到的是二进制格式的,那么就不需要编译,我们直接放到目录下就能运行起来了。

所以我们如果要执行源代码编译安装,那么我们需要额外的准备好几步。至少我们要想方设法保证我们的编译环境是完整的。而提供编译环境是相当劳心费神的工作。

我们学习程序包安装,首先了解二进制格式是怎么安装的,然后再去了解如何源代码编译安装。上面是C程序的格式。

那么如果是应用级的程序格式呢?

那么我们就以JAVAPython为例,那么JAVAPython应用程序格式是一样的,

java/python程序格式:

源代码

二进制

同样的道理,如果是源代码格式,那也只能编译了,如果是二进制格式那么直接使用即可

但是他们的编译不同在于,他们编译出来的不是二进制格式,或者不是能够直接在CPU上直接运行的二进制格式,而是编译成其能够在虚拟机上(jvm/pvm)运行的程序格式。也就是说通过虚拟机将其转换成能够运行的二进制格式。中间多了一层,所以性能很差。或者说比起C语言性能比较差。通常来说无论是C格式的源代码,还是java格式的源代码,通常他们的程序文件,都不止一个。那为什么不止一个?

将来我们的程序文件在编译时存在错综复杂的依赖关系,导致我们有一百个程序源文件,很有可能我们先编译第一个,在编译第三个,在编译第二个。因为第一个第三个被第二个所依赖。这样带来的结果就是谁先编译谁后编译,作为使用者我们没有能力管理他们的编译顺序。所以各种各样的源代码通常都使用一个项目管理工具。或者叫项目构建工具。

项目构建工具:

C/C++:最著名的项目构建工具就叫make

java:最经典的项目构建工具就叫maven

这样项目构建工具,也必须依赖项目开发环境才能构建。而对于java来讲他也有自己的开发环境。他也需要开发环境。只不过对于这两种的编程语言来讲。他们的开发环境,通常指的是,那个对应应用程序的虚拟机,和虚拟机上面的编译器。java源代码编译也需要开发环境,只不过是没有头文件,他们也需要:编译器,卡发库。这也是我们程序包的组成格式。

为了降低初级使用者的难度,我们应该怎样做?

我们应该使用程序包管理器,来协助这些终端用户的管理工作。

程序包管理器:

源代码---->目标二进制格式----->组织成为一个或有限几个"包"文件;

安装,升级,卸载,查询,(甚至对linux上还支持)校验

程序包管理器:

(程序包管理器大体上有哪几种呢?

对于windows来讲,大多数的应用程序都是exe格式的。这种格式应用程序给我们提供一个安装界面,我们只需点击下一步下一步即可,卸载的时候我们通过我们控制面板中的,卸载应用程序来实现,为什么我们能看到的只是下一步下一步就能完成安装呢?他其实就是通过一个程序包管理器打包成一个单个的exe文件,我们知道我们安装完以后他的确分散成多个可能放在program/files目录下某一个路径下,每一个应用程序都有一个目录,里面存放各种文件,有二进制的有库的,对于windows来说,他们的库是dll文件,动态链接库,dynamic link,而对于linux而言,叫so。)

目前linux有三大主流分支:

debiandpt,dpkg,".dep"

redhatredhat package manager,简称:rpm".rpm"rpm ispackage manager

S.u.S.Erpm".rpm"

Gentooports

ArchLinux

源代码:

命名方式:name-VERSION.tar.gz

VERSIONmajor.minor.release //主版本号.次版本号.发行版本

rpm包命名格式:

name-VERSION-release.arch.rpm //releaserpm包的release

VERSIONmajor.minor.release

release.arc:是指rpm包的发行号

release.os2.el7.i386.rpm

archetecture:架构,i386x64amd64),ppcnoarch

redis-3.0.2.targz----->redis-3.0.2-1.centos7.x64.rpm

changelog

拆包:主包和支包

主包:name-VERSION-release.arch.rpm

支包:name-function-VERSION-release.arch.rpm

functiondevel,utils,libs.....

依赖关系:

x,y,z

x------依赖--->y,z

y------依赖-------->A,B,C

c--------->y

前端工具:自动解决依赖关系;

yumrhel系列系统上rpm包管理器的前端工具;

apt-getapt-cache):dep包管理器的前端工具;

zyppersuserpm管理器前端工具;

dnfFedora 22+系统上rpm包管理器的前端工具;

程序包管理器:

功能:将编译好的应用程序的各组成文件打包成一个或几个程序包文件,从而更方便的 实现程序包的安装,升级,卸载,和查询等管理操作;

1.程序包的组成清单(每个程序包都单独实现);

文件清单

安装或卸载时运行的脚本

2数据库(公共)

程序包的名称和版本;

依赖关系;

功能说明;

安装生成的各个文件的文件路径及校验码信息;

CentOS系列系统的rpm数据库:路径

/var/lib/rpm

[root@centos7 ~]# ls /var/lib/rpm

Basenames __db.002 Group Obsoletename Requirename Triggername

Conflictname __db.003 Installtid Packages Sha1header

__db.001 Dirnames Name Providename Sigmd5

[root@centos7 ~]#

解释:

Group:包组,我们可以将程序划分成一个组,将来可以把一个程序包组全部安上,一个组 全部卸载。

Name:各个程序的名字;

Sigmd5MD5的校验码;

Triggername:触发器名称;

获取程序包的途径:

1)系统发行版的光盘或官方的文件服务器(或镜像站点):

http://mirrors.aliyun.com

http://mirrors.sohu.com

http://mirrors.163.com

2)项目的官方站点

举例:以thhp为例:站点httpd.apache.org

或者:

网站www.zabbix.com

打开网站www.zabbix.com网站,我们网站的导航栏中直接点击"Download",打开一个页面,前面的"Zabbix Packages"是zabbixrpm,再往下"Zabbix Sources"是源码包

所以说很多项目的官方站点也提供rpm包,并且我们一定注意,既然后能用rpm包安装的我们一定不要使用源码包编译安装。

3)第三方组织:EPEL

EPEL红帽官方的社区组织所维护的,发行光盘之外其他他们觉得比较有用比较重点,比较有名的程序包,都将制作成rpm包放在EPEL中。

其实国内的很多官方站点都有这样的epel

举例演示,我们以"mirrors.aliyun.com"为例:

在镜像阿里云的站点上"mirrors.aliyun.com"也有epel,在epel中会为我们的centos提供了众多的额外的补充包。

aEPEL

b)搜索引擎

http://pkgs.org(在rpm领域中,这是一个非常重要的);

rpmfind.net(要想搜哪个文件包直接搜。)

rpm.pbone.net

上面对话框中,我们可以输入正确的内容来进行查询,我们还可以通过后面的


Advanced RPM Search

来进行高级查询。

4)自己手动制作rpm包。

注意:不管上面的那种方式获得rpm包,只要是通过互联网上获得的,那么我们就可以认为即使原作者在里面没有做任何改变,或者修改,因为我们现在经常使用一些下载工具,像迅雷,那这种工具有点问题,为什么这种工具他的下载的速度非常的块,就是因为他们可能不是从官方的原站点下载的,或者是非完完全全的官方站点上下载的,其实从已经下载的用户的那里下载的,如果官方站点下载不到的话,而且这时候他恰恰搜索到某个用户的那里有这个文件,他也会传给你,但是这个文件可能被其他用户精心的修改过,制作了后门,所以以后我们在互联网上下载人格安全性较高的文件的,像银行官方站点的插件,还有像支付宝的证书等,我们就不应该在使用迅雷下载。以后这中工具少用,尤其是使用PHP下载的,是相当危险的,那么我们下载下来一个工具我们不知道他到底是不是官方提供的呢?

至少是所有的数据没有被篡改过,那怎么办呢?

我们建议要做MD5校验,提供一个MD5校验器,检验其合法性,来源合法性。

程序包的完整性要做校验;

CentOS系统上rpm命令管理程序包

安装,升级,卸载,查询和校验,数据库的维护

上面提到的"安装,升级,卸载,查询和校验"都用rpm命令来实现。

格式: rpm [OPTIONS] [PACKAGE_FILE]

安装用到的选项:-i=--install

升级用到的选项:-U=--update, -F=--freshen

卸载:-e=--erase

查询:-q=--query

校验:-V=--verify

数据库维护:--builddb,--initdb

安装:

安装时必须有对应的文件才行。

rpm {-i|--install} [install-options]PACKAGE_FILE ...

真正安装的时候应该使用选项:-ivh

rpm -ivh PACKAGE_FILE......

GENERAL OPTIONS(通用选项)

-vverbose;(输出相信信息)

-vv:(输出更详细的过程信息)

[install-options]

-hhash marks输出进度条;每个#表示2%的进度;

--test:如果不想真正的安装,仅仅是检查一下有没有潜在的冲 突的可能我们就可以测试安装,检查并报告依赖关系及 冲突消息等;

--nodeps:忽略依赖关系;不建议;

--replacepkgs:重新安装 (如果某个包安装过,但是后来我们 更改了他的配置文件,出了错,那我们刻意先卸载 再重新安装,但是我们还有一个更好的方法就是, 直接重新安装。)

--noscripts

--oldpackage:降级用的

--justdb:只是升级一下数据库。

注意rpm包可以自带脚本:

这些脚本有四类:

如果这四类都不想执行:--noscripts

preinstall:安装过程开始之前运行脚本,%pre --nopre

postinstall:安装过程完成以后运行的脚本,%post --nopost

preuninstall:卸载过程真正开始执行之前运行的脚本,%preun --nopreun

postuninstall:卸载过程完成之后运行的脚本,%postun --nopostun

--nosignature:不检查签名信息,不检查来源合法性;

--nodigest:不检查包完整性信息;

举例演示:

[root@centos6 Packages]# rpm -ivh zsh-4.3.11-4.el6.centos.2.x86_64.rpm

warning:zsh-4.3.11-4.el6.centos.2.x86_64.rpm: Header V3 RSA/SHA1 Signature, key IDc105b9de: NOKEY

Preparing... ###########################################[100%]

1:zsh ########################################### [100%]

[root@centos6 Packages]# cat /etc/shells

/bin/sh

/bin/bash

/sbin/nologin

/bin/dash

/bin/tcsh

/bin/csh

/bin/zsh

[root@centos6 Packages]#

演示重新安装replacepkgs

之前我们zsh已经安装过了,那我们先编辑一下zsh的配置文件"/etc/zshrc",比方说我们删除其中的几行,然后保存,因为我们执行"wq"之后,他的配置文件信息就不能恢复了。那么这时,我们

[root@centos6 media]# vim /etc/zshrc //先将zsh的配置文件中的内容删除一部分

[root@centos6 media]# cat /etc/zshrc

[root@centos6media]#rpm -ivh --replacepkgs /media/Packages/zsh-4.3.11-4.el6.centos.2.x86_64.rpm

[root@centos6media]# cat /etc/zshrc //发现配置文件内容没有恢复,这时因为我们的系统在我们没有将原先的配置文件删除,就在重装软件,那么系统就会认为我们这个配置文件修改是有目的的,我们也要记住replacepkgs是不能修改原来的配置文件的。所以我们在重装之前要先删除配置文件。

[root@centos6media]# rm -f /etc/zshrc

[root@centos6 media]# rpm -ivh --replacepkgs/media/Packages/zsh-4.3.11-4.el6.centos.2.x86_64.rpm

[root@centos6media]# cat /etc/zshrc //发现配置文件内容恢复了

#

# /etc/zshrc is sourced in interactiveshells. It

# should contain commands to set upaliases, functions,

# options, key bindings, etc.

#

## shell functions

#setenv() { export $1=$2 } # csh compatibility

# Set prompts

PROMPT='[%n@%m]%~%# ' # default prompt

#RPROMPT=' %~' # prompt for right side of screen

# bindkey -v # vi key bindings

# bindkey -e # emacs key bindings

bindkey ' ' magic-space # also do history expansion on space

_src_etc_profile_d()

{

# Make the *.sh things happier,and have possible ~/.zshenv options like

#NOMATCH ignored.

emulate -L ksh

#from bashrc, with zsh fixes

if [[ ! -o login ]]; then # We're not a login shell

for i in /etc/profile.d/*.sh; do

if [ -r "$i" ]; then

. $i

fi

done

unset i

fi

}

_src_etc_profile_d

unset -f _src_etc_profile_d

[root@centos6 media]#

不检查签名演示:

[root@centos6 media]# rpm -ivh --replacepkgs/media/Packages/zsh-4.3.11-4.el6.centos.2.x86_64.rpm

warning: /media/Packages/zsh-4.3.11-4.el6.centos.2.x86_64.rpm: Header V3RSA/SHA1 Signature, key ID c105b9de: NOKEY

Preparing... ########################################### [100%]

1:zsh ########################################### [100%

上面在重新安装软件包的时候,前面出现了警告,我们要见检查文件的合法性,文件的完整性,要检查文件的这两个选项,要依赖一个秘钥文件,要依赖这个包制作者的公钥。

[root@centos6 media]# rpm -ivh--replacepkgs --nosignature/media/Packages/zsh-4.3.11-4.el6.centos.2.x86_64.rpm

Preparing... ###########################################[100%]

1:zsh ########################################### [100%]

[root@centos6 media]#

上面安装时,警告消失。


0