mirror of
https://github.com/pmmp/PocketMine-MP.git
synced 2025-05-19 04:05:31 +00:00
187 lines
5.0 KiB
PHP
187 lines
5.0 KiB
PHP
<?php
|
|
|
|
/**
|
|
*
|
|
* ____ _ _ __ __ _ __ __ ____
|
|
* | _ \ ___ ___| | _____| |_| \/ (_)_ __ ___ | \/ | _ \
|
|
* | |_) / _ \ / __| |/ / _ \ __| |\/| | | '_ \ / _ \_____| |\/| | |_) |
|
|
* | __/ (_) | (__| < __/ |_| | | | | | | | __/_____| | | | __/
|
|
* |_| \___/ \___|_|\_\___|\__|_| |_|_|_| |_|\___| |_| |_|_|
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* @author PocketMine Team
|
|
* @link http://www.pocketmine.net/
|
|
*
|
|
*
|
|
*/
|
|
|
|
class Matrix implements ArrayAccess{
|
|
private $matrix = array();
|
|
private $rows = 0;
|
|
private $columns = 0;
|
|
|
|
public function offsetExists($offset){
|
|
return isset($this->matrix[(int) $offset]);
|
|
}
|
|
|
|
public function offsetGet($offset){
|
|
return $this->matrix[(int) $offset];
|
|
}
|
|
|
|
public function offsetSet($offset, $value){
|
|
$this->matrix[(int) $offset] = $value;
|
|
}
|
|
|
|
public function offsetUnset($offset){
|
|
unset($this->matrix[(int) $offset]);
|
|
}
|
|
|
|
public function __construct($rows, $columns, array $set = array()){
|
|
$this->rows = max(1, (int) $rows);
|
|
$this->columns = max(1, (int) $columns);
|
|
$this->set($set);
|
|
}
|
|
|
|
public function set(array $m){
|
|
for($r = 0; $r < $this->rows; ++$r){
|
|
$this->matrix[$r] = array();
|
|
for($c = 0; $c < $this->columns; ++$c){
|
|
$this->matrix[$r][$c] = isset($m[$r][$c]) ? $m[$r][$c] : 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
public function getRows(){
|
|
return ($this->rows);
|
|
}
|
|
|
|
public function getColumns(){
|
|
return ($this->columns);
|
|
}
|
|
|
|
public function setElement($row, $column, $value){
|
|
if($row > $this->rows or $row < 0 or $column > $this->columns or $column < 0){
|
|
return false;
|
|
}
|
|
$this->matrix[(int) $row][(int) $column] = $value;
|
|
return true;
|
|
}
|
|
|
|
public function getElement($row, $column){
|
|
if($row > $this->rows or $row < 0 or $column > $this->columns or $column < 0){
|
|
return false;
|
|
}
|
|
return $this->matrix[(int) $row][(int) $column];
|
|
}
|
|
|
|
public function isSquare(){
|
|
return $this->rows === $this->columns;
|
|
}
|
|
|
|
public function add(Matrix $matrix){
|
|
if($this->rows !== $matrix->getRows() or $this->columns !== $matrix->getColumns()){
|
|
return false;
|
|
}
|
|
$result = new Matrix($this->rows, $this->columns);
|
|
for($r = 0; $r < $this->rows; ++$r){
|
|
for($c = 0; $c < $this->columns; ++$c){
|
|
$result->setElement($r, $c, $this->matrix[$r][$c] + $matrix->getElement($r, $c));
|
|
}
|
|
}
|
|
return $result;
|
|
}
|
|
|
|
public function substract(Matrix $matrix){
|
|
if($this->rows !== $matrix->getRows() or $this->columns !== $matrix->getColumns()){
|
|
return false;
|
|
}
|
|
$result = clone $this;
|
|
for($r = 0; $r < $this->rows; ++$r){
|
|
for($c = 0; $c < $this->columns; ++$c){
|
|
$result->setElement($r, $c, $this->matrix[$r][$c] - $matrix->getElement($r, $c));
|
|
}
|
|
}
|
|
return $result;
|
|
}
|
|
|
|
public function multiplyScalar($number){
|
|
$result = clone $this;
|
|
for($r = 0; $r < $this->rows; ++$r){
|
|
for($c = 0; $c < $this->columns; ++$c){
|
|
$result->setElement($r, $c, $this->matrix[$r][$c] * $number);
|
|
}
|
|
}
|
|
return $result;
|
|
}
|
|
|
|
|
|
public function divideScalar($number){
|
|
$result = clone $this;
|
|
for($r = 0; $r < $this->rows; ++$r){
|
|
for($c = 0; $c < $this->columns; ++$c){
|
|
$result->setElement($r, $c, $this->matrix[$r][$c] / $number);
|
|
}
|
|
}
|
|
return $result;
|
|
}
|
|
|
|
public function transpose(){
|
|
$result = new Matrix($this->columns, $this->rows);
|
|
for($r = 0; $r < $this->rows; ++$r){
|
|
for($c = 0; $c < $this->columns; ++$c){
|
|
$result->setElement($c, $r, $this->matrix[$r][$c]);
|
|
}
|
|
}
|
|
return $result;
|
|
}
|
|
|
|
//Naive Matrix product, O(n^3)
|
|
public function product(Matrix $matrix){
|
|
if($this->columns !== $matrix->getRows()){
|
|
return false;
|
|
}
|
|
$c = $matrix->getColumns();
|
|
$result = new Matrix($this->rows, $c);
|
|
for($i = 0; $i < $this->rows; ++$i){
|
|
for($j = 0; $j < $c; ++$j){
|
|
$sum = 0;
|
|
for($k = 0; $k < $this->columns; ++$k){
|
|
$sum += $this->matrix[$i][$k] * $matrix->getElement($k, $j);
|
|
}
|
|
$result->setElement($i, $j, $sum);
|
|
}
|
|
}
|
|
return $result;
|
|
}
|
|
|
|
|
|
//Computation of the determinant of 2x2 and 3x3 matrices
|
|
public function determinant(){
|
|
if($this->isSquare() !== true){
|
|
return false;
|
|
}
|
|
switch($this->rows){
|
|
case 1:
|
|
return 0;
|
|
case 2:
|
|
return $this->matrix[0][0] * $this->matrix[1][1] - $this->matrix[0][1] * $this->matrix[1][0];
|
|
case 3:
|
|
return $this->matrix[0][0] * $this->matrix[1][1] * $this->matrix[2][2] + $this->matrix[0][1] * $this->matrix[1][2] * $this->matrix[2][0] + $this->matrix[0][2] * $this->matrix[1][0] * $this->matrix[2][1] - $this->matrix[2][0] * $this->matrix[1][1] * $this->matrix[0][2] - $this->matrix[2][1] * $this->matrix[1][2] * $this->matrix[0][0] - $this->matrix[2][2] * $this->matrix[1][0] * $this->matrix[0][1];
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|
|
public function __toString(){
|
|
$s = "";
|
|
for($r = 0; $r < $this->rows; ++$r){
|
|
$s .= implode(",", $this->matrix[$r]).";";
|
|
}
|
|
return "Matrix({$this->rows}x{$this->columns};".substr($s, 0, -1).")";
|
|
}
|
|
|
|
} |