--- title: 为容器设置启动时要执行的命令及其入参 content_template: templates/task weight: 10 --- {{% capture overview %}} 本页将展示如何为 {{< glossary_tooltip term_id="pod" >}} 中的容器设置启动时要执行的命令及其入参。 {{% /capture %}} {{% capture prerequisites %}} {{< include "task-tutorial-prereqs.md" >}} {{< version-check >}} {{% /capture %}} {{% capture steps %}} ## 创建 Pod 时设置命令及入参 创建 Pod 时,可以为其下的容器设置启动时要执行的命令及其入参。如果要设置命令,就填写在配置文件的 `command` 字段下,如果要设置命令的入参,就填写在配置文件的 `args` 字段下。一旦 Pod 创建完成,该命令及其入参就无法再进行更改了。 如果在配置文件中设置了容器启动时要执行的命令及其入参,那么容器镜像中自带的命令与入参将会被覆盖而不再执行。如果配置文件中只是设置了入参,却没有设置其对应的命令,那么容器镜像中自带的命令会使用该新入参作为其执行时的入参。 {{< note >}} 在有些容器运行时中,`command` 字段对应 `entrypoint`,请参阅下面的 [注意](#注意)。 {{< /note >}} 本示例中,将创建一个只包含单个容器的 Pod。在 Pod 配置文件中设置了一个命令与两个入参: {{< codenew file="pods/commands.yaml" >}} 1. 基于 YAML 文件创建一个 Pod: ```shell kubectl apply -f https://k8s.io/examples/pods/commands.yaml ``` 2. 获取正在运行的 Pods: ```shell kubectl get pods ``` 查询结果显示在 command-demo 这个 Pod 下运行的容器已经启动完成。 3. 如果要获取容器启动时执行命令的输出结果,可以通过 Pod 的日志进行查看: ```shell kubectl logs command-demo ``` 日志中显示了 HOSTNAME 与 KUBERNETES_PORT 这两个环境变量的值: ``` command-demo tcp://10.3.240.1:443 ``` ## 使用环境变量来设置入参 在上面的示例中,我们直接将一串字符作为命令的入参。除此之外,我们还可以将环境变量作为命令的入参。 ```yaml env: - name: MESSAGE value: "hello world" command: ["/bin/echo"] args: ["$(MESSAGE)"] ``` 这意味着你可以将那些用来设置环境变量的方法应用于设置命令的入参,其中包括了 [ConfigMaps](/docs/tasks/configure-pod-container/configure-pod-configmap/) 与 [Secrets](/docs/concepts/configuration/secret/)。 {{< note >}} 环境变量需要加上括号,类似于 `"$(VAR)"`。这是在 `command` 或 `args` 字段使用变量的格式要求。 {{< /note >}} ## 通过 shell 来执行命令 有时候,您需要在 shell 脚本中运行命令。 例如,您要执行的命令可能由多个命令组合而成,或者它就是一个 shell 脚本。这时,就可以通过如下方式在 shell 中执行命令: ```shell command: ["/bin/sh"] args: ["-c", "while true; do echo hello; sleep 10;done"] ``` ## 注意 下表给出了 Docker 与 Kubernetes 中对应的字段名称。 | 描述 | Docker 字段名称 | Kubernetes 字段名称 | |------------------|--------------------|-----------------------| | 容器执行的命令 | Entrypoint | command | | 传给命令的参数 | Cmd | args | 如果要覆盖默认的 Entrypoint 与 Cmd,需要遵循如下规则: * 如果在容器配置中没有设置 `command` 或者 `args`,那么将使用 Docker 镜像自带的命令及其入参。 * 如果在容器配置中只设置了 `command` 但是没有设置 `args`,那么容器启动时只会执行该命令,Docker 镜像中自带的命令及其入参会被忽略。 * 如果在容器配置中只设置了 `args`,那么 Docker 镜像中自带的命令会使用该新入参作为其执行时的入参。 * 如果在容器配置中同时设置了 `command` 与 `args`,那么 Docker 镜像中自带的命令及其入参会被忽略。容器启动时只会执行配置中设置的命令,并使用配置中设置的入参作为命令的入参。 下表涵盖了各类设置场景: | 镜像 Entrypoint | 镜像 Cmd | 容器 command | 容器 args | 命令执行 | |--------------------|------------------|---------------------|--------------------|------------------| | `[/ep-1]` | `[foo bar]` | <not set> | <not set> | `[ep-1 foo bar]` | | `[/ep-1]` | `[foo bar]` | `[/ep-2]` | <not set> | `[ep-2]` | | `[/ep-1]` | `[foo bar]` | <not set> | `[zoo boo]` | `[ep-1 zoo boo]` | | `[/ep-1]` | `[foo bar]` | `[/ep-2]` | `[zoo boo]` | `[ep-2 zoo boo]` | {{% /capture %}} {{% capture whatsnext %}} * 了解更多 [pod 和容器的配置](/docs/tasks/)。 * 了解更多 [在容器中运行命令](/docs/tasks/debug-application-cluster/get-shell-running-container/)。 * 请参阅 [有关容器的文档](/docs/reference/generated/kubernetes-api/{{< param "version" >}}/#container-v1-core)。 {{% /capture %}}