中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Preemption搶占式調度的方法是什么

發布時間:2021-12-20 09:50:54 來源:億速云 閱讀:258 作者:iii 欄目:云計算

這篇文章主要講解了“Preemption搶占式調度的方法是什么”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“Preemption搶占式調度的方法是什么”吧!

ScheduleAlgorithm的變化

在Kubernetes 1.8中,對ScheduleAlgorithm Interface的定義發生了改變,多了一個Preempt(...)。因此,我在博文Kubernetes Scheduler原理解析(當時是基于kubernetes 1.5)中對scheduler調度過程開的一句話概括“將PodSpec.NodeName為空的Pods逐個地,經過預選(Predicates)和優選(Priorities)兩個步驟,挑選最合適的Node作為該Pod的Destination。”將不再準確了。

現在應該一句話這樣描述才算準確了:“將PodSpec.NodeName為空的Pods逐個地,經過預選(Predicates)和優選(Priorities)兩個步驟,挑選最合適的Node作為該Pod的Destination。如果經過預選和優選仍然沒有找到合適的節點,并且啟動了Pod Priority,那么該Pod將會進行Preempt搶占式調度找到最合適的節點及需要Evict的Pods。”

// ScheduleAlgorithm is an interface implemented by things that know how to schedule pods
// onto machines.
type ScheduleAlgorithm interface {
	Schedule(*v1.Pod, NodeLister) (selectedMachine string, err error)
	// Preempt receives scheduling errors for a pod and tries to create room for
	// the pod by preempting lower priority pods if possible.
	// It returns the node where preemption happened, a list of preempted pods, and error if any.
	Preempt(*v1.Pod, NodeLister, error) (selectedNode *v1.Node, preemptedPods []*v1.Pod, err error)
	// Predicates() returns a pointer to a map of predicate functions. This is
	// exposed for testing.
	Predicates() map[string]FitPredicate
	// Prioritizers returns a slice of priority config. This is exposed for
	// testing.
	Prioritizers() []PriorityConfig
}

Scheduler.scheduleOne開始真正的調度邏輯,每次負責一個Pod的調度,邏輯如下:

  • 從PodQueue中獲取一個Pod。

  • 執行對應Algorithm的Schedule,進行預選和優選。

  • AssumePod

  • Bind Pod, 如果Bind Failed,ForgetPod。

在1.8中,但預選和優選調度完整沒有找到合適node時(其實一定會是預選沒有找到nodes,優選只是挑更好的),還會調用sched.preempt進行搶占式調度。

plugin/pkg/scheduler/scheduler.go:293

func (sched *Scheduler) scheduleOne() {
	pod := sched.config.NextPod()
	if pod.DeletionTimestamp != nil {
		sched.config.Recorder.Eventf(pod, v1.EventTypeWarning, "FailedScheduling", "skip schedule deleting pod: %v/%v", pod.Namespace, pod.Name)
		glog.V(3).Infof("Skip schedule deleting pod: %v/%v", pod.Namespace, pod.Name)
		return
	}

	glog.V(3).Infof("Attempting to schedule pod: %v/%v", pod.Namespace, pod.Name)

	// Synchronously attempt to find a fit for the pod.
	start := time.Now()
	suggestedHost, err := sched.schedule(pod)
	metrics.SchedulingAlgorithmLatency.Observe(metrics.SinceInMicroseconds(start))
	if err != nil {
		// schedule() may have failed because the pod would not fit on any host, so we try to
		// preempt, with the expectation that the next time the pod is tried for scheduling it
		// will fit due to the preemption. It is also possible that a different pod will schedule
		// into the resources that were preempted, but this is harmless.
		if fitError, ok := err.(*core.FitError); ok {
			sched.preempt(pod, fitError)
		}
		return
	}

	// Tell the cache to assume that a pod now is running on a given node, even though it hasn't been bound yet.
	// This allows us to keep scheduling without waiting on binding to occur.
	assumedPod := *pod
	// assume modifies `assumedPod` by setting NodeName=suggestedHost
	err = sched.assume(&assumedPod, suggestedHost)
	if err != nil {
		return
	}

	// bind the pod to its host asynchronously (we can do this b/c of the assumption step above).
	go func() {
		err := sched.bind(&assumedPod, &v1.Binding{
			ObjectMeta: metav1.ObjectMeta{Namespace: assumedPod.Namespace, Name: assumedPod.Name, UID: assumedPod.UID},
			Target: v1.ObjectReference{
				Kind: "Node",
				Name: suggestedHost,
			},
		})
		metrics.E2eSchedulingLatency.Observe(metrics.SinceInMicroseconds(start))
		if err != nil {
			glog.Errorf("Internal error binding pod: (%v)", err)
		}
	}()
}

Scheduler.preemt

好的,關于預選和優選,我這里不做過多解讀,因為整個源碼邏輯和1.5是一樣,不同的是1.8增加了更多的Predicate和Priority Policys及其實現。下面只看搶占式調度Preempt的代碼。

plugin/pkg/scheduler/scheduler.go:191

func (sched *Scheduler) preempt(preemptor *v1.Pod, scheduleErr error) (string, error) {
	if !utilfeature.DefaultFeatureGate.Enabled(features.PodPriority) {
		glog.V(3).Infof("Pod priority feature is not enabled. No preemption is performed.")
		return "", nil
	}
	preemptor, err := sched.config.PodPreemptor.GetUpdatedPod(preemptor)
	if err != nil {
		glog.Errorf("Error getting the updated preemptor pod object: %v", err)
		return "", err
	}
	node, victims, err := sched.config.Algorithm.Preempt(preemptor, sched.config.NodeLister, scheduleErr)
	if err != nil {
		glog.Errorf("Error preempting victims to make room for %v/%v.", preemptor.Namespace, preemptor.Name)
		return "", err
	}
	if node == nil {
		return "", err
	}
	glog.Infof("Preempting %d pod(s) on node %v to make room for %v/%v.", len(victims), node.Name, preemptor.Namespace, preemptor.Name)
	annotations := map[string]string{core.NominatedNodeAnnotationKey: node.Name}
	err = sched.config.PodPreemptor.UpdatePodAnnotations(preemptor, annotations)
	if err != nil {
		glog.Errorf("Error in preemption process. Cannot update pod %v annotations: %v", preemptor.Name, err)
		return "", err
	}
	for _, victim := range victims {
		if err := sched.config.PodPreemptor.DeletePod(victim); err != nil {
			glog.Errorf("Error preempting pod %v/%v: %v", victim.Namespace, victim.Name, err)
			return "", err
		}
		sched.config.Recorder.Eventf(victim, v1.EventTypeNormal, "Preempted", "by %v/%v on node %v", preemptor.Namespace, preemptor.Name, node.Name)
	}
	return node.Name, err
}
  • 檢查FeaturesGate中是否開啟了PodPriority,如果沒開啟,則不會進行后續Preemption操作;

  • 由于該Pod在Predicate/Priortiy調度過程失敗后,會更新PodCondition,記錄調度失敗狀態及失敗原因。因此需要從apiserver中獲取PodCondition更新后的Pod Object;

  • 調用ScheduleAlgorithm.Preempt進行搶占式調度,選出最佳node和待preempt pods(稱為victims);

  • 調用apiserver給該pod(稱為Preemptor)打上Annotation:NominatedNodeName=nodeName;

  • 遍歷victims,調用apiserver進行逐個刪除這些pods;

注意:在scheduler調用shed.schedule(pod)進行預選和優選調度失敗時,Pod Bind Node失敗,該Pod會requeue unscheduled Cache podqueue中,如果在這個pod調度過程中又有新的pod加入到待調度隊列,那么該pod requeue時它前面就有其他pod,下一次調度就是先調度在它前面的pod,而這些pod的調度有可能會調度到剛剛通過Preempt釋放資源的Node上,導致把剛才Preemptor釋放的resource消耗掉。當再次輪到上次的Preemptor調度時,可能又需要觸發一次某個節點的Preempt。

genericScheduler.Preempt

ScheduleAlgorithm.Preempt是搶占式調度的關鍵實現,其對應的實現在genericScheduler中:

plugin/pkg/scheduler/core/generic_scheduler.go:181

// preempt finds nodes with pods that can be preempted to make room for "pod" to
// schedule. It chooses one of the nodes and preempts the pods on the node and
// returns the node and the list of preempted pods if such a node is found.
// TODO(bsalamat): Add priority-based scheduling. More info: today one or more
// pending pods (different from the pod that triggered the preemption(s)) may
// schedule into some portion of the resources freed up by the preemption(s)
// before the pod that triggered the preemption(s) has a chance to schedule
// there, thereby preventing the pod that triggered the preemption(s) from
// scheduling. Solution is given at:
// https://github.com/kubernetes/community/blob/master/contributors/design-proposals/pod-preemption.md#preemption-mechanics
func (g *genericScheduler) Preempt(pod *v1.Pod, nodeLister algorithm.NodeLister, scheduleErr error) (*v1.Node, []*v1.Pod, error) {
	// Scheduler may return various types of errors. Consider preemption only if
	// the error is of type FitError.
	fitError, ok := scheduleErr.(*FitError)
	if !ok || fitError == nil {
		return nil, nil, nil
	}
	err := g.cache.UpdateNodeNameToInfoMap(g.cachedNodeInfoMap)
	if err != nil {
		return nil, nil, err
	}
	if !podEligibleToPreemptOthers(pod, g.cachedNodeInfoMap) {
		glog.V(5).Infof("Pod %v is not eligible for more preemption.", pod.Name)
		return nil, nil, nil
	}
	allNodes, err := nodeLister.List()
	if err != nil {
		return nil, nil, err
	}
	if len(allNodes) == 0 {
		return nil, nil, ErrNoNodesAvailable
	}
	potentialNodes := nodesWherePreemptionMightHelp(pod, allNodes, fitError.FailedPredicates)
	if len(potentialNodes) == 0 {
		glog.V(3).Infof("Preemption will not help schedule pod %v on any node.", pod.Name)
		return nil, nil, nil
	}
	nodeToPods, err := selectNodesForPreemption(pod, g.cachedNodeInfoMap, potentialNodes, g.predicates, g.predicateMetaProducer)
	if err != nil {
		return nil, nil, err
	}
	for len(nodeToPods) > 0 {
		node := pickOneNodeForPreemption(nodeToPods)
		if node == nil {
			return nil, nil, err
		}
		passes, pErr := nodePassesExtendersForPreemption(pod, node.Name, nodeToPods[node], g.cachedNodeInfoMap, g.extenders)
		if passes && pErr == nil {
			return node, nodeToPods[node], err
		}
		if pErr != nil {
			glog.Errorf("Error occurred while checking extenders for preemption on node %v: %v", node, pErr)
		}
		// Remove the node from the map and try to pick a different node.
		delete(nodeToPods, node)
	}
	return nil, nil, err
}

sched.schedule error檢查

  • 只有前面sched.schedule()返回的error為FitError類型時,才會觸發后續的Preemption。FitError就是表示pod在Predicate階段進行某些PredicateFunc篩選時不通過。也就是說只有預選失敗的Pod才會進行搶占式調度。

更新scheduler cache中的NodeInfo

  • 更新scheduler cache中NodeInfo,主要是更新Node上scheduled 和Assumed Pods,作為后續Preempt Pods時的考慮范圍,確保Preemption是正確的。

podEligibleToPreemptOthers檢查pod是否有資格進行搶占式調度

  • invoke podEligibleToPreemptOthers來判斷該pod是否適合進行后續的Preemption,判斷邏輯是:

    • 如果該Pod已經包含Annotation:NominatedNodeName=nodeName(說明該pod之前已經Preempted),并且Annotation中的這個Node有比該pod優先級更低的pod正在Terminating,則認為該pod不適合進行后續的Preemption,流程結束。

    • 除此之外,繼續后續的流程。

    • 對應代碼如下:

      plugin/pkg/scheduler/core/generic_scheduler.go:756
      
      func podEligibleToPreemptOthers(pod *v1.Pod, nodeNameToInfo map[string]*schedulercache.NodeInfo) bool {
      	if nodeName, found := pod.Annotations[NominatedNodeAnnotationKey]; found {
      		if nodeInfo, found := nodeNameToInfo[nodeName]; found {
      			for _, p := range nodeInfo.Pods() {
      				if p.DeletionTimestamp != nil && util.GetPodPriority(p) < util.GetPodPriority(pod) {
      					// There is a terminating pod on the nominated node.
      					return false
      				}
      			}
      		}
      	}
      	return true
      }


nodesWherePreemptionMightHelp篩選出Potential Nodes

  • invoke nodesWherePreemptionMightHelp來獲取potential nodes。nodesWherePreemptionMightHelp的邏輯是:

    • NodeSelectorNotMatch,

    • PodNotMatchHostName,

    • TaintsTolerationsNotMatch,

    • NodeLabelPresenceViolated,

    • NodeNotReady,

    • NodeNetworkUnavailable,

    • NodeUnschedulable,

    • NodeUnknownCondition

    • 遍歷所有的nodes,對每個nodes在sched.schedule()在預選階段失敗的Predicate策略(failedPredicates)進行掃描,如果failedPredicates包含以下Policy,則說明該node不適合作為Preempt的備選節點。

    • 除此之外的Node均作為Potential Nodes。

    • 對應代碼如下:

      func nodesWherePreemptionMightHelp(pod *v1.Pod, nodes []*v1.Node, failedPredicatesMap FailedPredicateMap) []*v1.Node {
      	potentialNodes := []*v1.Node{}
      	for _, node := range nodes {
      		unresolvableReasonExist := false
      		failedPredicates, found := failedPredicatesMap[node.Name]
      		// If we assume that scheduler looks at all nodes and populates the failedPredicateMap
      		// (which is the case today), the !found case should never happen, but we'd prefer
      		// to rely less on such assumptions in the code when checking does not impose
      		// significant overhead.
      		for _, failedPredicate := range failedPredicates {
      			switch failedPredicate {
      			case
      				predicates.ErrNodeSelectorNotMatch,
      				predicates.ErrPodNotMatchHostName,
      				predicates.ErrTaintsTolerationsNotMatch,
      				predicates.ErrNodeLabelPresenceViolated,
      				predicates.ErrNodeNotReady,
      				predicates.ErrNodeNetworkUnavailable,
      				predicates.ErrNodeUnschedulable,
      				predicates.ErrNodeUnknownCondition:
      				unresolvableReasonExist = true
      				break
      				// TODO(bsalamat): Please add affinity failure cases once we have specific affinity failure errors.
      			}
      		}
      		if !found || !unresolvableReasonExist {
      			glog.V(3).Infof("Node %v is a potential node for preemption.", node.Name)
      			potentialNodes = append(potentialNodes, node)
      		}
      	}
      	return potentialNodes
      }


selectNodesForPreemption和selectVictimsOnNode選出可行Nodes及其對應的victims

  • invoke selectNodesForPreemption從Potential Nodes中找出所有可行的Nodes及對應的victim Pods,其對應的邏輯如為:啟動max(16, potentialNodesNum)個worker(對應goruntine)通過WaitGroups并發等待所有node的check完成:

    • 遍歷該node上所有的scheduled pods(包括assumed pods),將優先級比Preemptor更低的Pods都加入到Potential victims List中,并且將這些victims從NodeInfoCopy中刪除,下次進行Predicate時就意味著Node上有更多資源可用。

    • 對Potential victims中元素進行排序,排序規則是按照優先級從高到底排序的,index為0的對應的優先級最高。

    • 檢查Preemptor是否能scheduler配置的所有Predicates Policy(基于前面將這些victims從NodeInfoCopy中刪除,將所有更低優先級的pods資源全部釋放了),如果不通過則返回,表示該node不合適。All Predicate通過后,繼續下面流程。

    • 遍歷所有的Potential victims list item(已經按照優先級從高到底排序),試著把Potential victims中第一個Pod(優先級最高)加回到NodeInfoCopy中,再檢查Preemptor是否能scheduler配置的所有Predicates Policy,如果不滿足就把該pod再從NodeInfoCopy中刪除,并且正式加入到victims list中。接著對Potential victims中第2,3...個Pod進行同樣處理。這樣做,是為了保證盡量保留優先級更高的Pods,盡量刪除更少的Pods。

    • 最終返回每個可行node及其對應victims list。

    • selectNodesForPreemption代碼如下,其實核心代碼在selectVictimsOnNode

      plugin/pkg/scheduler/core/generic_scheduler.go:583
      
      func selectNodesForPreemption(pod *v1.Pod,
      	nodeNameToInfo map[string]*schedulercache.NodeInfo,
      	potentialNodes []*v1.Node,
      	predicates map[string]algorithm.FitPredicate,
      	metadataProducer algorithm.PredicateMetadataProducer,
      ) (map[*v1.Node][]*v1.Pod, error) {
      
      	nodeNameToPods := map[*v1.Node][]*v1.Pod{}
      	var resultLock sync.Mutex
      
      	// We can use the same metadata producer for all nodes.
      	meta := metadataProducer(pod, nodeNameToInfo)
      	checkNode := func(i int) {
      		nodeName := potentialNodes[i].Name
      		var metaCopy algorithm.PredicateMetadata
      		if meta != nil {
      			metaCopy = meta.ShallowCopy()
      		}
      		pods, fits := selectVictimsOnNode(pod, metaCopy, nodeNameToInfo[nodeName], predicates)
      		if fits {
      			resultLock.Lock()
      			nodeNameToPods[potentialNodes[i]] = pods
      			resultLock.Unlock()
      		}
      	}
      	workqueue.Parallelize(16, len(potentialNodes), checkNode)
      	return nodeNameToPods, nil
      }


      plugin/pkg/scheduler/core/generic_scheduler.go:659
      
      func selectVictimsOnNode(
      	pod *v1.Pod,
      	meta algorithm.PredicateMetadata,
      	nodeInfo *schedulercache.NodeInfo,
      	fitPredicates map[string]algorithm.FitPredicate) ([]*v1.Pod, bool) {
      	potentialVictims := util.SortableList{CompFunc: util.HigherPriorityPod}
      	nodeInfoCopy := nodeInfo.Clone()
      
      	removePod := func(rp *v1.Pod) {
      		nodeInfoCopy.RemovePod(rp)
      		if meta != nil {
      			meta.RemovePod(rp)
      		}
      	}
      	addPod := func(ap *v1.Pod) {
      		nodeInfoCopy.AddPod(ap)
      		if meta != nil {
      			meta.AddPod(ap, nodeInfoCopy)
      		}
      	}
      	// As the first step, remove all the lower priority pods from the node and
      	// check if the given pod can be scheduled.
      	podPriority := util.GetPodPriority(pod)
      	for _, p := range nodeInfoCopy.Pods() {
      		if util.GetPodPriority(p) < podPriority {
      			potentialVictims.Items = append(potentialVictims.Items, p)
      			removePod(p)
      		}
      	}
      	potentialVictims.Sort()
      	// If the new pod does not fit after removing all the lower priority pods,
      	// we are almost done and this node is not suitable for preemption. The only condition
      	// that we should check is if the "pod" is failing to schedule due to pod affinity
      	// failure.
      	// TODO(bsalamat): Consider checking affinity to lower priority pods if feasible with reasonable performance.
      	if fits, _, err := podFitsOnNode(pod, meta, nodeInfoCopy, fitPredicates, nil); !fits {
      		if err != nil {
      			glog.Warningf("Encountered error while selecting victims on node %v: %v", nodeInfo.Node().Name, err)
      		}
      		return nil, false
      	}
      	victims := []*v1.Pod{}
      	// Try to reprieve as many pods as possible starting from the highest priority one.
      	for _, p := range potentialVictims.Items {
      		lpp := p.(*v1.Pod)
      		addPod(lpp)
      		if fits, _, _ := podFitsOnNode(pod, meta, nodeInfoCopy, fitPredicates, nil); !fits {
      			removePod(lpp)
      			victims = append(victims, lpp)
      			glog.V(5).Infof("Pod %v is a potential preemption victim on node %v.", lpp.Name, nodeInfo.Node().Name)
      		}
      	}
      	return victims, true
      }


pickOneNodeForPreemption從可行Nodes中找出最合適的一個Node

  • 如果上一步至少找到一個可行node,則調用pickOneNodeForPreemption按照以下邏輯選擇一個最合適的node:

    • 選擇victims中最高pod優先級最低的那個Node。

    • 如果上一步有不止一個Nodes滿足條件,則再對選擇所有victims優先級之和最小的那個Node。

    • 如果上一步有不止一個Nodes滿足條件,則再選擇victims pod數最少的Node。

    • 如果上一步有不止一個Nodes滿足條件,則再隨機選擇一個Node。

    • 以上每一步的Nodes列表,都是基于上一步篩選后的Nodes。

      plugin/pkg/scheduler/core/generic_scheduler.go:501
      
      func pickOneNodeForPreemption(nodesToPods map[*v1.Node][]*v1.Pod) *v1.Node {
      	type nodeScore struct {
      		node            *v1.Node
      		highestPriority int32
      		sumPriorities   int64
      		numPods         int
      	}
      	if len(nodesToPods) == 0 {
      		return nil
      	}
      	minHighestPriority := int32(math.MaxInt32)
      	minPriorityScores := []*nodeScore{}
      	for node, pods := range nodesToPods {
      		if len(pods) == 0 {
      			// We found a node that doesn't need any preemption. Return it!
      			// This should happen rarely when one or more pods are terminated between
      			// the time that scheduler tries to schedule the pod and the time that
      			// preemption logic tries to find nodes for preemption.
      			return node
      		}
      		// highestPodPriority is the highest priority among the victims on this node.
      		highestPodPriority := util.GetPodPriority(pods[0])
      		if highestPodPriority < minHighestPriority {
      			minHighestPriority = highestPodPriority
      			minPriorityScores = nil
      		}
      		if highestPodPriority == minHighestPriority {
      			minPriorityScores = append(minPriorityScores, &nodeScore{node: node, highestPriority: highestPodPriority, numPods: len(pods)})
      		}
      	}
      	if len(minPriorityScores) == 1 {
      		return minPriorityScores[0].node
      	}
      	// There are a few nodes with minimum highest priority victim. Find the
      	// smallest sum of priorities.
      	minSumPriorities := int64(math.MaxInt64)
      	minSumPriorityScores := []*nodeScore{}
      	for _, nodeScore := range minPriorityScores {
      		var sumPriorities int64
      		for _, pod := range nodesToPods[nodeScore.node] {
      			// We add MaxInt32+1 to all priorities to make all of them >= 0. This is
      			// needed so that a node with a few pods with negative priority is not
      			// picked over a node with a smaller number of pods with the same negative
      			// priority (and similar scenarios).
      			sumPriorities += int64(util.GetPodPriority(pod)) + int64(math.MaxInt32+1)
      		}
      		if sumPriorities < minSumPriorities {
      			minSumPriorities = sumPriorities
      			minSumPriorityScores = nil
      		}
      		nodeScore.sumPriorities = sumPriorities
      		if sumPriorities == minSumPriorities {
      			minSumPriorityScores = append(minSumPriorityScores, nodeScore)
      		}
      	}
      	if len(minSumPriorityScores) == 1 {
      		return minSumPriorityScores[0].node
      	}
      	// There are a few nodes with minimum highest priority victim and sum of priorities.
      	// Find one with the minimum number of pods.
      	minNumPods := math.MaxInt32
      	minNumPodScores := []*nodeScore{}
      	for _, nodeScore := range minSumPriorityScores {
      		if nodeScore.numPods < minNumPods {
      			minNumPods = nodeScore.numPods
      			minNumPodScores = nil
      		}
      		if nodeScore.numPods == minNumPods {
      			minNumPodScores = append(minNumPodScores, nodeScore)
      		}
      	}
      	// At this point, even if there are more than one node with the same score,
      	// return the first one.
      	if len(minNumPodScores) > 0 {
      		return minNumPodScores[0].node
      	}
      	glog.Errorf("Error in logic of node scoring for preemption. We should never reach here!")
      	return nil
      }

最合適的Node仍然要交給extender(if configed)檢查

  • 如果scheduler配置extender scheduler,則還需要通過invoke nodePassesExtendersForPreemption再次將該pod和(假設)剔除victims的該node交給extender.Filter進行一下檢查,只有檢查通過了才返回該node作為最終選擇的Preempt node。

  • 關于extender的理解,請參考如何對kubernetes scheduler進行二次開發和Kubernetes Scheduler源碼分析。其實用的場景不多,現在支持自定義調度器了,就更少需要使用scheduler extender了。

感謝各位的閱讀,以上就是“Preemption搶占式調度的方法是什么”的內容了,經過本文的學習后,相信大家對Preemption搶占式調度的方法是什么這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

南华县| 沂南县| 通城县| 桃园县| 平乡县| 侯马市| 达尔| 益阳市| 龙海市| 黄浦区| 财经| 樟树市| 墨玉县| 合川市| 克拉玛依市| 鹰潭市| 刚察县| 宿迁市| 卫辉市| 会理县| 石景山区| 武平县| 长沙市| 西乡县| 航空| 西城区| 石屏县| 大方县| 云浮市| 洛隆县| 九寨沟县| 宁阳县| 奉化市| 晋州市| 兴海县| 琼结县| 衡山县| 沙田区| 咸丰县| 外汇| 安化县|