如何隐藏GRUB2菜单,除非按住Shift键:如何实现这一点?

我有一个Ubuntu - Windows 7双启动设置,我希望我的笔记本电脑在启动后按下Shift键并从Grub2菜单中选择Ubuntu之前会自动启动Windows 7。
我研究了Grub2和/etc/default/grub选项,并尝试使用GRUB_TIMEOUT和GRUB_HIDDEN_TIMEOUT值的组合进行操作,但没有成功。我尝试将GRUB_HIDDEN_TIMEOUT设置得比GRUB_TIMEOUT更高,以为两者同时开始倒计时,但不是这样-GRUB_TIMEOUT只有在另一个完成后才开始。
这种行为是否可以实现?
如果可以,怎么做?

1我所做的方式是将超时设置为0(使用Grub自定义工具进行设置)。除非我在BIOS后按住Shift键,否则不会显示GRUB2菜单,这时候菜单才会出现。sudo add-apt-repository ppa:danielrichter2007/grub-customizer sudo apt-get update sudo apt-get install grub-customizer - Ryan McClure
据我了解,这是某些安装的默认行为。 - RobotHumans
@RyanMcClure 我尝试将超时时间手动设置为0,但是按住右Shift键并没有弹出Grub2菜单。 - Bez Hermoso
13个回答

我觉得我找到了一个更简单的解决方案。 修改文件/etc/default/grub中的以下行:

GRUB_HIDDEN_TIMEOUT=0.0

GRUB_TIMEOUT=0.0

当然,我们最终运行了update-grub。 在我的电脑上它有效果。

7算了,这样做并不能让我进入引导菜单,而且按下Shift键也无效。 - zorkerz

这是针对Ubuntu 16.04的。
我通过结合上面找到的一些答案来解决了这个问题,所以不需要修改30_os-prober,避免了将来更新grub软件包时的合并问题...
希望能有所帮助。
在/etc/default/grub中更改了一些变量以反映我想要的内容;
GRUB_DEFAULT=4
GRUB_HIDDEN_TIMEOUT=5
GRUB_HIDDEN_TIMEOUT_QUIET=true
GRUB_TIMEOUT=0
GRUB_DISTRIBUTOR=`lsb_release -i -s 2> /dev/null || echo Debian`
GRUB_CMDLINE_LINUX_DEFAULT="quiet splash"
GRUB_CMDLINE_LINUX=""

GRUB_FORCE_HIDDEN_MENU="true"
export GRUB_FORCE_HIDDEN_MENU

然后创建一个新文件;
sudo touch /etc/grub.d/50_hidemenu
sudo chmod +x /etc/grub.d/50_hidemenu

填充它与这个内容;
#!/bin/sh
cat <<EOF
if [ "x\${timeout}" != "x-1" ]; then
  if keystatus; then
    if keystatus --shift; then
      set timeout=-1
    else
      set timeout=0
    fi
  else
    if sleep --interruptible ${GRUB_HIDDEN_TIMEOUT} ; then
      set timeout=0
    fi
  fi
fi
EOF

并且更新了grub;

sudo update-grub

干杯。

对我来说不起作用。 - Menasheh
我只在16.04版本中进行了测试。你使用的是不同的版本吗? - tux1c
可能是16.10号。你说得对,肯定应该提到这个。 - Menasheh
适用于Debian Squeeze。 - Jocelyn delalande

在Kubuntu 14.04中,如果你设置GRUB_TIMEOUT=0,这个方法将不起作用,奇怪的是,应用了这个设置并重启后,grub菜单仍然可见,并持续显示10秒。
原因在于,在另一个grub脚本中有一段代码,如果grub_timeout的值为0,则将其设置为10。
所以这是我让它正常工作的方法:
首先备份grub30_osprober两个文件!!!当出现问题时,可以将原始文件恢复回来。
接下来,在/etc/default/grub文件中:
GRUB_DEFAULT=4    #set this value to the OS you want to boot
GRUB_HIDDEN_TIMEOUT=0

GRUB_TIMEOUT=0

/etc/grub.d/30_os-prober中, 有一段代码,如果/etc/default/grub中的超时时间设置为零,则将grub的超时时间设置为10秒。我不知道为什么开发人员把它放在那里,但它阻止了隐藏grub菜单。
set timeout_style=menu

if [ "\${timeout}" = 0 ]; then    
  set timeout=10  # I changed this from 10 to 0

在进行这些更改之后,运行sudo update-grub以生成新的grub引导顺序。
我将所有我的Linux修复/修改记录在https://sites.google.com/site/marcshomesite/Home/linux-tuning上。

我希望你在GRUB_TIMEOUT= 0中没有在=后面加了空格。 - muru

要在不必从GRUB菜单中选择的情况下自动启动Windows,我们需要编辑/etc/default/grub文件,并设置类似于以下数值的值:
GRUB_DEFAULT= <number of default entry starting from 0>
GRUB_TIMEOUT= 0 # time in seconds to boot default
# GRUB_HIDDEN_TIMEOUT=0 # warning: activating this may lock you out from GRUB menu

在您编辑这些文件之前,始终备份是一个好主意。在进行更改后,我们需要激活它们。
sudo update-grub

GRUB_DEFAULT的值取决于我们个人的系统。为了始终引导Windows,我们可以选择将条目的名称完全输入,就像在Grub菜单中显示的那样(例如"Microsoft Windows XP Professional"),而不是输入条目编号。
重要提示:在某些机器上,按住左侧Shift键无法显示GRUB菜单(参见bug #425979)。因此,建议首先使用Ubuntu(而不是Windows)作为默认操作系统进行测试。这样,如果受到影响,您就可以轻松地撤消GRUB设置。

这会不会导致Ubuntu无法启动,或者在我想要选择Ubuntu时是否有足够的时间来拦截启动? - Bez Hermoso
1@Shasteriskt,你应该将超时时间设置为足够长的值,以便按下一个键选择另一个启动操作系统。 - Takkat
我将超时设置为0并按住Shift键,但菜单仍然没有显示出来。(幸好我将它设置为启动到Ubuntu,这样我就可以将设置编辑回原来的状态。) - Bez Hermoso
简而言之,我需要的行为是无法实现的吗? - Bez Hermoso
@Shasteriskt:不幸的是,在某些机器上,按住左SHIFT键启动时无法打开GRUB菜单(例如此错误)。如果ESC或SHIFT+ESC可以使用的话,你可能会有好运。 - Takkat
不好意思,我有点搞不清楚报告的错误是Grub2的错误还是一些BIOS的错误。在我的机器上,按ESC或Shift+ESC可以显示启动设备顺序。唉... - Bez Hermoso
这很可能是与USB相关的BIOS问题。很好,ESC键能帮到你那里 :) - Takkat
呵呵,不完全是 - 它会显示启动设备的选项,而不是Grub2菜单。无论如何,我将超时设置为1,这已经足够满足我的需求了。谢谢。 - Bez Hermoso
这是16.04版本最干净的解决方案。 - Taylor R

我也试过同样的事情,在Ubuntu论坛上找到了很大的帮助,尤其是在这个this主题中。
假设你已经正确地处理了/etc/default/grub文件,将GRUB_TIMEOUTGRUB_HIDDEN_TIMEOUT分别设置为0和5,并且甚至使用GRUB_DEFAULT指定默认启动镜像,下面是使它正常工作的步骤。
  1. 将以下内容添加到/etc/default/grub文件的末尾:

    GRUB_FORCE_HIDDEN_MENU="true"
    export GRUB_FORCE_HIDDEN_MENU
    
  2. 备份你的/etc/grub.d/30_os-prober文件,并用此处找到的版本进行覆盖。

  3. 测试一下!按下SHIFT键显示菜单。

这个问题与多操作系统设置有关,编辑脚本是纠正的一种方式之一。 30_os-prober:
#! /bin/sh
set -e

# grub-mkconfig helper script.
# Copyright (C) 2006,2007,2008,2009  Free Software Foundation, Inc.
#
# GRUB is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# GRUB is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with GRUB.  If not, see <http://www.gnu.org/licenses/>.

prefix="/usr"
exec_prefix="${prefix}"
datarootdir="${prefix}/share"

. "${datarootdir}/grub/grub-mkconfig_lib"

found_other_os=

make_timeout () {

  if [ "x${found_other_os}" = "x" ] || [ "x${GRUB_FORCE_HIDDEN_MENU}" = "xtrue" ] ; then 
    if [ "x${1}" != "x" ] ; then
      if [ "x${GRUB_HIDDEN_TIMEOUT_QUIET}" = "xtrue" ] ; then
    verbose=
      else
    verbose=" --verbose"
      fi

      if [ "x${1}" = "x0" ] ; then
    cat <<EOF
if [ "x\${timeout}" != "x-1" ]; then
  if keystatus; then
    if keystatus --shift; then
      set timeout=-1
    else
      set timeout=0
    fi
  else
    if sleep$verbose --interruptible 3 ; then
      set timeout=0
    fi
  fi
fi
EOF
      else
    cat << EOF
if [ "x\${timeout}" != "x-1" ]; then
  if sleep$verbose --interruptible ${GRUB_HIDDEN_TIMEOUT} ; then
    set timeout=0
  fi
fi
EOF
      fi
    fi
  fi
}

adjust_timeout () {
  if [ "x$GRUB_BUTTON_CMOS_ADDRESS" != "x" ]; then
    cat <<EOF
if cmostest $GRUB_BUTTON_CMOS_ADDRESS ; then
EOF
    make_timeout "${GRUB_HIDDEN_TIMEOUT_BUTTON}" "${GRUB_TIMEOUT_BUTTON}"
    echo else
    make_timeout "${GRUB_HIDDEN_TIMEOUT}" "${GRUB_TIMEOUT}"
    echo fi
  else
    make_timeout "${GRUB_HIDDEN_TIMEOUT}" "${GRUB_TIMEOUT}"
  fi
}

if [ "x${GRUB_DISABLE_OS_PROBER}" = "xtrue" ]; then
  adjust_timeout
  exit 0
fi

if [ -z "`which os-prober 2> /dev/null`" -o -z "`which linux-boot-prober 2> /dev/null`" ] ; then
  # missing os-prober and/or linux-boot-prober
  adjust_timeout
  exit 0
fi

OSPROBED="`os-prober | tr ' ' '^' | paste -s -d ' '`"
if [ -z "${OSPROBED}" ] ; then
  # empty os-prober output, nothing doing
  adjust_timeout
  exit 0
fi

osx_entry() {
    found_other_os=1
        cat << EOF
menuentry "${LONGNAME} (${2}-bit) (on ${DEVICE})" --class osx --class darwin --class os {
EOF
    save_default_entry | sed -e "s/^/\t/"
    prepare_grub_to_access_device ${DEVICE} | sed -e "s/^/\t/"
    cat << EOF
        load_video
        set do_resume=0
        if [ /var/vm/sleepimage -nt10 / ]; then
           if xnu_resume /var/vm/sleepimage; then
             set do_resume=1
           fi
        fi
        if [ \$do_resume = 0 ]; then
           xnu_uuid ${OSXUUID} uuid
           if [ -f /Extra/DSDT.aml ]; then
              acpi -e /Extra/DSDT.aml
           fi
           $1 /mach_kernel boot-uuid=\${uuid} rd=*uuid
           if [ /System/Library/Extensions.mkext -nt /System/Library/Extensions ]; then
              xnu_mkext /System/Library/Extensions.mkext
           else
              xnu_kextdir /System/Library/Extensions
           fi
           if [ -f /Extra/Extensions.mkext ]; then
              xnu_mkext /Extra/Extensions.mkext
           fi
           if [ -d /Extra/Extensions ]; then
              xnu_kextdir /Extra/Extensions
           fi
           if [ -f /Extra/devprop.bin ]; then
              xnu_devprop_load /Extra/devprop.bin
           fi
           if [ -f /Extra/splash.jpg ]; then
              insmod jpeg
              xnu_splash /Extra/splash.jpg
           fi
           if [ -f /Extra/splash.png ]; then
              insmod png
              xnu_splash /Extra/splash.png
           fi
           if [ -f /Extra/splash.tga ]; then
              insmod tga
              xnu_splash /Extra/splash.tga
           fi
        fi
}
EOF
}

wubi=

for OS in ${OSPROBED} ; do
  DEVICE="`echo ${OS} | cut -d ':' -f 1`"
  LONGNAME="`echo ${OS} | cut -d ':' -f 2 | tr '^' ' '`"
  LABEL="`echo ${OS} | cut -d ':' -f 3 | tr '^' ' '`"
  BOOT="`echo ${OS} | cut -d ':' -f 4`"

  if [ -z "${LONGNAME}" ] ; then
    LONGNAME="${LABEL}"
  fi

  echo "Found ${LONGNAME} on ${DEVICE}" >&2

  case ${BOOT} in
    chain)

      case ${LONGNAME} in
    Windows*)
      if [ -z "$wubi" ]; then
        if [ -x /usr/share/lupin-support/grub-mkimage ] && \
           /usr/share/lupin-support/grub-mkimage --test; then
          wubi=yes
        else
          wubi=no
        fi
      fi
      if [ "$wubi" = yes ]; then
        echo "Skipping ${LONGNAME} on Wubi system" >&2
        continue
      fi
      ;;
      esac

      found_other_os=1
      cat << EOF
menuentry "${LONGNAME} (on ${DEVICE})" --class windows --class os {
EOF
      save_default_entry | sed -e "s/^/\t/"
      prepare_grub_to_access_device ${DEVICE} | sed -e "s/^/\t/"

      case ${LONGNAME} in
    Windows\ Vista*|Windows\ 7*|Windows\ Server\ 2008*)
    ;;
    *)
      cat << EOF
    drivemap -s (hd0) \${root}
EOF
    ;;
      esac

      cat <<EOF
    chainloader +1
}
EOF
    ;;
    linux)
      LINUXPROBED="`linux-boot-prober ${DEVICE} 2> /dev/null | tr ' ' '^' | paste -s -d ' '`"
      prepare_boot_cache=

      for LINUX in ${LINUXPROBED} ; do
        LROOT="`echo ${LINUX} | cut -d ':' -f 1`"
        LBOOT="`echo ${LINUX} | cut -d ':' -f 2`"
        LLABEL="`echo ${LINUX} | cut -d ':' -f 3 | tr '^' ' '`"
        LKERNEL="`echo ${LINUX} | cut -d ':' -f 4`"
        LINITRD="`echo ${LINUX} | cut -d ':' -f 5`"
        LPARAMS="`echo ${LINUX} | cut -d ':' -f 6- | tr '^' ' '`"

        if [ -z "${LLABEL}" ] ; then
          LLABEL="${LONGNAME}"
        fi

    if [ "${LROOT}" != "${LBOOT}" ]; then
      LKERNEL="${LKERNEL#/boot}"
      LINITRD="${LINITRD#/boot}"
    fi

    if [ -z "${prepare_boot_cache}" ]; then
      prepare_boot_cache="$(prepare_grub_to_access_device ${LBOOT} | sed -e "s/^/\t/")"
      [ "${prepare_boot_cache}" ] || continue
    fi
    found_other_os=1
        cat << EOF
menuentry "${LLABEL} (on ${DEVICE})" --class gnu-linux --class gnu --class os {
EOF
    save_default_entry | sed -e "s/^/\t/"
    printf '%s\n' "${prepare_boot_cache}"
    cat <<  EOF
    linux ${LKERNEL} ${LPARAMS}
EOF
        if [ -n "${LINITRD}" ] ; then
          cat << EOF
    initrd ${LINITRD}
EOF
        fi
        cat << EOF
}
EOF
      done
    ;;
    macosx)
      OSXUUID="`grub-probe --target=fs_uuid --device ${DEVICE} 2> /dev/null`"
      osx_entry xnu_kernel 32
      osx_entry xnu_kernel64 64
    ;;
    hurd)
      found_other_os=1
      cat << EOF
menuentry "${LONGNAME} (on ${DEVICE})" --class hurd --class gnu --class os {
EOF
      save_default_entry | sed -e "s/^/\t/"
      prepare_grub_to_access_device ${DEVICE} | sed -e "s/^/\t/"
      grub_device="`${grub_probe} --device ${DEVICE} --target=drive`"
      mach_device="`echo "${grub_device}" | sed -e 's/(\(hd.*\),msdos\(.*\))/\1s\2/'`"
      grub_fs="`${grub_probe} --device ${DEVICE} --target=fs`"
      case "${grub_fs}" in
    *fs)    hurd_fs="${grub_fs}" ;;
    *)  hurd_fs="${grub_fs}fs" ;;
      esac
      cat << EOF
    multiboot /boot/gnumach.gz root=device:${mach_device}
    module /hurd/${hurd_fs}.static ${hurd_fs} --readonly \\
            --multiboot-command-line='\${kernel-command-line}' \\
            --host-priv-port='\${host-port}' \\
            --device-master-port='\${device-port}' \\
            --exec-server-task='\${exec-task}' -T typed '\${root}' \\
            '\$(task-create)' '\$(task-resume)'
    module /lib/ld.so.1 exec /hurd/exec '\$(exec-task=task-create)'
}
EOF
    ;;
    *)
      echo "  ${LONGNAME} is not yet supported by grub-mkconfig." >&2
    ;;
  esac
done

adjust_timeout


    cat <<EOF
if [ "x\${timeout}" != "x-1" ]; then
  if keystatus; then
    if keystatus --shift; then
      set timeout=-1
    else
      set timeout=0
    fi
  else
    if sleep$verbose --interruptible 3 ; then
      set timeout=0
    fi
  fi
fi
EOF

1我尝试了你的方法,但是失败了。Ubuntu 14.04 LTS - Mayukh Nair
1我应该修改文件的哪些部分?我的14.04文件与这个文件有很多差异,所以我想我应该只做有针对性的更改。 - MakisH
@MayukhNair:我使用的是Ubuntu 14.04,这种方法有效。我按照以下步骤操作:
  1. 在/etc/default/grub中设置GRUB_DEFAULT=0
  2. 设置GRUB_HIDDEN_TIMEOUT=0(与Dariopnc所说的不同)
  3. 备份你的/etc/grub.d/30_os-prober文件,并将上述提到的30-os_prober文件复制粘贴进去
  4. 运行sudo update-grub2命令
- Pushpak Dagade
请注意,在更改/etc/default/grub文件后,需要使用sudo update-grub命令来更新grub。 - Wilf

在Ubuntu 16.04上,我做了以下更改。当然,您需要成为特权用户才能这样做。
1. 第一个更改是在`/etc/grub.d/30_os-prober`文件中: - 使用`sudo gedit /etc/grub.d/30_os-prober`打开它 - 将`quick_boot="1"`更改为`quick_boot="0"` - 保存文件
2. 然后,您可以将`/etc/default/grub`中的超时值设置为零或其他值: - 使用`sudo gedit /etc/default/grub`打开它 - 设置`GRUB_HIDDEN_TIMEOUT=0`和`GRUB_TIMEOUT=0` - 保存文件
3. 最后一步是更新grub: - 运行`sudo update-grub`
第一个文件的更改是因为默认设置是,如果发现其他操作系统(例如Windows 10),超时值始终设置为10秒,因此无法通过/etc/default/grub进行更改。这是通过一个名为adjust_timeout的函数实现的,该函数询问quick_boot是否设置并且是否存在其他操作系统。

终于有一个适用于16.04的东西可用了! - JLTD

安装 Grub 优化器。

sudo add-apt-repository ppa:danielrichter2007/grub-customizer

sudo apt-get update
sudo apt-get install grub-customizer
Grub Customizer现在将显示在“应用程序”>“系统工具”菜单中。
或者您可以从命令行打开它。

gksudo grub-customizer

image inside for grub options

enter image description here


3这个有什么功能不能通过编辑“etc/default/grub”来实现? - Bez Hermoso
3@Shasteriskt:这对于非英语母语者、不熟悉命令行的人或只想使用计算机而不想深入配置文件的人来说可能很有用。 - Flimm
为什么这个被点赞了?它似乎没有解决任何问题;我尝试过 grub-customizer(不是 'optimizer'),虽然它能工作,但不能帮助我在不按下 Shift 键的情况下隐藏 GRUB 菜单。 - Nickolai Leschov

我在单系统启动时遇到了同样的问题。如果GRUB_TIMEOUT不为零,无论GRUB_HIDDEN_TIMEOUT如何,菜单都会弹出来。如果GRUB_TIMEOUT为零,它会等待GRUB_HIDDEN_TIMEOUT秒,但是按下SHIFT或ESC键都不会显示菜单。
我通过在30_os-prober中的adjust_time()函数的sleep语句末尾添加一个else子句来解决了这个问题。现在,如果没有被打断,超时时间为零,系统将无菜单启动,但如果被打断,我将超时时间设置为10。
实际上,这似乎是相反的,因为如果未被打断,sleep应该返回0(假),如果被打断,则返回1(真),但对我来说似乎起作用了。

只是为了结束这个讨论:
我不建议任何人编辑"30_os-prober"文件!(代码在那里有其原因!)
解决问题的正确方法是禁用Windows安装中的快速启动,如 这里(适用于Win8)和 这里(适用于Win10)所述。
完成此操作后,您需要关机(而不是重新启动)您的计算机,以防止Windows创建在启动时加载以更快引导的文件。
现在,如果grub被正确配置,它应该能够正常工作,而无需编辑"30_os-prober"文件,因为上面发布的if条件中的"[ "$quick_boot" = 1 ]"检查返回0。

Shasteriskt,一个替代“SHIFT”解决方案的选择是创建一个脚本来执行一些移动操作。我的意思是创建一个名为grub.bak的文件,它是你的grub.cfg文件的精确副本,唯一的区别是将set timeout=0行更改为set timeout=10,例如。创建一个shell脚本,根据你的需求允许你重新启动。我创建了一个等效的Python脚本来满足这些要求,但当然需要使用sudo调用。我只是将它放在我的设置中的/usr/bin目录下,作为指向实际脚本本身的符号链接。注意:我在Arch上运行fluxbox,所以不确定这个脚本和Ubuntu之间是否有任何差异,但应该几乎没有修改。脚本如下:
#! /usr/bin/python
from shlex import split
from subprocess import call
from os import rename, chdir

chdir('/boot/grub/')
try:
    with open('grub.bak','r') as alternate:
        pass
    if raw_input('Auto-boot is set. Un-set?> ') == 'y':
        rename('grub.cfg','grub.back')
        rename('grub.bak','grub.cfg')

    if raw_input('Reboot system?> ') == 'y':
        call(split('shutdown -r now'))

except:
    if raw_input('Auto-boot is not set. Set?> ') == 'y':
        rename('grub.cfg','grub.bak')
        rename('grub.back','grub.cfg')

    if raw_input('Reboot system?> ') == 'y':
        call(split('shutdown -r now'))

脚本基本上只是确定您是否有菜单要加载(自动启动或不启动)。根据其结果,它会询问您是否想要打开或关闭它,然后询问您是否想要重新启动。相反地,如果您希望它以相反的方式工作(即在Windows中),并且您已经设置了Windows以访问Linux中的文件,如果您愿意,也可以在该环境中运行相同的脚本。希望这能帮到您。