1. 背景
一个php的项目,里面涉及一些图片操作。原本在测试环境运行正常的功能,但是发布到生产环境后,却报错了:“call to undefined function imagecreatefromjpeg”。
2. 原因分析
经过排查,笔者发现两个环境在处理png格式图片时,都能成功。但是如果上传了jpeg或webp格式,生产环境就会报错。
进一步检查两个环境的phpinfo,笔者发现两边虽然都安装了GD库,但测试环境的GD库支持所有图片格式,而生产环境的GD库只支持png和gif。
除了phpinfo()函数,还可以使用php -ri gd 命令查看gd模块的细节信息。
测试环境
GD Support => enabled
GD Version => bundled (2.1.0 compatible)
FreeType Support => enabled
FreeType Linkage => with freetype
FreeType Version => 2.8.0
GIF Read Support => enabled
GIF Create Support => enabled
JPEG Support => enabled
libJPEG Version => 8
PNG Support => enabled
libPNG Version => 1.5.13
WBMP Support => enabled
XBM Support => enabled
WebP Support => enabled
Directive => Local Value => Master Value
gd.jpeg_ignore_warning => 1 => 1
生产环境
GD Support => enabled
GD Version => bundled (2.1.0 compatible)
GIF Read Support => enabled
GIF Create Support => enabled
PNG Support => enabled
libPNG Version => 1.5.13
WBMP Support => enabled
XBM Support => enabled
Directive => Local Value => Master Value
gd.jpeg_ignore_warning => 1 => 1
所以生产环境之所以会报错,是因为:生产环境的GD库,并没有支持jpeg和webp格式的能力。因此,需要针对GD库进行扩展。
那么问题来了,为什么测试环境的GD库支持的全,而生产环境的GD库却支持的少呢?
这是由于当时搭建测试环境和生产环境的人,不是同一人,他们用了截然不同的方式进行搭建。测试环境使用了yum安装,而生产环境使用了源码编译安装。
处理这个问题时,我不禁思考,yum安装和源码编译安装,孰优孰劣?
下面以扩展gd库为例,我会从使用方式、优缺点这两个方面进行探讨这个问题。
3. 区别一:一般步骤
3.1 yum安装
yum安装很简单,只需要输入几个命令就行。
- 如果不确定安装的名字,可以先搜索一下: yum search [关键字];
- 如果不确定是否已安装,可以使用whereis [关键字] 或 yum info [关键字];
- 安装:yum install [软件包名]。
举个栗子1:用yum方式为GD库开启webp支持
- 搜索名字:yum search webp
- 是否已安装:yum info libwebp-devel
- 安装:yum install libwebp-devel
3.2 源码安装
源码安装相对复杂,不过和windows安装应用的步骤相似:
- 访问对应的网址,然后找到对应的平台和版本的安装包并下载;
- 对源码包解压缩,进入压缩后的目录;
- 编译源码,进行安装。
举个栗子2:用源码编译为GD库开启jpeg支持
- 下载jpeg包:wget http://www.ijg.org/files/jpegsrc.v9c.tar.gz
- 解压缩:tar -zxvf jpegsrc.v9c.tar.gz
- 进入目录:cd jpeg-9c/
- 编译jpeg包并指定安装目录 ./configure --prefix=/usr/local/phpjpeglib --enable-shared --enable-static
- 进入php扩展的gd源码目录(注意不是php源码根目录):cd /root/php-7.4.10/ext/gd
最好先make clean一下,我就是因为没有clean,虽然编译安装没报错,但依然无法打开支持。- 执行phpize:/usr/local/php/bin/phpize
- 重新编译gd,指明php-config和刚刚编译安装的jpeg包和freetype的路径。
./configure --with-php-config=/usr/local/php/bin/php-config --with-jpeg=/usr/local/phpjpeglib --with-freetype=/usr/local/freetype- 安装 make && make install
- 重启apache或nginx
4. 区别二:优缺点
4.1. yum安装
优点
- 简单:基本一个命令搞定。
- 自动安装依赖包:比如这次的gd库,使用yum安装的gd库就是因为自动安装了依赖包,所以能支持全部格式的图片处理。
- 自动安装服务,可以使用service或systemctl命令启动或关闭服务。
缺点
- 版本单一:基本一个源只能安装一个固定的版本,如centos默认yum源只能安装php5.4。
解决方法:更新或使用新的yum源。
- 安装成功后,找不到装在哪了。
解决方法:其实大部分软件的安装路径都有规律,如配置文件都在/etc/下面,日志在/var/log下面。
也可以用 whereis 命令查看应用的路径。
4.2. 源码编译安装
优点
- 灵活:可以根据需要选择软件的任意版本。
- 安装路径自定义:编译时使用prefix参数可以自定义安装路径。
缺点
- 复杂:相比yum,编译的步骤很多。
适合动手能力强的,不喜欢使用默认安装目录的运维同学。(毕竟目录不同了,也能增加一定程度的安全性。)
- 不会自动安装依赖包:本案例中,必须先安装依赖(jpegsrc, libwebp)再进行gd的二次编译才行。
虽然需要花时间下载安装依赖,但依靠网络搜索也不难。也可以减少安装用不到的软件。
- 不会自动安装服务。
/etc/init.d/下面依然可以毕竟方便的启动或关闭软件。
小结
yum安装和编译安装各有千秋,yum安装简单但不灵活,编译安装灵活但比较复杂。
笔者喜欢yum安装,毕竟更新一下yum源也很不难。
那么,你更喜欢哪种方式呢?
2022年7月11日更新:
- centos7.5下安装php7.2,需要更新的yum源
# 1. 安装epel_release
yum install epel-release
# 2. 安装 remi 源(可以使用php-*安装):
yum install http://rpms.remirepo.net/enterprise/remi-release-7.rpm
# 或者安装webtatic源(可以使用php72w-*安装)
rpm -Uvh https://mirror.webtatic.com/yum/el7/webtatic-release.rpm
# 3. 安装 yum 扩展包:
yum install yum-utils
# 4. 启用 remi 仓库:
yum-config-manager --enable remi-php72
yum update
# 5. 安装 PHP7.2
yum install php
#安装 php-fpm 和一些其他模块
yum install php-fpm php-gd php-json php-mbstring php-mysqlnd php-xml php-xmlrpc php-opcache
yum install php-pecl-zip php-ldap php-odbc php-posix php-shmop php-sysvmsg php-sysvsem php-sysvshm php-xmlrpc
# 6. 输入 php -v 查看安装结果
- php-fpm 服务
# 1.设置开机自启
$ sudo systemctl enable php-fpm.service
# 2. 常用 php-fpm 命令
# 开启服务
$ sudo systemctl start php-fpm.service
# 停止服务
$ sudo systemctl stop php-fpm.service
# 查看状态
$ sudo systemctl status php-fpm.service
# 3. 配置权限
# 通过 egrep 查询 nginx 服务器的用户和用户组:
$ egrep '^(user|group)' /etc/nginx/nginx.conf
# 结果示例:
user nginx;
# 编辑 /etc/php-fpm.d/www.conf,修改执行 php-fpm 的权限:
$ sudo vi /etc/php-fpm.d/www.conf
# 设置用户和用户组为 nginx:
user = nginx
group = nginx
# 保存并关闭文件,重启 php-fpm 服务:
$ sudo systemctl restart php-fpm.service