Error[8]: Undefined offset: 482, File: /www/wwwroot/outofmemory.cn/tmp/plugin_ss_superseo_model_superseo.php, Line: 121
File: /www/wwwroot/outofmemory.cn/tmp/plugin_ss_superseo_model_superseo.php, Line: 473, decode(

概述很多童鞋在启动mysql的时候,碰到过这个错误, 首先,澄清一点,出现这个错误的前提是:通过服务脚本来启动mysql。通过mysqld_safe或mysqld启动mysql实例并不会报这个错误。 那么

很多童鞋在启动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 0
var

 

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.cnf 
parse_server_arguments `$print_defaults $extra_args MysqLd server MysqL_server MysqL.server`

-r file 如果文件可读,则为真

设置pID file的路径## Set pID 

my_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”错误的原因所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

)
File: /www/wwwroot/outofmemory.cn/tmp/route_read.php, Line: 126, InsideLink()
File: /www/wwwroot/outofmemory.cn/tmp/index.inc.php, Line: 165, include(/www/wwwroot/outofmemory.cn/tmp/route_read.php)
File: /www/wwwroot/outofmemory.cn/index.php, Line: 30, include(/www/wwwroot/outofmemory.cn/tmp/index.inc.php)
Error[8]: Undefined offset: 483, File: /www/wwwroot/outofmemory.cn/tmp/plugin_ss_superseo_model_superseo.php, Line: 121
File: /www/wwwroot/outofmemory.cn/tmp/plugin_ss_superseo_model_superseo.php, Line: 473, decode(

概述很多童鞋在启动mysql的时候,碰到过这个错误, 首先,澄清一点,出现这个错误的前提是:通过服务脚本来启动mysql。通过mysqld_safe或mysqld启动mysql实例并不会报这个错误。 那么

很多童鞋在启动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 0
var

 

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.cnf 
parse_server_arguments `$print_defaults $extra_args MysqLd server MysqL_server MysqL.server`

-r file 如果文件可读,则为真

设置pID file的路径## Set pID 

my_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”错误的原因所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

)
File: /www/wwwroot/outofmemory.cn/tmp/route_read.php, Line: 126, InsideLink()
File: /www/wwwroot/outofmemory.cn/tmp/index.inc.php, Line: 165, include(/www/wwwroot/outofmemory.cn/tmp/route_read.php)
File: /www/wwwroot/outofmemory.cn/index.php, Line: 30, include(/www/wwwroot/outofmemory.cn/tmp/index.inc.php)
Error[8]: Undefined offset: 484, File: /www/wwwroot/outofmemory.cn/tmp/plugin_ss_superseo_model_superseo.php, Line: 121
File: /www/wwwroot/outofmemory.cn/tmp/plugin_ss_superseo_model_superseo.php, Line: 473, decode(

概述很多童鞋在启动mysql的时候,碰到过这个错误, 首先,澄清一点,出现这个错误的前提是:通过服务脚本来启动mysql。通过mysqld_safe或mysqld启动mysql实例并不会报这个错误。 那么

很多童鞋在启动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 0
var

 

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.cnf 
parse_server_arguments `$print_defaults $extra_args MysqLd server MysqL_server MysqL.server`

-r file 如果文件可读,则为真

设置pID file的路径## Set pID 

my_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”错误的原因所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

)
File: /www/wwwroot/outofmemory.cn/tmp/route_read.php, Line: 126, InsideLink()
File: /www/wwwroot/outofmemory.cn/tmp/index.inc.php, Line: 165, include(/www/wwwroot/outofmemory.cn/tmp/route_read.php)
File: /www/wwwroot/outofmemory.cn/index.php, Line: 30, include(/www/wwwroot/outofmemory.cn/tmp/index.inc.php)
Error[8]: Undefined offset: 485, File: /www/wwwroot/outofmemory.cn/tmp/plugin_ss_superseo_model_superseo.php, Line: 121
File: /www/wwwroot/outofmemory.cn/tmp/plugin_ss_superseo_model_superseo.php, Line: 473, decode(

概述很多童鞋在启动mysql的时候,碰到过这个错误, 首先,澄清一点,出现这个错误的前提是:通过服务脚本来启动mysql。通过mysqld_safe或mysqld启动mysql实例并不会报这个错误。 那么

很多童鞋在启动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 0
var

 

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.cnf 
parse_server_arguments `$print_defaults $extra_args MysqLd server MysqL_server MysqL.server`

-r file 如果文件可读,则为真

设置pID file的路径## Set pID 

my_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”错误的原因所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

)
File: /www/wwwroot/outofmemory.cn/tmp/route_read.php, Line: 126, InsideLink()
File: /www/wwwroot/outofmemory.cn/tmp/index.inc.php, Line: 165, include(/www/wwwroot/outofmemory.cn/tmp/route_read.php)
File: /www/wwwroot/outofmemory.cn/index.php, Line: 30, include(/www/wwwroot/outofmemory.cn/tmp/index.inc.php)
深度解析MySQL启动时报“The server quit without updating PID file”错误的原因_mysql_内存溢出

深度解析MySQL启动时报“The server quit without updating PID file”错误的原因

深度解析MySQL启动时报“The server quit without updating PID file”错误的原因,第1张

概述很多童鞋在启动mysql的时候,碰到过这个错误, 首先,澄清一点,出现这个错误的前提是:通过服务脚本来启动mysql。通过mysqld_safe或mysqld启动mysql实例并不会报这个错误。 那么

很多童鞋在启动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 0
var

 

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.cnf 
parse_server_arguments `$print_defaults $extra_args MysqLd server MysqL_server MysqL.server`

-r file 如果文件可读,则为真

设置pID file的路径## Set pID 

my_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”错误的原因所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/sjk/1151576.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-31
下一篇 2022-05-31

发表评论

登录后才能评论

评论列表(0条)

保存