Linux下rpm仓库搭建
一、概念
1.rpm是什么?在帮助文档里我们可以看到,rpm的名字是rpm package manage 的缩写, 从名字上看就可以知道rpm就是一个包管理工具。简单说rpm包就是把一些程序编译成二进制,和一些配置文件以及帮助文档等一些程序所依赖库打包在一起,这样形成的一个包。rpm大大的简化用户在Linux上安装软件的复杂度。
2.rpm仓库是什么?从字面理解,仓库就是集中存放东西的地方,同样rpm仓库就是集中存放rpm包的一个地方。这里说的‘地方’就是服务器。把很多rpm包放在一个特定的服务器的特定目录里, 再由比如http、nfs、ftp、file这些协议是用来提供给其他服务器下载rpm包的服务器,我们把这样的服务器称为rpm仓库服务器。
3.yum是什么?yum是Yellowdog Updater Modified 的缩写,翻译过来就是Yellowdog 更新修改,完全没有搞懂这个翻译到底是什么意思,查下百度百科吧!yum是一个在Fedora、Redhat以及CentOS中的shell前端软件包管理器。基于rpm包管理,能够从指定的服务器自下载rpm包并且安装,可以自动处理包依赖关系,并且能够一次安装所有所依赖的软件包,无需频繁的一次次下载安装。看了百度的解释,我们应该知道yum是什么了吧!简单说就是基于rpm包的前端管理工具,它可以帮我们解决包与包的依赖关系。
4.yum的工作机制。yum是c/s架构的工作机制,服务端提供rpm仓库,客户端根据服务端提供的仓库,通过特定协议去服务端下载所需要的包来进行安装。第一次去服务端,它会把服务端的仓库元数据缓存到本地,这样就不需要每次下载都去服务端查询。所谓仓库的元数据就是记录着该仓库有哪些包,包名,包与包的依赖关系等一些数据。每次要装包的时候,yum会去看看本地缓存的元数据,有没有需要的包的信息,如果有,他们会把相应的包以及所依赖的包一并去服务端下载下来,一并安装。
二、仓库的搭建
了解了以上的基本概念,下面我们开始来搭建rpm仓库吧
1.创建目录结构
[root@test ~]$mkdir -pv /rpm/centos/{,}/os/x86_64
mkdir: created directory `/rpm'
mkdir: created directory `/rpm/centos'
mkdir: created directory `/rpm/centos/'
mkdir: created directory `/rpm/centos//os'
mkdir: created directory `/rpm/centos//os/x86_64'
mkdir: created directory `/rpm/centos/'
mkdir: created directory `/rpm/centos//os'
mkdir: created directory `/rpm/centos//os/x86_64'
说明:本次实验是在centos6或centos7平台架构是x86_64的系统上来创建的目录
2.上传包
[root@test ~]$cd /rpm/centos//os/x86_64/
[root@test x86_64]$pwd
/rpm/centos//os/x86_64
[root@test x86_64]$mv /root/samba-3.6.-.el6.x86_64.rpm /root/vsftpd-2.2.-.el6.x86_64.rpm .
[root@test x86_64]$cd ../../..//os/x86_64/
[root@test x86_64]$pwd
/rpm/centos//os/x86_64
[root@test x86_64]$mv /root/tomcat-7.0.-.el7.noarch.rpm /root/samba-4.6.-.el7.x86_64.rpm .
说明: 上传了4个包,在/rpm/centos/6/os/x86_64/ 放了samba-3.6.23-41.el6.x86_64.rpm和vsftpd-2.2.2-24.el6.x86_64.rpm两个包,在/rpm/centos/7/os/x86_64放了samba-4.6.2-8.el7.x86_64.rpm和tomcat-7.0.76-2.el7.noarch.rpm两个包,本人先是把包传到服务器,然后移动到相应的目录里。
3.创建元数据
首先要检查自己的服务器有没有装createrepo这个工具
[root@test x86_64]$rpm -q createrepo
package createrepo is not installed
如果没有就安装
[root@test x86_64]$yum install -y createrepo
创建仓库元数据
[root@test ~]$createrepo /rpm/centos//os/x86_64/
Spawning worker with pkgs
Workers Finished
Gathering worker results Saving Primary metadata
Saving file lists metadata
Saving other metadata
Generating sqlite DBs
Sqlite DBs complete
[root@test ~]$createrepo /rpm/centos//os/x86_64/
Spawning worker with pkgs
Workers Finished
Gathering worker results Saving Primary metadata
Saving file lists metadata
Saving other metadata
Generating sqlite DBs
Sqlite DBs complete
[root@test ~]$ll /rpm/centos//os/x86_64/
total
drwxr-xr-x root root Sep : repodata
-rw-r--r--+ root root Sep : samba-3.6.-.el6.x86_64.rpm
-rw-r--r--+ root root Sep : vsftpd-2.2.-.el6.x86_64.rpm
[root@test ~]$ll /rpm/centos//os/x86_64/
total
drwxr-xr-x root root Sep : repodata
-rw-r--r--+ root root Sep : samba-4.6.-.el7.x86_64.rpm
-rw-r--r--+ root root Sep : tomcat-7.0.-.el7.noarch.rpm
说明:可以看到在两个仓库下各自生成了一个repodata的目录,这里说一下,要确定仓库的路径就是要看repodata这个目录的父目录是什么,repodata目录所在的父目录就是仓库的路径。
4.在客户端编写repo配置文件
[root@test ~]$cd /etc/yum.repos.d/
[root@test yum.repos.d]$ls
CentOS-Base.repo CentOS-Debuginfo.repo CentOS-fasttrack.repo CentOS-Media.repo CentOS-Vault.repo epel.repo epel-testing.repo
[root@test yum.repos.d]$mkdir repo_bak
[root@test yum.repos.d]$mv *.repo repo_bak
[root@test yum.repos.d]$ls
repo_bak
说明:本人采用的是客户端服务端在同一台设备上,为了不受其他repo配置文件的干扰,我们把以前的repo文件全部移动到另外一个备份目录里,然后在/etc/yum.repo.d/ 下创建一个repo文件。
[my_base]
name=this is test repo
baseurl=file:///rpm/centos/$releasever/os/$basearch
gpgcheck=
说明:创建一个.repo的文件,将以上内容添加到该文件,解释一下这个文件的内容,第一个中括号里表示仓库的名字,通常这个名字可以顺便起,一般不建议里面带空格,name=表示仓库的简要说明,baseurl= 表示仓库的路径,也就是repodata所在的目录的父目录,gpgcheck=0表示不检查包的gpg验证,若gpgcheck=1表示要验证。添加以上内容我们客户端就可以根据以上定义的baseurl去服务器仓库里找相应的包。
5.在客户端清除yum缓存,查看仓库列表
[root@test yum.repos.d]$yum clean all
Loaded plugins: fastestmirror, security
Cleaning repos: my_base
Cleaning up Everything
Cleaning up list of fastest mirrors
[root@test yum.repos.d]$yum repolist
Loaded plugins: fastestmirror, security
Determining fastest mirrors
my_base | 2.9 kB : ...
my_base/primary_db | 6.0 kB : ...
repo id repo name status
my_base this is test repo
repolist:
说明:可以看到我们刚才配置的仓库信息,repo id 就是我们刚才在repo文件中写的,也可看到里面有2个包。看到以上信息就表示rpm仓库在客户端上的配置成功了。
6.装包验证
[root@test yum.repos.d]$yum install vsftpd
Loaded plugins: fastestmirror, security
Setting up Install Process
Loading mirror speeds from cached hostfile
Resolving Dependencies
--> Running transaction check
---> Package vsftpd.x86_64 :2.2.-.el6 will be installed
--> Finished Dependency Resolution Dependencies Resolved ===================================================================================================================================
Package Arch Version Repository Size
===================================================================================================================================
Installing:
vsftpd x86_64 2.2.-.el6 my_base k Transaction Summary
===================================================================================================================================
Install Package(s) Total download size: k
Installed size: k
Is this ok [y/N]: y
Downloading Packages:
Running rpm_check_debug
Running Transaction Test
Transaction Test Succeeded
Running Transaction
Installing : vsftpd-2.2.-.el6.x86_64 /
Verifying : vsftpd-2.2.-.el6.x86_64 / Installed:
vsftpd.x86_64 :2.2.-.el6 Complete!
说明:本人以两个包作为例子来演示搭建rpm的过程,可能装这个包还有其他依赖包,但是仓库里面没有其他依赖,可能会出现报错的情况。
以上就是本地搭建repo仓库的过程。
下面我再说下如何搭建网络rpm仓库
1.先安装httpd web服务器
[root@test yum.repos.d]$yum install httpd -y
Loaded plugins: fastestmirror, security
Setting up Install Process
Determining fastest mirrors
* base: mirrors.aliyun.com
* extras: mirrors.aliyun.com
* updates: mirrors.aliyun.com
base | 3.7 kB :
base/primary_db | 4.7 MB :
epel | 5.3 kB :
epel/primary_db | 6.1 MB :
extras | 3.4 kB :
extras/primary_db | kB :
updates | 3.4 kB :
updates/primary_db | 5.7 MB :
Resolving Dependencies
--> Running transaction check
---> Package httpd.x86_64 :2.2.-.el6.centos will be installed
--> Finished Dependency Resolution Dependencies Resolved ===================================================================================================================================
Package Arch Version Repository Size
===================================================================================================================================
Installing:
httpd x86_64 2.2.-.el6.centos base k Transaction Summary
===================================================================================================================================
Install Package(s) Total download size: k
Installed size: 3.0 M
Downloading Packages:
httpd-2.2.-.el6.centos.x86_64.rpm | kB :
Running rpm_check_debug
Running Transaction Test
Transaction Test Succeeded
Running Transaction
Installing : httpd-2.2.-.el6.centos.x86_64 /
Verifying : httpd-2.2.-.el6.centos.x86_64 / Installed:
httpd.x86_64 :2.2.-.el6.centos Complete!
说明:httpd默认工作路径在/var/www/html/ 下,也就是放html网页代码的地方。
2.在httpd 的工作路径下 创建目录结构
[root@test ~]$mkdir -pv /var/www/html/centos/{,}/os/x86_64
mkdir: created directory `/var/www/html/centos'
mkdir: created directory `/var/www/html/centos/'
mkdir: created directory `/var/www/html/centos//os'
mkdir: created directory `/var/www/html/centos//os/x86_64'
mkdir: created directory `/var/www/html/centos/'
mkdir: created directory `/var/www/html/centos//os'
mkdir: created directory `/var/www/html/centos//os/x86_64'
3.挂载安装光盘到相应的位置
[root@test ~]$mount /dev/sr0 /var/www/html/centos//os/x86_64/
mount: block device /dev/sr0 is write-protected, mounting read-only
说明:我这里只挂载了centos6的一个安装光盘,centos7下面的目录也可以使用同样的方式挂载centos7的安装光盘,这里用centos6作为演示。这里还要说一下安装光盘本身就有repodata目录,我们这里就不需要再创建元数据信息。
4.开启httpd
[root@test ~]$/etc/init.d/httpd start
Starting httpd: httpd: apr_sockaddr_info_get() failed for test
httpd: Could not reliably determine the server's fully qualified domain name, using 127.0.0.1 for ServerName
[ OK ]
5.客户端配置repo文件
[my_base]
name=this is test repo
baseurl=http://127.0.0.1/centos/$releasever/os/$basearch
gpgcheck=
说明:这里只需要把仓库的路径改成以http的协议方式去获取,当然为了不受其他yum源的影响,本次实验我们也要把其他的repo文件放在一个备份目录。
6.客户端清除yum缓存,并查看仓库列表
[root@test x86_64]$yum clean all
Loaded plugins: fastestmirror, security
Cleaning repos: my_base
Cleaning up Everything
[root@test x86_64]$yum repolist
Loaded plugins: fastestmirror, security
Determining fastest mirrors
my_base | 4.0 kB :
my_base/primary_db | 4.6 MB :
repo id repo name status
my_base this is test repo ,
repolist: ,
说明:可以看到刚才搭建的仓库成功的出现在仓库列表中了。
7.安装一个包,验证仓库
[root@test ~]$yum install java -y
Loaded plugins: fastestmirror, security
Setting up Install Process
Loading mirror speeds from cached hostfile
Resolving Dependencies
--> Running transaction check
---> Package java-1.8.-openjdk.x86_64 :1.8.0.45-.b13.el6 will be installed
--> Processing Dependency: java-1.8.-openjdk-headless = :1.8.0.45-.b13.el6 for package: :java-1.8.-openjdk-1.8.0.45-.b13.el6.x86_64
--> Processing Dependency: xorg-x11-fonts-Type1 for package: :java-1.8.-openjdk-1.8.0.45-.b13.el6.x86_64
--> Processing Dependency: libjvm.so(SUNWprivate_1.)(64bit) for package: :java-1.8.-openjdk-1.8.0.45-.b13.el6.x86_64
--> Processing Dependency: libjli.so(SUNWprivate_1.)(64bit) for package: :java-1.8.-openjdk-1.8.0.45-.b13.el6.x86_64
--> Processing Dependency: libjava.so(SUNWprivate_1.)(64bit) for package: :java-1.8.-openjdk-1.8.0.45-.b13.el6.x86_64
--> Processing Dependency: libjli.so()(64bit) for package: :java-1.8.-openjdk-1.8.0.45-.b13.el6.x86_64
--> Processing Dependency: libjava.so()(64bit) for package: :java-1.8.-openjdk-1.8.0.45-.b13.el6.x86_64
--> Processing Dependency: libgif.so.()(64bit) for package: :java-1.8.-openjdk-1.8.0.45-.b13.el6.x86_64
--> Processing Dependency: libawt.so()(64bit) for package: :java-1.8.-openjdk-1.8.0.45-.b13.el6.x86_64
--> Running transaction check
---> Package giflib.x86_64 :4.1.-3.1.el6 will be installed
---> Package java-1.8.-openjdk-headless.x86_64 :1.8.0.45-.b13.el6 will be installed
--> Processing Dependency: tzdata-java >= 2014f- for package: :java-1.8.-openjdk-headless-1.8.0.45-.b13.el6.x86_64
--> Processing Dependency: jpackage-utils for package: :java-1.8.-openjdk-headless-1.8.0.45-.b13.el6.x86_64
---> Package xorg-x11-fonts-Type1.noarch :7.2-.el6 will be installed
--> Processing Dependency: ttmkfdir for package: xorg-x11-fonts-Type1-7.2-.el6.noarch
--> Processing Dependency: ttmkfdir for package: xorg-x11-fonts-Type1-7.2-.el6.noarch
--> Processing Dependency: mkfontdir for package: xorg-x11-fonts-Type1-7.2-.el6.noarch
--> Processing Dependency: mkfontdir for package: xorg-x11-fonts-Type1-7.2-.el6.noarch
--> Running transaction check
---> Package jpackage-utils.noarch :1.7.-3.14.el6 will be installed
---> Package ttmkfdir.x86_64 :3.0.-32.1.el6 will be installed
---> Package tzdata-java.noarch :2015e-.el6 will be installed
---> Package xorg-x11-font-utils.x86_64 :7.2-.el6 will be installed
--> Processing Dependency: libfontenc.so.()(64bit) for package: :xorg-x11-font-utils-7.2-.el6.x86_64
--> Processing Dependency: libXfont.so.()(64bit) for package: :xorg-x11-font-utils-7.2-.el6.x86_64
--> Running transaction check
---> Package libXfont.x86_64 :1.4.-.el6_6 will be installed
---> Package libfontenc.x86_64 :1.0.-.el6 will be installed
--> Finished Dependency Resolution Dependencies Resolved ===========================================================================================================================================================================
Package Arch Version Repository Size
===========================================================================================================================================================================
Installing:
java-1.8.-openjdk x86_64 :1.8.0.45-.b13.el6 my_base k
Installing for dependencies:
giflib x86_64 4.1.-3.1.el6 my_base k
java-1.8.-openjdk-headless x86_64 :1.8.0.45-.b13.el6 my_base M
jpackage-utils noarch 1.7.-3.14.el6 my_base k
libXfont x86_64 1.4.-.el6_6 my_base k
libfontenc x86_64 1.0.-.el6 my_base k
ttmkfdir x86_64 3.0.-32.1.el6 my_base k
tzdata-java noarch 2015e-.el6 my_base k
xorg-x11-font-utils x86_64 :7.2-.el6 my_base k
xorg-x11-fonts-Type1 noarch 7.2-.el6 my_base k Transaction Summary
===========================================================================================================================================================================
Install Package(s) Total download size: M
Installed size: M
Downloading Packages:
(/): ttmkfdir-3.0.-32.1.el6.x86_64.rpm | kB :
(/): java-1.8.-openjdk-1.8.0.45-.b13.el6.x86_64.rpm | kB :
(/): jpackage-utils-1.7.-3.14.el6.noarch.rpm | kB :
(/): tzdata-java-2015e-.el6.noarch.rpm | kB :
(/): java-1.8.-openjdk-headless-1.8.0.45-.b13.el6.x86_64.rpm | MB :
(/): xorg-x11-font-utils-7.2-.el6.x86_64.rpm | kB :
(/): libXfont-1.4.-.el6_6.x86_64.rpm | kB :
(/): xorg-x11-fonts-Type1-7.2-.el6.noarch.rpm | kB :
(/): giflib-4.1.-3.1.el6.x86_64.rpm | kB :
(/): libfontenc-1.0.-.el6.x86_64.rpm | kB :
---------------------------------------------------------------------------------------------------------------------------------------------------------------------------
Total MB/s | MB :
Running rpm_check_debug
Running Transaction Test
Transaction Test Succeeded
Running Transaction
Installing : libfontenc-1.0.-.el6.x86_64 /
Installing : libXfont-1.4.-.el6_6.x86_64 /
Installing : :xorg-x11-font-utils-7.2-.el6.x86_64 /
Installing : giflib-4.1.-3.1.el6.x86_64 /
Installing : tzdata-java-2015e-.el6.noarch /
Installing : jpackage-utils-1.7.-3.14.el6.noarch /
Installing : :java-1.8.-openjdk-headless-1.8.0.45-.b13.el6.x86_64 /
Installing : ttmkfdir-3.0.-32.1.el6.x86_64 /
Installing : xorg-x11-fonts-Type1-7.2-.el6.noarch /
Installing : :java-1.8.-openjdk-1.8.0.45-.b13.el6.x86_64 /
Verifying : ttmkfdir-3.0.-32.1.el6.x86_64 /
Verifying : :java-1.8.-openjdk-1.8.0.45-.b13.el6.x86_64 /
Verifying : jpackage-utils-1.7.-3.14.el6.noarch /
Verifying : tzdata-java-2015e-.el6.noarch /
Verifying : :java-1.8.-openjdk-headless-1.8.0.45-.b13.el6.x86_64 /
Verifying : :xorg-x11-font-utils-7.2-.el6.x86_64 /
Verifying : libXfont-1.4.-.el6_6.x86_64 /
Verifying : xorg-x11-fonts-Type1-7.2-.el6.noarch /
Verifying : giflib-4.1.-3.1.el6.x86_64 /
Verifying : libfontenc-1.0.-.el6.x86_64 / Installed:
java-1.8.-openjdk.x86_64 :1.8.0.45-.b13.el6 Dependency Installed:
giflib.x86_64 :4.1.-3.1.el6 java-1.8.-openjdk-headless.x86_64 :1.8.0.45-.b13.el6 jpackage-utils.noarch :1.7.-3.14.el6
libXfont.x86_64 :1.4.-.el6_6 libfontenc.x86_64 :1.0.-.el6 ttmkfdir.x86_64 :3.0.-32.1.el6
tzdata-java.noarch :2015e-.el6 xorg-x11-font-utils.x86_64 :7.2-.el6 xorg-x11-fonts-Type1.noarch :7.2-.el6 Complete!
到此网络rpm仓库就搭建完毕!!
Linux下rpm仓库搭建的更多相关文章
- Linux下RPM包的安装
Linux下RPM包安装 二进制包(RPM包.系统默认包) RPM安装 rpm -ivh 包全名(查询依赖网址:http://www.rpmfind.net) -i(install):安装 -v(ve ...
- Linux下RPM软件包的安装及卸载
http://os.51cto.com/art/201001/177866.htm 在 Linux 操作系统下,几乎所有的软件均通过RPM 进行安装.卸载及管理等操作.RPM 的全称为Redhat P ...
- [转]Linux下RPM软件包的安装及卸载 yum操作
在 Linux 操作系统下,几乎所有的软件均通过RPM 进行安装.卸载及管理等操作.RPM 的全称为Redhat Package Manager ,是由Redhat 公司提出的,用于管理Linux 下 ...
- linux下rpm包安装、配置和卸载mysql
WIN10下虚拟机:VMware workstation 12 PRO 安装 # 1.查看系统版本 [root@vm-xiluhua][/home/xiluhua]$ cat /etc/redhat- ...
- 64位linux下rpm安装mysql的5.5.55版本
昨天同事开了个阿里云环境,让我帮他安装mysql数据库,本想着很简单的一件事,结果还是折腾了一番.坑很多,一路趟过,一个接一个,只能硬着头皮冲. 首先是下载压缩包,因为采用了rpm安装方式,所以下载的 ...
- Linux下rpm及yum安装jdk
笔者使用的Linux操作系统为CentOS7,Linux中有三种安装jdk的方式,在这里笔者主要介绍两种方式,即rpm和yum安装及卸载jdk,还有一种是直接从官网下载 tar.gz 压缩文件上传到L ...
- linux下rpm包安装、配置和卸载mysq
l WIN10下虚拟机:VMware workstation 12 PRO 安装 # 1.查看系统版本 [root@vm-xiluhua][/home/xiluhua]$ cat /etc/red ...
- Linux下RPM、tar.gz、DEB格式软件包的区别
初接解Linux的朋友一定对软件的安装特别头疼,同样都是for Linux,但RPM.tar.gz.DEB包还是有很大区别的,这种区别很有可能使你的安装过程进行不下去.那我们应该下载什么格式的包呢 ...
- Linux 下RPM打包制作流程
原文地址:https://www.cnblogs.com/postgres/p/5726339.html 开始前的准备 安装rpmbuild软件包 yum -y install rpm-build 生 ...
随机推荐
- Python flask 构建可扩展的restful apl☝☝☝
Python flask 构建可扩展的restful apl☝☝☝ Flask-RESTful是flask的扩展,增加了对快速构建REST API的支持.Flask-RESTful通过最少的设置鼓励最 ...
- java23种设计模式(二)抽象工厂模式
我们接着上一章的工厂方法模式继续学习一下抽象工厂模式. 抽象工厂模式:在工厂模式中,如果有多个产品,则就是抽象工厂模式. 例子: 有一个工厂开了两个子公司,专门用来生产电脑配件键盘和鼠标,一个是联想工 ...
- 毕业生想学习web前端,应该怎么学才能最快找到工作?
首先无论你要学习任何技能,必须有一个清晰的版图,什么是清晰的版图呢?首先了解你学的技术将来要从事什么工作,这个工作的条件是哪些? 然后你要有一个非常清晰的学习大纲,切记学习任何东西都要系统,不可胡乱的 ...
- Ubuntu 搜狗输入法输入异常
电脑放置一段时间,不使用.过了一会,再使用 sogou 输入法的时候,发现,输入法不起作用了. 切花到用户目录 ~/.config 里面 rm -rf Sogou* 删除搜狗的配置文件,退出当前账户, ...
- 我要学并发-Java内存模型到底是什么
内存模型 在计算机CPU,内存,IO三者之间速度差异,为了提高系统性能,对这三者速度进行平衡. CPU 增加了缓存,以均衡与内存的速度差异: 操作系统增加了进程.线程,以分时复用 CPU,进而均衡 C ...
- git生成SSH秘钥
1.进入git bash , 输入 cd ~/.ssh/ ,没有的话,自己创建 mkdir ~/.ssh , 然后进入该文件夹完成生成秘钥步骤 2.配置全局的name和email,这里是的你githu ...
- .NET Framework概述
1.NET Framework是为其运行的应用程序提供各种服务的托管执行环境,它包括两个主要组件:(1).公共语言运行时 (CLR),(2)..NET Framework 类库: 2.NET Fram ...
- Beetlex服务框架之Webapi版本访问控制
在应用服务中API更新是很普遍的事情,为了服务良好地运作很多时候需要新旧版本同时兼容:为了应对这一系列的需求FastHttpApi在新版中强化了Url重写机制来支持API访问版本控制,由原来固定的重写 ...
- Veins(车载通信仿真框架)入门教程(三)——多跳路由实现指导
Veins(车载通信仿真框架)入门教程(三)——多跳路由实现指导 Veins(车载通信仿真框架)入门教程(三)——多跳路由实现指导 必要的message类实现 从下面开始是在veins/src/vei ...
- unity 内置shader
几个有用的Unity 内置shader: (一)Standard RenderingMode:Opaque为实体渲染,更改Color的透明通道不会有影响:Cutout会把图片的透明通道显示出来,非严格 ...