Assign Memory Resources to Containers and Pods

    You need to have a Kubernetes cluster, and the kubectl command-line tool must be configured to communicate with your cluster. It is recommended to run this tutorial on a cluster with at least two nodes that are not acting as control plane hosts. If you do not already have a cluster, you can create one by using minikube or you can use one of these Kubernetes playgrounds:

    To check the version, enter .

    Each node in your cluster must have at least 300 MiB of memory.

    A few of the steps on this page require you to run the service in your cluster. If you have the metrics-server running, you can skip those steps.

    If you are running Minikube, run the following command to enable the metrics-server:

    To see whether the metrics-server is running, or another provider of the resource metrics API (metrics.k8s.io), run the following command:

    1. kubectl get apiservices

    If the resource metrics API is available, the output includes a reference to metrics.k8s.io.

    1. NAME
    2. v1beta1.metrics.k8s.io

    Create a namespace

    Create a namespace so that the resources you create in this exercise are isolated from the rest of your cluster.

    1. kubectl create namespace mem-example

    Specify a memory request and a memory limit

    To specify a memory request for a Container, include the resources:requests field in the Container’s resource manifest. To specify a memory limit, include resources:limits.

    In this exercise, you create a Pod that has one Container. The Container has a memory request of 100 MiB and a memory limit of 200 MiB. Here’s the configuration file for the Pod:

    pods/resource/memory-request-limit.yaml

    1. apiVersion: v1
    2. kind: Pod
    3. metadata:
    4. name: memory-demo
    5. namespace: mem-example
    6. spec:
    7. containers:
    8. - name: memory-demo-ctr
    9. image: polinux/stress
    10. resources:
    11. limits:
    12. memory: "200Mi"
    13. requests:
    14. memory: "100Mi"
    15. command: ["stress"]
    16. args: ["--vm", "1", "--vm-bytes", "150M", "--vm-hang", "1"]

    The args section in the configuration file provides arguments for the Container when it starts. The "--vm-bytes", "150M" arguments tell the Container to attempt to allocate 150 MiB of memory.

    Create the Pod:

    1. kubectl apply -f https://k8s.io/examples/pods/resource/memory-request-limit.yaml --namespace=mem-example

    Verify that the Pod Container is running:

    1. kubectl get pod memory-demo --namespace=mem-example

    View detailed information about the Pod:

    1. kubectl get pod memory-demo --output=yaml --namespace=mem-example

    The output shows that the one Container in the Pod has a memory request of 100 MiB and a memory limit of 200 MiB.

    1. ...
    2. resources:
    3. limits:
    4. memory: 200Mi
    5. requests:

    Run kubectl top to fetch the metrics for the pod:

    1. kubectl top pod memory-demo --namespace=mem-example

    The output shows that the Pod is using about 162,900,000 bytes of memory, which is about 150 MiB. This is greater than the Pod’s 100 MiB request, but within the Pod’s 200 MiB limit.

    1. NAME CPU(cores) MEMORY(bytes)
    2. memory-demo <something> 162856960

    Delete your Pod:

    In this exercise, you create a Pod that attempts to allocate more memory than its limit. Here is the configuration file for a Pod that has one Container with a memory request of 50 MiB and a memory limit of 100 MiB:

    Assign Memory Resources to Containers and Pods - 图2

    1. apiVersion: v1
    2. kind: Pod
    3. metadata:
    4. name: memory-demo-2
    5. namespace: mem-example
    6. spec:
    7. containers:
    8. - name: memory-demo-2-ctr
    9. image: polinux/stress
    10. resources:
    11. requests:
    12. memory: "50Mi"
    13. limits:
    14. memory: "100Mi"
    15. command: ["stress"]
    16. args: ["--vm", "1", "--vm-bytes", "250M", "--vm-hang", "1"]

    In the args section of the configuration file, you can see that the Container will attempt to allocate 250 MiB of memory, which is well above the 100 MiB limit.

    Create the Pod:

    1. kubectl apply -f https://k8s.io/examples/pods/resource/memory-request-limit-2.yaml --namespace=mem-example

    View detailed information about the Pod:

    1. kubectl get pod memory-demo-2 --namespace=mem-example

    At this point, the Container might be running or killed. Repeat the preceding command until the Container is killed:

    1. NAME READY STATUS RESTARTS AGE
    2. memory-demo-2 0/1 OOMKilled 1 24s

    Get a more detailed view of the Container status:

    1. kubectl get pod memory-demo-2 --output=yaml --namespace=mem-example

    The output shows that the Container was killed because it is out of memory (OOM):

    1. lastState:
    2. terminated:
    3. containerID: docker://65183c1877aaec2e8427bc95609cc52677a454b56fcb24340dbd22917c23b10f
    4. exitCode: 137
    5. finishedAt: 2017-06-20T20:52:19Z
    6. reason: OOMKilled
    7. startedAt: null

    The Container in this exercise can be restarted, so the kubelet restarts it. Repeat this command several times to see that the Container is repeatedly killed and restarted:

    1. kubectl get pod memory-demo-2 --namespace=mem-example

    The output shows that the Container is killed, restarted, killed again, restarted again, and so on:

    1. kubectl get pod memory-demo-2 --namespace=mem-example
    2. memory-demo-2 0/1 OOMKilled 1 37s
    1. kubectl get pod memory-demo-2 --namespace=mem-example
    2. NAME READY STATUS RESTARTS AGE
    3. memory-demo-2 1/1 Running 2 40s

    View detailed information about the Pod history:

    1. kubectl describe pod memory-demo-2 --namespace=mem-example

    The output shows that the Container starts and fails repeatedly:

    View detailed information about your cluster’s Nodes:

    1. kubectl describe nodes

    The output includes a record of the Container being killed because of an out-of-memory condition:

      Delete your Pod:

      1. kubectl delete pod memory-demo-2 --namespace=mem-example

      Specify a memory request that is too big for your Nodes

      Memory requests and limits are associated with Containers, but it is useful to think of a Pod as having a memory request and limit. The memory request for the Pod is the sum of the memory requests for all the Containers in the Pod. Likewise, the memory limit for the Pod is the sum of the limits of all the Containers in the Pod.

      Pod scheduling is based on requests. A Pod is scheduled to run on a Node only if the Node has enough available memory to satisfy the Pod’s memory request.

      In this exercise, you create a Pod that has a memory request so big that it exceeds the capacity of any Node in your cluster. Here is the configuration file for a Pod that has one Container with a request for 1000 GiB of memory, which likely exceeds the capacity of any Node in your cluster.

      1. apiVersion: v1
      2. kind: Pod
      3. metadata:
      4. name: memory-demo-3
      5. namespace: mem-example
      6. spec:
      7. containers:
      8. - name: memory-demo-3-ctr
      9. image: polinux/stress
      10. resources:
      11. limits:
      12. memory: "1000Gi"
      13. requests:
      14. memory: "1000Gi"
      15. command: ["stress"]
      16. args: ["--vm", "1", "--vm-bytes", "150M", "--vm-hang", "1"]
      1. kubectl apply -f https://k8s.io/examples/pods/resource/memory-request-limit-3.yaml --namespace=mem-example

      View the Pod status:

      1. kubectl get pod memory-demo-3 --namespace=mem-example

      The output shows that the Pod status is PENDING. That is, the Pod is not scheduled to run on any Node, and it will remain in the PENDING state indefinitely:

      1. kubectl get pod memory-demo-3 --namespace=mem-example
      2. NAME READY STATUS RESTARTS AGE
      3. memory-demo-3 0/1 Pending 0 25s

      View detailed information about the Pod, including events:

      1. kubectl describe pod memory-demo-3 --namespace=mem-example

      The output shows that the Container cannot be scheduled because of insufficient memory on the Nodes:

      1. Events:
      2. ... Reason Message
      3. ------ -------
      4. ... FailedScheduling No nodes are available that match all of the following predicates:: Insufficient memory (3).

      Memory units

      The memory resource is measured in bytes. You can express memory as a plain integer or a fixed-point integer with one of these suffixes: E, P, T, G, M, K, Ei, Pi, Ti, Gi, Mi, Ki. For example, the following represent approximately the same value:

      Delete your Pod:

      If you do not specify a memory limit for a Container, one of the following situations applies:

      • The Container has no upper bound on the amount of memory it uses. The Container could use all of the memory available on the Node where it is running which in turn could invoke the OOM Killer. Further, in case of an OOM Kill, a container with no resource limits will have a greater chance of being killed.

      • The Container is running in a namespace that has a default memory limit, and the Container is automatically assigned the default limit. Cluster administrators can use a to specify a default value for the memory limit.

      Motivation for memory requests and limits

      By configuring memory requests and limits for the Containers that run in your cluster, you can make efficient use of the memory resources available on your cluster’s Nodes. By keeping a Pod’s memory request low, you give the Pod a good chance of being scheduled. By having a memory limit that is greater than the memory request, you accomplish two things:

      • The Pod can have bursts of activity where it makes use of memory that happens to be available.
      • The amount of memory a Pod can use during a burst is limited to some reasonable amount.

      Clean up

      Delete your namespace. This deletes all the Pods that you created for this task:

      1. kubectl delete namespace mem-example

      For cluster administrators