很多童鞋在启动MysqL的时候,碰到过这个错误,
首先,澄清一点,出现这个错误的前提是:通过服务脚本来启动MysqL。通过MysqLd_safe或MysqLd启动MysqL实例并不会报这个错误。
那么,出现这个错误的原因具体是什么呢?
哈哈,对分析过程不care的童鞋可直接跳到文末的总结部分~
总结
下面,来分析下MysqL的服务启动脚本
脚本完整内容如下:
#!/bin/sh# copyright Abandoned 1996 TCX DataKonsult AB & Monty Program KB & Detron HB# This file is public domain and comes with NO WARRANTY of any kind# MysqL daemon start/stop script.# Usually this is put in /etc/init.d (at least on machines SYSV R4 based# systems) and linked to /etc/rc3.d/S99MysqL and /etc/rc0.d/K01MysqL.# When this is done the MysqL server will be started when the machine is# started and shut down when the systems goes down.# Comments to support chkconfig on RedHat linux# chkconfig: 2345 64 36# description: A very fast and reliable sql database engine.# Comments to support LSB init script conventions### BEGIN INIT INFO# ProvIDes: MysqL# required-Start: $local_fs $network $remote_fs# Should-Start: ypbind nscd ldap ntpd xntpd# required-Stop: $local_fs $network $remote_fs# Default-Start: 2 3 4 5# Default-Stop: 0 1 6# Short-Description: start and stop MysqL# Description: MysqL is a very fast and reliable sql database engine.### END INIT INFO # If you install MysqL on some other places than /usr/local/MysqL,then you# have to do one of the following things for this script to work:## - Run this script from within the MysqL installation directory# - Create a /etc/my.cnf with the following information:# [MysqLd]# basedir=<path-to-MysqL-installation-directory># - Add the above to any other configuration file (for example ~/.my.ini)# and copy my_print_defaults to /usr/bin# - Add the path to the MysqL-installation-directory to the basedir variable# below.## If you want to affect other MysqL variables,you should make your changes# in the /etc/my.cnf,~/.my.cnf or other MysqL configuration files.# If you change base dir,you must also change datadir. These may get# overwritten by settings in the MysqL configuration files.basedir=datadir=# Default value, seconds,afterwhich the script should timeout waiting# server start. # Value here is overrIDen by value my.cnf. # 0 means don't wait at all# Negative numbers mean to wait indefinitelyservice_startup_timeout=900# Lock directory for RedHat / SuSE.lockdir=/var/lock/subsys'lock_file_path="$lockdir/MysqL"# The following variables are only set for letting MysqL.server find things.# Set some defaultsMysqLd_pID_file_path=if test -z $basedir" basedir=/usr/local/MysqL bindir=/usr/local/MysqL/bin $datadir" datadir=/usr/local/MysqL/data fi sbindir=/usr/local/MysqL/bin libexecdir=/usr/local/MysqL/binelse bindir=$basedir/bin datadir=$basedir/data sbindir=$basedir/sbin libexecdir=$basedir/libexec# datadir_set is used to determine if datadir was set (and so should be# *not* set insIDe of the --basedir= handler.)datadir_set=## Use LSB init script functions for printing messages,1)"> possible#lsb_functions=/lib/lsb/init-functionsif test -f $lsb_functions ; . $lsb_functions log_success_msg() { echo SUCCESS! $@ } log_failure_msg() { ERROR! $@ }PATH=/sbin:/usr/sbin:/bin:/usr/bin:$basedir/binexport PATHmode=$1 # start or stop[ $# -ge 1 ] && shiftother_args=$* # uncommon,but needed when called from an RPM upgrade action # Expected: --skip-networking --skip-grant-tables # They are not checked here,intentionally,as it is the resposibility # of the spec" author to give correct arguments only.case `testing\c"`,`echo -n testing` in *c*,-n*) echo_n= echo_c= ;; *c*,*) echo_n=-n echo_c= ;; *) echo_n= echo_c=\c ;;esacparse_server_arguments() { for arg do case $argin --basedir=*) basedir=`" | sed -e s/^[^=]*=//` bindir=" $datadir_set"; datadir= sbindir= libexecdir= ;; --datadir=*) datadir=`` datadir_set= ;; --pID-file=*) MysqLd_pID_file_path=`` ;; --service-startup-timeout=*) service_startup_timeout=`` ;; esac }wait_for_pID () { verb=" # created | removed pID=" # process ID of the program operating on the pID- pID_file_path=" # path to the PID . i=0 avoID_race_condition=by checking again" while test $i -ne $service_startup_timeout ; do $verbin created) # wait for a PID- to pop into existence. test -s $pID_file_path" && i='' && break ;; removedfor this PID- to disappear test ! -s break ;; *) wait_for_pID () usage: wait_for_pID created|removed pID pID_file_path exit ;; # if server isnt running,then pID-file will never be updated if test -n $pIDthen if kill -" 2>/dev/null; : # the server still runs # The server may have exited between the last pID- check and Now. $avoID_race_condition avoID_race_condition="" continue # Check again. # theres nothing that will affect the file. log_failure_msg The server quit without updating PID file ($pID_file_path). return 1 # not waiting any more. fi fi echo $echo_n .$echo_c i=`expr $i + ` sleep 1 done $i" ; log_success_msg return 0 log_failure_msg return 1 }# Get arguments from the my.cnf which is read from Now on is [MysqLd]if test -x ./bin/my_print_defaults print_defaults=./bin/my_print_defaultselif test -x $bindir/$bindir/my_print_defaultsMysqL_print_defaults$bindir/MysqL_print_defaults # Try to find basedir my.cnf conf=/etc/my.cnf print_defaults= if test -r $conf subpat=^[^=]*basedir[^=]*=\(.*\)$ dirs=`/$subpat/!d" -e s/// $conf` for d $dirs do d=`echo $d | s/[ ]//g` if test -x $d/bin/my_print_defaults" print_defaults= break fi $d/bin/MysqL_print_defaultsdone # Hope its in the PATH ... but I doubt it test -z $print_defaults" && print_defaults=my_print_defaults## Read defaults file from basedir'. If there is no defaults there# check if its in the old (depricated) place (datadir) and read it from there#extra_args=""if test -r $basedir/my.cnf extra_args=-e $basedir/my.cnfelse $datadir/my.cnf extra_args=-e $datadir/my.cnffiparse_server_arguments `$print_defaults $extra_args MysqLd server MysqL_server MysqL.server`## Set pID file not given#$MysqLd_pID_file_path MysqLd_pID_file_path=$datadir/`hostname`.pIDin /* ) ;; * ) MysqLd_pID_file_path="$datadir/$MysqLd_pID_file_path" ;; esacficase "$mode" in 'start') # Start daemon # Safeguard (relative paths,core dumps..) cd $basedir echo $echo_n "Starting MysqL" if test -x $bindir/MysqLd_safe then # Give extra arguments to MysqLd with the my.cnf file. This script # may be overwritten at next upgrade. $bindir/MysqLd_safe --datadir="$datadir" --pID-file="$MysqLd_pID_file_path" $other_args >/dev/null 2>&1 & wait_for_pID created "$!" "$MysqLd_pID_file_path"; return_value=$? # Make lock for RedHat / SuSE if test -w "$lockdir" then touch "$lock_file_path" fi exit $return_value else log_failure_msg "Couldn't find MysqL server ($bindir/MysqLd_safe)" fi ;; 'stop') # Stop daemon. We use a signal here to avoID having to kNow the # root password. if test -s "$MysqLd_pID_file_path" then MysqLd_pID=`cat "$MysqLd_pID_file_path"` if (kill -0 $MysqLd_pID 2>/dev/null) then echo $echo_n "Shutting down MysqL" kill $MysqLd_pID # MysqLd should remove the pID file when it exits,so wait for it. wait_for_pID removed "$MysqLd_pID" "$MysqLd_pID_file_path"; return_value=$? else log_failure_msg "MysqL server process #$MysqLd_pID is not running!" rm "$MysqLd_pID_file_path" fi # Delete lock for RedHat / SuSE if test -f "$lock_file_path" then rm -f "$lock_file_path" fi exit $return_value else log_failure_msg "MysqL server PID file Could not be found!" fi ;; 'restart') # Stop the service and regardless of whether it was # running or not,start it again. if VIEw Code stop $other_args; then 首先,定义相关参数 start $other_args else log_failure_msg "Failed to stop running server,so refusing to try to start." exit 1 fi ;; 'reload'|'force-reload') if test -s "$MysqLd_pID_file_path" ; then read MysqLd_pID < "$MysqLd_pID_file_path" kill -HUP $MysqLd_pID && log_success_msg "Reloading service MysqL" touch "$MysqLd_pID_file_path" else log_failure_msg "MysqL PID file Could not be found!" exit 1 fi ;; 'status') # First,check to see if pID file exists if test -s "$MysqLd_pID_file_path" ; then read MysqLd_pID < "$MysqLd_pID_file_path" if kill -0 $MysqLd_pID 2>/dev/null ; then log_success_msg "MysqL running ($MysqLd_pID)" exit 0 else log_failure_msg "MysqL is not running,but PID file exists" exit 1 fi else # Try to find appropriate MysqLd process MysqLd_pID=`pIDof $libexecdir/MysqLd` # test if multiple pIDs exist pID_count=`echo $MysqLd_pID | wc -w` if test $pID_count -gt 1 ; then log_failure_msg "Multiple MysqL running but PID file Could not be found ($MysqLd_pID)" exit 5 elif test -z $MysqLd_pID ; then if test -f "$lock_file_path" ; then log_failure_msg "MysqL is not running,but lock file ($lock_file_path) exists" exit 2 fi log_failure_msg "MysqL is not running" exit 3 else log_failure_msg "MysqL is running but PID file Could not be found" exit 4 fi fi ;; *) # usage basename=`basename """` echo "Usage: $basename {start|stop|restart|reload|force-reload|status} [ MysqL server options ]" exit 1 ;;esacexit 0var
lock
basedir=lock
其中,
basedir 指的二进制压缩包解压后所在的目录,譬如/usr/local/MysqL。
datadir 指的是数据目录
service_startup_timeout=900 定义MysqL服务启动的时间限制,如果在900s中没有启动成功,则该脚本会退出。
lockdir='/var/lock/subsys'
关于/var/lock/subsys,网上的解释如下,后续会用到。
总的来说,系统关闭的过程(发出关闭信号,调用服务自身的进程)中会检查/subsys下没有相应的选项。在系统关闭的时候,会像杀死普通进程一样杀死这个服务。通过察看/subsys下相应的服务。很多程序需要判断是否当前已经有一个实例在运行,这个目录就是让程序判断是否有实例运行的标志,比如说xinetd,如果存在这个文件,表示已经有xinetd在运行了,否则就是没有,当然程序里面还要有相应的判断措施来真正确定是否有实例在运行。通常与该目录配套的还有/subsys下的文件,逐一关闭每个服务,如果某一运行的服务在/var/判断basedir和datadir/etc/rc.d/init.d下的脚本,可以发现每个服务自己 *** 纵时都会去查看/# Set some defaultsMysqLd_pID_file_path/fi/run目录,用来存放对应实例的PID,如果你写脚本的话,会发现这2个目录结合起来可以很方便的判断出许多服务是否在运行,运行的相关信息等等。
定义log_success_msg()和log_failure_msg()函数
## Use LSB init script functions =fi
其中,
MysqLd_pID_file_path 指定pID文件的路径
-z string 判断字符串是否为空
如果basedir没有显示设置,则默认为/usr/local/MysqL,这也是为什么很多MysqL安装教程都推荐将MysqL相关文件放到/usr/local/MysqL下。
如果datadir没有显示设置,则默认为$basedir/data。
传递参数
首先,判断/lib/lsb/init-functions文件是否存在,如果存在,则使定义在init-functions文件中的所有shell函数在当前脚本中生效。
如果没有,则定义两个函数,一个用于打印成功日志,一个是打印错误日志。
在RHCS 6.7中,该文件并不存在,已被/etc/init.d/functions所替代。
esac解析配置文件中的参数
parse_server_arguments() {
将第一个参数传递给mode,剩下的参数传递给other_args
PATH=}
判断my_print_defaults的位置
这个函数在脚本后面会涉及到。
主要涉及如下参数:--basedir,--datadir,--pID-file,--service-startup-timeout。
查找默认的配置文件## Read defaults
解析配置文件中的参数
首先,它判断当前路径下的bin目录中是否存在该可执行文件,如果不存在,则再判断$bindir(通常指的是$basedir/bin)目录下是否存在。
如果还是没有,则会判断/etc/my.cnf是否存在并且可读,如果是,则判断该配置文件中是否指定了basedir参数,
如果指定了,则取出该参数的值,并判断该值对应的目录中是否存在bin/my_print_defaults可执行文件
最后一步,如果在上述目录中实在没发现my_print_defaults文件,
索性就将print_defaults设置为"my_print_defaults",寄希望于该命令在当前的PATH环境中。
# Get arguments from the my.cnfparse_server_arguments `$print_defaults $extra_args MysqLd server MysqL_server MysqL.server`-r file 如果文件可读,则为真
设置pID file的路径## Set pIDmy_print_defaults的用法如下:
my_print_defaults --defaults-file=example.cnf clIEnt MysqL
即读取配置文件中,clIEnt和MysqL部分的参数配置,
具体在本脚本中,是读取MysqLd,server,MysqL_server,MysqL.server四个部分的配置参数。
) ;; * ) MysqLd_pID_file_path="$datadir/$MysqLd_pID_file_path" ;; esacfi
服务脚本start选项
-z string 判断字符串是否为空
如果--pID-file没有在读取到的配置文件中设置或者脚本刚开始的MysqLd_pID_file_path参数没有设置,
则pID file默认设置在datadir下,以主机名.pID命名。
如果该参数设置了,还需要进一步判断
如果该参数中带有斜杠,则代表给定的值带有路径,可直接使用。
如果该参数中没带路径,则代表给定的值只是pID的文件名,可将其设在datadir下。
$modein
start
首先,切换到$basedir中
其次,判断$basedir/bin中的MysqLd_safe是否是可执行文件,如果是,则启动MysqLd实例,如果不是,则报错退出。
那么,启动流程又是如何实现的呢?
首先,执行$bindir/MysqLd_safe --datadir="$datadir" --pID-file="$MysqLd_pID_file_path" $other_args >/dev/null 2>&1 &命令,启动MysqLd实例。
注意到没有,MysqLd_safe其实是在basedir中执行的,包括MysqL初始化脚本MysqL_install_db,也建议在basedir中执行,具体可参考:
分析MariaDB初始化脚本mysql_install_db
然后通过wait_for_pID函数进行判断,具体可见下文对于wait_for_pID函数的分析
判断完毕后,
查看$lockdir目录是否可写,可写的话,则在目录上创建一个文件。
) # Start daemon # Safeguard (relative paths,core dumps..) cd $basedir Starting MysqL "ifMysqLd_safe # Give extra arguments to MysqLd with the my.cnf . This script # may be overwritten at next upgrade. $bindir test -x $bindir/"file"/MysqLd_safe --datadir=null --pID-2=1 $other_args >/dev/ wait_for_pID created $!>&" & # Make lock SuSE w; return_value=$?$lockdirthentouch $lock_file_path exit $return_value log_failure_msg Couldn't find MysqL server ($bindir/MysqLd_safe) ;;wait_for_pID函数fii
`
在利用MysqLd_safe启动MysqL实例后,会调用该参数
wait_for_pID created "$!" "$MysqLd_pID_file_path"; return_value=$?
其中$!在shell中用于获取最后运行的后台Process的PID,具体在本例中,是MysqLd_safe进程的pID。
因为第一个参数是created,所以会执行test -s "$pID_file_path" && i='' && break命令。
-s file 如果文件的长度不为零,则为真
该命令的意思是如果pID文件存在,则将变量i设置为空,并退出while循环。
然后执行如下判断,
1
如果$i为空,则打印成功日志,并退出脚本,很显然,在pID文件存在的情况下,会将变量i设置为空。
再来看看pID文件不存在的情况
首先,会判断$pID是否不为空(即if test -n "$pID")
如果不为空,则代表在执行完MysqLd_safe后,已经捕捉到了该进程的pID。
在这种情况下,进一步通过kill -0 "$pID"确认该进程是否存在。
kill -0就是不发送任何信号,但是系统会进行错误检查,所以经常用来检查一个进程是否存在,当进程不存在时,kill -0 pID会返回错误
如果该进程存在,则不执行任何 *** 作,直接跳到如下 *** 作
wait_for_pID () { verb=`服务脚本stop选项杀死进程最安全的方法是单纯使用kill命令,不加修饰符,不带标志。 标准的kill命令通常会终止有问题的进程,并把进程的资源释放给系统。然而,如果进程启动了子进程,只杀死父进程,子进程仍在运行,因此仍消耗资源。为了防止这些所谓的“僵尸进程”,应确保在杀死父进程之前,先杀死其所有的子进程。将变量i加1,并sleep 1s。
然后,继续while循环,之所以这样做,是考虑到MysqLd_safe已经执行,但是MysqLd实例还在启动过程中,还没创建好pID文件。
一直到$1达到$service_startup_timeout定义的时长。
如果在while循环的过程中,通过kill -0 "$pID"判断到进程已经不存在了,
则会再判断一次,如果这次判断的结果依旧是pID file不存在,且进程不存在,则会执行
log_failure_msg "The server quit without updating PID file ($pID_file_path)."
这就是大名鼎鼎的“The server quit without updating PID file”的由来。
stop=) # Stop daemon. We use a signal here to avoID having to kNow the # root password.首先,判断pID文件的长度是否不为零。
-s file 如果文件的长度不为零,则为真
此时,会通过pID文件获取MysqLd进程的pID,注意,不是MysqLd_safe进程的pID
然后,判断MysqLd进程是否在正常运行,
如果是,则通过kill $MysqLd_pID的方式来关闭MysqLd进程
if
然后,调用wait_for_pID函数进行判断,其实,wait_for_pID函数中设置avoID_race_condition变量的目的是为了stop选项,确实有可能出现,MysqLd是在检查pID file之后,检查进程是否存活之前退出的。
如果MysqLd进程没有正常运行,在会打印“MysqL server process #$MysqLd_pID is not running!”信息,并删除pID文件。
如果在执行stop的时候,判断pID文件的长度为0,则会打印"MysqL server PID file Could not be found!"信息。
所以,在pID文件不存在的情况下,通过服务脚本执行stop选项并不会关闭MysqLd进程,这个时候,就可通过kill $MysqLd_pID的方式来关闭MysqLd进程。
MysqLd_pIDcat` test -s if=`0 null) (Shutting down MysqL $MysqLd_pID "kill $MysqLd_pID # MysqLd should remove the pID file it. wait_for_pID removed $MysqLd_pID" when it exits,so log_failure_msg MysqL server process #$MysqLd_pID is not running!rm; return_value=$? # Delete lock ifrm exit $return_value MysqL server PID file Could not be found! test -f 服务脚本restart选项 -f restart) # Stop the service and regardless of whether it was # running or not,start it again. if首先,先执行stop *** 作,如果stop *** 作成功的话,则继续执行start *** 作。
如果stop *** 作失败的话,则会输出"Failed to stop running server,so refusing to try to start."信息,并退出脚本。
0 $ start $other_args $Failed to stop running server,so refusing to try to start. stop $other_args; exit 1服务脚本reload选项killHUP pID pID 是进程标识。如果想要更改配置而不需停止并重新启动服务,请使用该命令。在对配置文件作必要的更改后,发出该命令以动态更新服务配置。根据约定,当您发送一个挂起信号(信号 1首先,判断pID文件的长度是否为0,如果不为0,则将该文件中的值设置为MysqLd_pID变量的值。
然后对该进程执行kill -HUP *** 作。
reload -'force-reload 或 HUP)时,大多数服务器进程(所有常用的进程)都会进行复位 *** 作并重新加载它们的配置文件。如果pID文件的长度为0,则输出"MysqL PID file Could not be found!"。
) read MysqLd_pID |killReloading service MysqLMysqL PID file Could not be found!< 服务脚本status选项 -HUP $MysqLd_pID && log_success_msg status) # First,check to see if首先,判断pID文件长度是否为0,如果不是,则读取该文件中的值,并判断pID对应的进程是否运行正常,
如果运行正常,则输出"MysqL running"
如果不正常,则输出"MysqL is not running,but PID file exists"
如果pID文件的长度为0,则试图通过MysqLd的启动命令来获取其pID,
这个时候,可能存在一个MysqLd程序启动了多个实例,这会导致pID_count=`echo $MysqLd_pID | wc -w`大于1。
这个时候,会输出"Multiple MysqL running but PID file Could not be found"信息,并退出脚本。
如果MysqLd_pID为空,则会继续判断"$lock_file_path"是否存在,如果存在,
则会输出"MysqL is not running,but lock file ($lock_file_path) exists"信息。
如果"$lock_file_path"不存在,则会输出"MysqL is not running"信息。
如果MysqLd_pID等于1,则会输出"MysqL is running but PID file Could not be found"信息。
exists read MysqLd_pID null pID log_success_msg MysqL running ($MysqLd_pID)< 0 ; MysqL is not running,but PID file exists1 # Try to appropriate MysqLd process MysqLd_pIDpIDof MysqLd` # test multiple pIDs exist pID_count=`echo $libexecdir/wcw=`if $MysqLd_pID | 1 -Multiple MysqL running but PID file Could not be found ($MysqLd_pID)5 test $pID_count -gt elif ; then log_failure_msg exit test -z $MysqLd_pID ; 2 log_failure_msg MysqL is not running3 MysqL is running but PID file Could not be found4服务脚本其它选项 ) # usage basename basename如果脚本的第一个参数不是上述几个选项,则会输出Usage信息。
*Usage: $basename {start|stop|restart|reload|force-reload|status} [ MysqL server options ] ;;=`总结 5.6235
至此,MysqL的服务脚本分析完毕~
x86_64datadir
在通过服务脚本启动MysqL的过程中,报“The server quit without updating PID file”错误,有两个条件
首先,pID文件不存在
其次,通过kill -0 $pID检查到进程并不存在
这个时候,只能通过MysqL数据库的错误日志来定位。
服务脚本如果不做任何调整的话,默认的basedir是/usr/local/MysqL,datadir是/usr/local/MysqL/data
如果自己的MysqL服务不是默认路径,
则需要在该脚本中显式设置
经测试,需设置如下几处:
1. 设置basedir和添加conf变量
其中,conf指的是MysqLd的配置文件,建议配置文件中显式指定basedir和datadir的值。
在这里,datadir可不设置,因为datadir可通过配置文件来获取。
但是basedir必须要指定,因为要首先根据basedir来判断my_print_deefauts命令
basedir=/usr/local/MysqL-advanced-conf.5-linux-glibc2. -e $basedir/my.cnf.bak-extra_args= -c $conf=/usr/local/MysqL-advanced-"-x86_64/my_3308.cnf
2. 第256行,添加extra_args=" -c $conf"
extra_args=1$conf="1
3. 修改285行MysqLd_safe的启动参数
将
$bindir/MysqLd_safe --datadir=参考 &
修改为,
$bindir/MysqLd_safe --defaults- --datadir=&主要是添加了--defaults-file选项
1. shell中test命令用法详解
2. shell中$0,$?,$!等的特殊用法
总结以上是内存溢出为你收集整理的深度解析MySQL启动时报“The server quit without updating PID file”错误的原因全部内容,希望文章能够帮你解决深度解析MySQL启动时报“The server quit without updating PID file”错误的原因所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)