Heim >Backend-Entwicklung >Golang >Eine einfache Go-Simulation – Probleme mit der Parallelität
php-Editor Xinyi bietet Ihnen ein einfaches, aber interessantes Go-Simulationsspiel namens „Concurrency Problem“. Dieses Spiel thematisiert die gleichzeitige Programmierung und ermöglicht es den Spielern, den Charme der gleichzeitigen Programmierung in einer virtuellen Welt zu erleben. Im Spiel müssen die Spieler Code schreiben, um die gleichzeitige Ausführung mehrerer Aufgaben zu bewältigen und ihre gleichzeitigen Programmierfähigkeiten zu testen. Die Spieloberfläche ist prägnant und klar, die Bedienung ist einfach, für Anfänger geeignet und bietet den Spielern außerdem mehrere Schwierigkeits- und Herausforderungsmodi zur Auswahl. Egal, ob Sie Anfänger oder erfahrener Entwickler sind, in diesem Simulationsspiel können Sie den Spaß am gleichzeitigen Programmieren genießen.
Ich bin ein Student aus Polen und habe dieses Semester einen parallelen Programmierkurs begonnen (Go, Ada und in Zukunft einige theoretische und CSP-Sprachen). Um ehrlich zu sein, sieht Golang interessant aus, aber ich bin etwas verwirrt. Unterm Strich würde ich mich selbst als unterdurchschnittlichen Programmierer bezeichnen. Grundsätzlich besteht meine Aufgabe darin, eine Simulation zu erstellen, die ich wie folgt beschreiben werde:
Es gibt ein n*m-Raster
Reisende können zufällig generiert werden, bis zu k Reisende, jeder Reisende hat eine eindeutige ID (1, 2, 3 usw., bis zu k)
Wenn der Platz frei ist (ich bin sicher, dass der freie Platz 0 ist), kann sich der Reisende zu zufälligen Zeitpunkten im Raster nach oben, links, rechts oder unten bewegen
Verfügt außerdem über eine Kamera, die manchmal den aktuellen Zustand des Netzes sowie aktuelle Bewegungen ausdruckt (noch nicht implementiert)
Inoffiziell habe ich gehört, dass ich Kanäle verwenden sollte, was auch immer das bedeutet
Meine Idee ist es, eine Struktur mit der ID und den Koordinaten jedes Reisenden zu erstellen und seine ID an den Kanal zu senden, der die Bereitschaft zur Bewegung darstellt, und dann werde ich zufällig die Bewegungsrichtung auswählen.
Ich bin etwas verwirrt über Parallelität – nicht nur, ob und wo ich WGs und Mutexe verwenden soll, sondern auch, ob ich zum Beispiel func(){} gehe, ob die Schleife innerhalb oder außerhalb sein soll. Ich würde mich sehr über Tipps, Hilfe oder Korrekturen/Ideen zur Korrektur meines Codes freuen, da er derzeit, wie Sie vermutet haben, nicht richtig funktioniert (z. B. wenn die Kamera das Raster druckt, gibt es manchmal mehr als k Reisende, wo mehrere). Reisende teilen sich die gleiche Nummer und manchmal scheinen sie zu verschwinden). Ich hoffe, alle haben einen tollen Tag und ich würde mich wirklich über jede Hilfe freuen :)
package main; import( "fmt" "os" "strconv" "math/rand" //"sync" "time" ) type traveler struct{ id int; x int; y int; } func main(){ //command line n, err := strconv.Atoi(os.Args[1]); m, err := strconv.Atoi(os.Args[2]); k, err := strconv.Atoi(os.Args[3]); if err != nil{ panic(err) return } //board var grid [][]int; grid = make([][]int, n) for i:=0; i<n; i++{ grid[i] = make([]int, m) } //array of travelers, channel for moves and id travelers := make([]traveler, k) no_of_travelers := 0; move := make(chan int, k); id := 1; //var wg sync.WaitGroup go camera(grid); go func() { for i:=0; i<len(travelers); i++ { if no_of_travelers<k{ travelers[i] = spawnTraveler(&id,grid); no_of_travelers++; } } }() go func() { for{ a:= rand.Intn(k); sendMoveTraveler(&travelers[a], move); } }() receiveMoveTraveler(travelers, move, grid); } func spawnTraveler(id *int, grid [][]int) traveler{ x:=-1; y:=-1; for{ x = rand.Intn(len(grid)); y = rand.Intn(len(grid)); if(grid[x][y]==0){ break; } } t := traveler{id: *id, x: x, y:y}; grid[x][y] = *id; *id++; return t; } func sendMoveTraveler(t *traveler, move chan int){ move <- t.id } func receiveMoveTraveler(travelers []traveler, move chan int, grid [][]int){ for{ id := <- move for i:=0; i<len(travelers); i++{ if travelers[i].id == id{ direction := rand.Intn(4); //1-left 2-up 3-right 4-down switch direction { case 0: if travelers[i].x>0 && grid[travelers[i].x-1][travelers[i].y] == 0{ grid[travelers[i].x-1][travelers[i].y] = grid[travelers[i].x][travelers[i].y]; grid[travelers[i].x][travelers[i].y] = 0; travelers[i].x = travelers[i].x-1; travelers[i].y = travelers[i].y; } case 1: if travelers[i].y>0 && grid[travelers[i].x][travelers[i].y-1] == 0{ grid[travelers[i].x][travelers[i].y-1] = grid[travelers[i].x][travelers[i].y]; grid[travelers[i].x][travelers[i].y] = 0; travelers[i].x = travelers[i].x; travelers[i].y = travelers[i].y-1; } case 2: if travelers[i].x<len(grid)-1 && grid[travelers[i].x+1][travelers[i].y] == 0{ grid[travelers[i].x+1][travelers[i].y] = grid[travelers[i].x][travelers[i].y]; grid[travelers[i].x][travelers[i].y] = 0; travelers[i].x = travelers[i].x+1; travelers[i].y = travelers[i].y; } case 3: if travelers[i].y<len(grid)-1 && grid[travelers[i].x][travelers[i].y+1] == 0{ grid[travelers[i].x][travelers[i].y+1] = grid[travelers[i].x][travelers[i].y]; grid[travelers[i].x][travelers[i].y] = 0; travelers[i].x = travelers[i].x; travelers[i].y = travelers[i].y+1; } } //fmt.Println("Ściagnalem ruch", travelers[i].id); } } } } func camera(grid [][]int){ for{ for i:=0; i<len(grid); i++{ for j:=0; j<len(grid); j++{ if grid[i][j]!= 0{ fmt.Printf("%02d ", grid[i][j]); } else{ fmt.Printf("-- "); } } fmt.Println(); } fmt.Println(); time.Sleep(time.Second * 3); } }
Ich bin ein wenig überwältigt von all den Ideen – WGs, Mutexe, Atome usw.
Davon abgesehen:
Das obige ist der detaillierte Inhalt vonEine einfache Go-Simulation – Probleme mit der Parallelität. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!