recherche
Maisondéveloppement back-endTutoriel PythonExplication détaillée d'exemples d'implémentation interne du verrouillage mutex Golang

Cet article présente et explique principalement en détail l'implémentation interne du verrouillage mutex Golang. L'éditeur pense que c'est assez bon, je vais donc le partager avec vous maintenant et le donner comme référence. Suivons l'éditeur et jetons un coup d'œil

Le langage go fournit un moyen prêt à l'emploi de partager des ressources, un verrouillage mutex (sync.Mutex), une valeur nulle de sync.Mutex signifie que l'on est non verrouillé Oui, il peut être utilisé directement. Une fois qu'un goroutine a obtenu le verrou mutex, les autres goroutines ne peuvent qu'attendre que le goroutine libère le verrou mutex. Il n'y a que deux fonctions exposées dans la structure Mutex, à savoir Verrouiller et Déverrouiller. verrouillage mutex C'est très simple et cet article n'explique pas son utilisation.

Ne copiez jamais les valeurs lors de l'utilisation de sync.Mutex, car cela pourrait entraîner l'expiration du verrou. Lorsque nous ouvrons notre IDE et passons à notre code sync.Mutex, nous constaterons qu'il a la structure suivante :


type Mutex struct {
 state int32   //互斥锁上锁状态枚举值如下所示
 sema uint32  //信号量,向处于Gwaitting的G发送信号
}

const (
 mutexLocked = 1 << iota // 1 互斥锁是锁定的
 mutexWoken       // 2 唤醒锁
 mutexWaiterShift = iota // 2 统计阻塞在这个互斥锁上的goroutine数目需要移位的数值
)

Les valeurs d'état ci-dessus sont 0 ( Disponible) 1 (verrouillé) 2~31 En attenteQueueCount

Ce qui suit est le code source du verrouillage mutex. Il existe quatre autres méthodes importantes qui doivent être expliquées à l'avance, à savoir runtime_canSpin, runtime_doSpin, runtime_SemacquireMutex, runtime_Semrelease,

1. runtime_canSpin : le spin lock dans golang ne tournera pas éternellement. La méthode runtime_canSpin dans le package d'exécution applique certaines restrictions : l'iter transmis est supérieur ou égal à 4 ou le nombre de cœurs de processeur est inférieur ou égal à 1, le processeur logique maximum est supérieur à 1, il existe au moins une file d'attente P locale et la file d'attente P locale peut exécuter le La file d'attente G est vide.


//go:linkname sync_runtime_canSpin sync.runtime_canSpin
func sync_runtime_canSpin(i int) bool {
 if i >= active_spin || ncpu <= 1 || gomaxprocs <= int32(sched.npidle+sched.nmspinning)+1 {
 return false
 }
 if p := getg().m.p.ptr(); !runqempty(p) {
 return false
 }
 return true
}

2. runtime_doSpin : appellera la fonction proyield, qui est également implémentée en langage assembleur. A l'intérieur de la fonction, la boucle appelle l'instruction PAUSE. L'instruction PAUSE ne fait rien, mais elle consomme du temps CPU Lors de l'exécution de l'instruction PAUSE, le CPU n'y effectuera pas d'optimisations inutiles.


//go:linkname sync_runtime_doSpin sync.runtime_doSpin
func sync_runtime_doSpin() {
 procyield(active_spin_cnt)
}

3. runtime_SemacquireMutex :


//go:linkname sync_runtime_SemacquireMutex sync.runtime_SemacquireMutex
func sync_runtime_SemacquireMutex(addr *uint32) {
 semacquire(addr, semaBlockProfile|semaMutexProfile)
}

4 , runtime_Semrelease :


//go:linkname sync_runtime_Semrelease sync.runtime_Semrelease
func sync_runtime_Semrelease(addr *uint32) {
 semrelease(addr)
}
Mutex的Lock函数定义如下

func (m *Mutex) Lock() {
    //先使用CAS尝试获取锁
 if atomic.CompareAndSwapInt32(&m.state, 0, mutexLocked) {
        //这里是-race不需要管它
 if race.Enabled {
  race.Acquire(unsafe.Pointer(m))
 }
        //成功获取返回
 return
 }

 awoke := false //循环标记
 iter := 0    //循环计数器
 for {
 old := m.state //获取当前锁状态
 new := old | mutexLocked //将当前状态最后一位指定1
 if old&mutexLocked != 0 { //如果所以被占用
  if runtime_canSpin(iter) { //检查是否可以进入自旋锁
  if !awoke && old&mutexWoken == 0 && old>>mutexWaiterShift != 0 &&
   atomic.CompareAndSwapInt32(&m.state, old, old|mutexWoken) { 
                    //awoke标记为true
   awoke = true
  }
                //进入自旋状态
  runtime_doSpin()
  iter++
  continue
  }
            //没有获取到锁,当前G进入Gwaitting状态
  new = old + 1<<mutexWaiterShift
 }
 if awoke {
  if new&mutexWoken == 0 {
  throw("sync: inconsistent mutex state")
  }
            //清除标记
  new &^= mutexWoken
 }
        //更新状态
 if atomic.CompareAndSwapInt32(&m.state, old, new) {
  if old&mutexLocked == 0 {
  break
  }
             
            // 锁请求失败,进入休眠状态,等待信号唤醒后重新开始循环
  runtime_SemacquireMutex(&m.sema)
  awoke = true
  iter = 0
 }
 }

 if race.Enabled {
 race.Acquire(unsafe.Pointer(m))
 }
}
Mutex的Unlock函数定义如下

func (m *Mutex) Unlock() {
 if race.Enabled {
 _ = m.state
 race.Release(unsafe.Pointer(m))
 }

 // 移除标记
 new := atomic.AddInt32(&m.state, -mutexLocked)
 if (new+mutexLocked)&mutexLocked == 0 {
 throw("sync: unlock of unlocked mutex")
 }

 old := new
 for {
 //当休眠队列内的等待计数为0或者自旋状态计数器为0,退出
 if old>>mutexWaiterShift == 0 || old&(mutexLocked|mutexWoken) != 0 {
  return
 }
 // 减少等待次数,添加清除标记
 new = (old - 1<<mutexWaiterShift) | mutexWoken
 if atomic.CompareAndSwapInt32(&m.state, old, new) {
            // 释放锁,发送释放信号
  runtime_Semrelease(&m.sema)
  return
 }
 old = m.state
 }
}

Mutex sans conflit est le cas le plus simple Lorsqu'il y a un conflit, la rotation est effectuée en premier, car la plupart des segments de code sont protégés. par Mutex sont très courts et peuvent être obtenus après une courte rotation ; si l'attente de rotation échoue, la Goroutine actuelle doit entrer dans l'état Gwaitting via un sémaphore.

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Quelles sont les alternatives pour concaténer deux listes dans Python?Quelles sont les alternatives pour concaténer deux listes dans Python?May 09, 2025 am 12:16 AM

Il existe de nombreuses méthodes pour connecter deux listes dans Python: 1. Utilisez des opérateurs, qui sont simples mais inefficaces dans les grandes listes; 2. Utiliser la méthode Extende, qui est efficace mais modifiera la liste d'origine; 3. Utilisez l'opérateur = qui est à la fois efficace et lisible; 4. Utilisez la fonction itertools.chain, qui est efficace de la mémoire mais nécessite une importation supplémentaire; 5. Utilisez l'analyse de la liste, qui est élégante mais peut être trop complexe. La méthode de sélection doit être basée sur le contexte et les exigences du code.

Python: moyens efficaces de fusionner deux listesPython: moyens efficaces de fusionner deux listesMay 09, 2025 am 12:15 AM

Il existe de nombreuses façons de fusionner les listes Python: 1. Utilisez des opérateurs, qui sont simples mais pas efficaces par la mémoire pour les grandes listes; 2. Utiliser la méthode Extende, qui est efficace mais modifiera la liste d'origine; 3. Utilisez itertools.chain, qui convient aux grands ensembles de données; 4. Utiliser * l'opérateur, fusionner les listes de petites à moyennes dans une ligne de code; 5. Utilisez Numpy.concatenate, qui convient aux grands ensembles de données et scénarios avec des exigences de performance élevées; 6. Utilisez la méthode d'ajout, qui convient aux petites listes mais est inefficace. Lors de la sélection d'une méthode, vous devez considérer la taille de la liste et les scénarios d'application.

Compilé vs langues interprétées: avant et inconvénientsCompilé vs langues interprétées: avant et inconvénientsMay 09, 2025 am 12:06 AM

CompiledLanguagesOffersPeedandSecurity, tandis que l'interprété des langues de la durée de la valeur et de la sport.1) Compilé LanguagesLikec ArefasterandSecureButhAvelongerDevelopmentCyclesandPlatformDependency.2)

Python: pour et pendant que les boucles, le guide le plus completPython: pour et pendant que les boucles, le guide le plus completMay 09, 2025 am 12:05 AM

Dans Python, une boucle pour une boucle est utilisée pour traverser les objets itérable, et une boucle WHE est utilisée pour effectuer des opérations à plusieurs reprises lorsque la condition est satisfaite. 1) Pour l'exemple de boucle: traversez la liste et imprimez les éléments. 2) Place de Loop: Devinez le jeu numérique jusqu'à ce que vous le devassiez correctement. Les principes du cycle de maîtrise et les techniques d'optimisation peuvent améliorer l'efficacité et la fiabilité du code.

Python concaténate se liste en une chaînePython concaténate se liste en une chaîneMay 09, 2025 am 12:02 AM

Pour concaténer une liste dans une chaîne, l'utilisation de la méthode join () dans Python est le meilleur choix. 1) Utilisez la méthode join () pour concaténer les éléments de liste en une chaîne, telle que '' .join (my_list). 2) Pour une liste contenant des numéros, convertissez la carte (STR, numéros) en une chaîne avant de concaténer. 3) Vous pouvez utiliser des expressions de générateur pour le formatage complexe, telles que ','. JOIN (f '({fruit})' forfruitInfruits). 4) Lors du traitement des types de données mixtes, utilisez MAP (STR, mixtes_list) pour vous assurer que tous les éléments peuvent être convertis en chaînes. 5) Pour les grandes listes, utilisez '' .join (grand_li

Approche hybride de Python: compilation et interprétation combinéesApproche hybride de Python: compilation et interprétation combinéesMay 08, 2025 am 12:16 AM

Pythonusesahybridapproach, combinantcompilationToByteDodeAnd Intrepretation.1) CodeSompiledToplatForment-indépendantBytecode.2) ByteCodeisInterpretedByThepyThonVirtualmachine, améliorant la performance et la portabilité.

Apprenez les différences entre les 'pour' de PythonApprenez les différences entre les 'pour' de PythonMay 08, 2025 am 12:11 AM

Thekeydifferencesbetweenpython "pour" et "tandis que" Loopsare: 1) "pour" LoopsareIdEalForitatriant sur les séquences ouvraires, tandis que 2) "tandis que" LoopsarebetterforcontinUnUntilaconditionMetStwithoutPredefinedIberations.un.un

Python concaténate répertorie avec des doublonsPython concaténate répertorie avec des doublonsMay 08, 2025 am 12:09 AM

Dans Python, vous pouvez connecter des listes et gérer des éléments en double via une variété de méthodes: 1) Utiliser les opérateurs ou prolonger () pour conserver tous les éléments en double; 2) Convertissez en ensembles puis revenez aux listes pour supprimer tous les éléments en double, mais l'ordre d'origine sera perdu; 3) Utilisez des boucles ou des compréhensions de liste pour combiner des ensembles pour supprimer les éléments en double et maintenir l'ordre d'origine.

See all articles

Outils d'IA chauds

Undresser.AI Undress

Undresser.AI Undress

Application basée sur l'IA pour créer des photos de nu réalistes

AI Clothes Remover

AI Clothes Remover

Outil d'IA en ligne pour supprimer les vêtements des photos.

Undress AI Tool

Undress AI Tool

Images de déshabillage gratuites

Clothoff.io

Clothoff.io

Dissolvant de vêtements AI

Video Face Swap

Video Face Swap

Échangez les visages dans n'importe quelle vidéo sans effort grâce à notre outil d'échange de visage AI entièrement gratuit !

Outils chauds

SublimeText3 version Mac

SublimeText3 version Mac

Logiciel d'édition de code au niveau de Dieu (SublimeText3)

SublimeText3 Linux nouvelle version

SublimeText3 Linux nouvelle version

Dernière version de SublimeText3 Linux

ZendStudio 13.5.1 Mac

ZendStudio 13.5.1 Mac

Puissant environnement de développement intégré PHP

SublimeText3 version anglaise

SublimeText3 version anglaise

Recommandé : version Win, prend en charge les invites de code !

Navigateur d'examen sécurisé

Navigateur d'examen sécurisé

Safe Exam Browser est un environnement de navigation sécurisé permettant de passer des examens en ligne en toute sécurité. Ce logiciel transforme n'importe quel ordinateur en poste de travail sécurisé. Il contrôle l'accès à n'importe quel utilitaire et empêche les étudiants d'utiliser des ressources non autorisées.