Robots Musterlösung
Links
Zielsetzung
Dies ist eine funktionierende Lösung der Robotersimulaltion.
<?php
define('DEFAULT_COLOR', "\t");
define('WHITE', ' ');
/**
* Implements a sheet area with ascii sheet.
*/
class Sheet
{
public array $lines;
public int $width;
public int $height;
public function __construct(int $width = 80, int $height = 40)
{
$this->width = $width;
$this->height = $height;
$this->lines = [];
for ($y = 0; $y < $this->height; $y++) {
array_push($this->lines, str_repeat(' ', $this->width));
}
}
public function clearTerminal()
{
echo "\x1b\x5b\x48\x1b\x5b\x32\x4a\xc";
}
public function setRandomPosition(&$point)
{
do {
$x = rand(0, $this->width - 1);
$y = rand(0, $this->height - 1);
} while ($this->isOccupied($x, $y));
$point->x = $x;
$point->y = $y;
}
public function show()
{
$this->clearTerminal();
echo str_repeat('-', $this->width + 2), "\n";
for ($y = $this->height - 1; $y >= 0; $y--) {
echo '|', $this->lines[$y], "|\n";
}
echo str_repeat('-', $this->width + 2), "\n";
}
public function setPoint(int $x, int $y, string $color)
{
if ($x >= 0 && $x < $this->width && $y >= 0 && $y < $this->height) {
$this->lines[$y][$x] = $color;
}
}
public function isOccupied(int $x, int $y): bool
{
$color = $this->lines[$y][$x];
return $color !== WHITE && $color < "a";
}
}
class Point
{
public int $x;
public int $y;
public string $color;
public function __construct(int $x, int $y, string $color)
{
$this->x = $x;
$this->y = $y;
$this->color = $color;
}
public function draw(Sheet &$sheet, string $color = DEFAULT_COLOR)
{
if ($color == DEFAULT_COLOR) {
$color = $this->color;
}
$sheet->setPoint($this->x, $this->y, $color);
}
}
class Rectangle extends Point
{
public $width;
public $height;
public function __construct(int $x, int $y, int $width, int $height, string $color)
{
parent::__construct($x, $y, $color);
$this->width = $width;
$this->height = $height;
}
public function draw(Sheet &$sheet, string $color = DEFAULT_COLOR)
{
if ($color === DEFAULT_COLOR) {
$color = $this->color;
}
for ($x = $this->x; $x < $this->x + $this->width; $x++) {
for ($y = $this->y; $y <= $this->y + $this->height; $y++) {
$sheet->setPoint($x, $y, $color);
}
}
}
}
class Robot extends Point
{
public $target;
public string $colorTarget;
public int $xDirection = 1;
public int $yDirection = 1;
public function __construct(string $color, Sheet &$sheet)
{
parent::__construct(0, 0, $color);
$this->target = null;
$this->color = $color;
$this->colorTarget = strtolower($color);
$sheet->setRandomPosition($this);
$this->setTarget($sheet);
}
public function checkDirection(Sheet &$sheet)
{
if ($this->x <= 0) {
$this->xDirection = 1;
} elseif ($this->x >= $sheet->width - 1) {
$this->xDirection = -1;
}
if ($this->y <= 0) {
$this->yDirection = 1;
} elseif ($this->y >= $sheet->height - 1) {
$this->yDirection = -1;
}
}
public function move(Sheet &$sheet): bool
{
$rc = false;
if ($this->target->x == $this->x && $this->target->y == $this->y) {
// $sheet->setRandomPosition($this->target, $sheet);
} else {
$this->target->draw($sheet);
$x = $this->x;
$y = $this->y;
if (abs($x - $this->target->x) <= 1 && abs($y - $this->target->y) <= 1) {
$x = $this->target->x;
$y = $this->target->y;
} else {
$x = $this->target->x < $x ? $x - 1 : $x + 1;
$y = $this->target->y < $y ? $y - 1 : $y + 1;
}
$this->checkDirection($sheet);
if ($sheet->isOccupied($x, $y)) {
$x = $this->x + $this->xDirection;
$y = $this->y;
if ($sheet->isOccupied($x, $y)) {
$x = $this->x;
$y = $this->y + $this->yDirection;
}
}
if (!$sheet->isOccupied($x, $y)) {
// Remove current point:
$this->draw($sheet, WHITE);
$this->x = $x;
$this->y = $y;
$this->draw($sheet);
$rc = true;
}
}
return $rc;
}
public function setTarget(Sheet &$sheet)
{
if ($this->target == null) {
$this->target = new Point(0, 0, $this->colorTarget);
} else {
$sheet->setPoint($this->target->x, $this->target->y, WHITE);
}
$sheet->setRandomPosition($this->target);
$sheet->setPoint($this->target->x, $this->target->y, $this->colorTarget);
}
}
class Simulator
{
public array $statics = [];
public array $mobiles = [];
public Sheet $sheet;
public $stepNo = 0;
public function __construct(Sheet &$sheet)
{
$this->sheet = $sheet;
}
public function add($item, bool $isStatic)
{
if ($isStatic) {
array_push($this->statics, $item);
$item->draw($this->sheet);
} else {
array_push($this->mobiles, $item);
$item->draw($this->sheet);
}
}
public function buildBarriers(int $rows = 5, int $cols = 5)
{
$width = ($this->sheet->width - 1) / $rows / 2;
$height = ($this->sheet->height - 1) / $cols / 2;
for ($row = 0; $row < $rows; $row++) {
for ($col = 0; $col < $cols; $col++) {
$x = $width / 2 + $row * 2 * $width;
$y = $height / 2 + $col * 2 * $height;
$item = new Rectangle($x, $y, $width, $height, '*');
$this->add($item, true);
}
}
}
public function buildRobots(int $count)
{
$names = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
for ($ix = 0; $ix < $count; $ix++) {
$car = new Robot($names[$ix], $this->sheet);
$this->add($car, false);
}
}
public function show()
{
$this->sheet->clearTerminal();
$this->sheet->show();
}
public function step()
{
$this->stepNo++;
$barrier = $this->statics[0];
if ($this->stepNo % ($barrier->width * 2) == 0) {
foreach ($this->mobiles as $robot) {
$robot->xDirection = -$robot->xDirection;
}
}
if ($this->stepNo % ($barrier->height * 2) == 0) {
foreach ($this->mobiles as $robot) {
$robot->yDirection = -$robot->yDirection;
}
}
$moves = 0;
foreach ($this->mobiles as $car) {
if ($car->move($this->sheet)) {
$moves++;
}
}
return $moves > 0;
}
}
//srand(44);
$sheet = new Sheet(30, 30);
$simulator = new Simulator($sheet);
$simulator->buildBarriers(2, 2);
$simulator->buildRobots(12);
$simulator->show();
$rounds = 0;
while ($simulator->step()) {
$rounds++;
$simulator->show();
sleep(1);
}
echo "== ready after $rounds rounds\n";