在 Kubernetes 中,可以使用 NodeSelector 字段来指定 Pod 调度到哪些节点上运行。NodeSelector 是一个键值对的 map,其中键是节点的标签名,值是标签值。具体步骤如下:
首先需要在节点上添加相应的标签,例如:
kubectl label nodes node1 app=example-app
在编写 Pod 的 YAML 文件时,需要添加 nodeSelector 字段来指定 Pod 要调度到哪些节点上运行。
apiVersion: v1
kind: Pod
metadata:
name: example-pod
spec:
containers:
- name: example-container
image: your_image_name
nodeSelector:
app: example-app # 指定要调度到带有 app=example-app 标签的节点上运行
kubectl apply -f your_pod_yaml_file.yaml
kubectl get pods example-pod -o wide # -o wide 可以查看分配给该 pod 的节点名称和 IP 地址等信息。
以上就是使用 golang NodeSelector 定向调度的步骤。
二,Node亲和性调度NodeAffinity
在 Kubernetes 中,可以使用 NodeAffinity 字段来指定 Pod 的亲和性调度规则,NodeAffinity 是一个结构体,其中包含两个属性:
下面是一个使用 golang NodeAffinity 进行亲和性调度的例子:
import (
"k8s.io/api/core/v1"
)
// 创建 PodSpec 对象
podSpec := &v1.PodSpec{
...
Affinity: &v1.Affinity{
NodeAffinity: &v1.NodeAffinity{
RequiredDuringSchedulingIgnoredDuringExecution: &v1.NodeSelector{ // 必须满足以下标签要求
NodeSelectorTerms: []v1.NodeSelectorTerm{
{
MatchExpressions: []v1.NodeSelectorRequirement{
{
Key: "zone",
Operator: v1.NodeSelectorOpIn,
Values: []string{"cn-east-01", "cn-east-02"},
},
},
},
},
},
PreferredDuringSchedulingIgnoredDuringExecution: nil, // 暂不需要优先级较低的亲和性规则
},
},
}
// 创建 Pod 对象
pod := &v1.Pod{
...
Spec: *podSpec,
}
在上述代码中,我们通过定义 RequiredDuringSchedulingIgnoredDuringExecution
属性,并设置 NodeSelectorTerms
和 MatchExpressions
来指定 Pod 调度到哪些节点上运行。这里我们要求节点的 zone
标签的值必须是 cn-east-01
或者 cn-east-02
。
需要注意的是,如果指定了多个亲和性规则,则每个规则都必须满足才能将 Pod 调度到对应的节点上。
除了使用 golang 代码进行调度之外,也可以在 Pod 的 YAML 文件中添加相应的 NodeAffinity 字段来指定 Pod 的亲和性调度规则。
三,Pod亲和度与互斥调度PodAffinity
在 Kubernetes 中,Pod 的亲和度与互斥调度可以通过 PodAffinity 和 PodAntiAffinity 字段进行设置。其中,PodAffinity 用于指定必须共存的 Pod,而 PodAntiAffinity 用于指定不得共存的 Pod。
这两个字段都包含了一个 requiredDuringSchedulingIgnoredDuringExecution
属性和一个 preferredDuringSchedulingIgnoredDuringExecution
属性:
下面是一个使用 golang 进行 Pod 亲和度与互斥调度的例子:
import (
"k8s.io/api/core/v1"
)
// 创建 PodSpec 对象
podSpec := &v1.PodSpec{
...
Affinity: &v1.Affinity{
PodAffinity: &v1.PodAffinity{ // 设置必须共存规则
RequiredDuringSchedulingIgnoredDuringExecution: []v1.PodAffinityTerm{
{
LabelSelector: &metav1.LabelSelector{
MatchLabels: map[string]string{"app": "nginx"},
},
Namespaces: []string{"default"},
TopologyKey: "kubernetes.io/hostname",
},
},
PreferredDuringSchedulingIgnoredDuringExecution: nil, // 暂不需要优先级较低的规则
},
PodAntiAffinity: &v1.PodAntiAffinity{ // 设置不得共存规则
RequiredDuringSchedulingIgnoredDuringExecution: []v1.PodAffinityTerm{
{
LabelSelector: &metav1.LabelSelector{
MatchLabels: map[string]string{"app": "nginx"},
},
Namespaces: []string{"default"},
TopologyKey: "kubernetes.io/hostname",
},
},
PreferredDuringSchedulingIgnoredDuringExecution: nil, // 暂不需要优先级较低的规则
},
},
}
// 创建 Pod 对象
pod := &v1.Pod{
...
Spec: *podSpec,
}
在上述代码中,我们通过定义 PodAffinity
和 PodAntiAffinity
属性,并设置 requiredDuringSchedulingIgnoredDuringExecution
字段和相应的规则,来指定 Pod 的亲和度与互斥调度。这里我们要求必须共存的 Pod 必须是标签为 app=nginx
的 Pod,而不得共存的 Pod 也必须是标签为 app=nginx
的 Pod。
需要注意的是,在设置亲和度与互斥调度时,可以通过多种方式指定匹配规则,如使用 label selector、namespace 等。在实际使用中,请根据实际情况进行选择。
四,pod Taints 和Tolerations
在Kubernetes集群中,Taints和Tolerations是用于控制Pod是否可以被调度到某个Node上的机制。
例如,假设我们有三个节点:node1、node2和node3。我们希望防止任何Pod被调度到node1上,并且只允许具有“gpu=true”标签的Pod在node2和node3上进行调度。我们可以执行以下操作:
kubectl taint nodes node1 key=value:NoSchedule
apiVersion: v1
kind: Pod
metadata:
name: gpu-pod
spec:
containers:
- name: gpu-container
image: gpu-image:v1.0
tolerations:
- key: "gpu"
operator: "Equal"
value: "true"
effect: "NoSchedule"
这样,只有具有“gpu=true”标签的Pod才能被调度到node2和node3上。而在node1上添加了taint,因此不会有任何Pod被调度到该节点上。
五,pod Priority 优先级调度
在 Kubernetes 集群中,Pod 被分配给 Node 运行。而 Pod 之间的优先级是由 Kubernetes 的调度器(Scheduler)来控制的。Pod 的优先级越高,则它被分配到节点上的机会就越大。
为了设置 Pod 的优先级,可以使用 PriorityClass 对象。PriorityClass 是一个资源对象,用于定义 Pod 的优先级和访问策略。
下面是一个示例 PriorityClass:
apiVersion: scheduling.k8s.io/v1
kind: PriorityClass
metadata:
name: high-priority
value: 1000000
globalDefault: false
description: "This priority class should be used for important pods only."
在这个示例中,我们创建了一个名为 high-priority
的 PriorityClass,并将其值设置为 1000000
。这意味着当一个 Pod 使用此 PriorityClass 来定义它的优先级时,它将比其他具有较低值的 PriorityClass 更容易被分配到节点上运行。
要使用这个 PriorityClass,请在 Pod 定义文件中添加如下字段:
apiVersion: v1
kind: Pod
metadata:
name: my-pod
spec:
priorityClassName: high-priority
containers:
- name: my-container
image: nginx
注意,在设置完 priorityClassName
字段后,请确保所有相关的 Node 上都安装了 kube-scheduler 组件,并且该组件已经启动。这样才能使调度器根据定义的优先级来调度 Pod 的运行。