Вывод таблицы истинности в PHP
Я наткнулся на этот сайт генератора таблиц правды и попытался имитировать его в PHP (я понимаю, что исходный код доступен, но я знаю 0 perl).
Теперь мой вопрос не об оценке выражения, а о том, как вывести таблицу, чтобы показать каждую комбинацию T и F для переменных.
Например, с 3 переменными таблица будет выглядеть так:
a | b | c
-----------
T | T | T
T | T | F
T | F | T
T | F | F
F | T | T
F | T | F
F | F | T
F | F | F
и с 4 переменными..
a | b | c | d
-------------
T | T | T | T
T | T | T | F
T | T | F | T
T | T | F | F
T | F | T | T
T | F | T | F
T | F | F | T
T | F | F | F
F | T | T | T
F | T | T | F
F | T | F | T
F | T | F | F
F | F | T | T
F | F | T | F
F | F | F | T
F | F | F | F
Какова логика / шаблон для его создания?
7 ответов
Как насчет этой рекурсивной функции? Возвращает двумерный массив, в котором каждая строка имеет $count
элементы. Вы можете использовать это для генерации вашей таблицы.
function getTruthValues($count) {
if (1 === $count) {
// true and false for the first variable
return array(array('T'), array('F'));
}
// get 2 copies of the output for 1 less variable
$trues = $falses = getTruthValues(--$count);
for ($i = 0, $total = count($trues); $i < $total; $i++) {
// the true copy gets a T added to each row
array_unshift($trues[$i], 'T');
// and the false copy gets an F
array_unshift($falses[$i], 'F');
}
// combine the T and F copies to give this variable's output
return array_merge($trues, $falses);
}
function toTable(array $rows) {
$return = "<table>\n";
$headers = range('A', chr(64 + count($rows[0])));
$return .= '<tr><th>' . implode('</th><th>', $headers) . "</th></tr>\n";
foreach ($rows as $row) {
$return .= '<tr><td>' . implode('</td><td>', $row) . "</td></tr>\n";
}
return $return . '</table>';
}
echo toTable(getTruthValues(3));
РЕДАКТИРОВАТЬ: Codepad, с добавленной функцией для преобразования массива в таблицу.
$nbBooleans = 5; // change to whatever you want
// show header
for($i = 0; $i < $nbBooleans ; $i++)
{
if ($i > 0)
echo " | ";
echo chr($i + 65); // 1 => A, 2 => B etc.
}
// separator line (dynamic size)
echo "\n".str_repeat("-", ($nbBooleans - 1) * 3 + $nbBooleans)."\n";
// show combinations
$nbInterations = pow(2, $nbBooleans);
for($i = 0; $i < $nbInterations; $i++)
{
for ($iBit = 0; $iBit < $nbBooleans; $iBit++)
{
if ($iBit > 0)
echo " | ";
echo (($i & pow(2, $iBit)) != 0 ? 'Y' : 'N');
}
echo "\n";
}
Этот подход может быть некрасивым, но он кажется универсальным:)
$length = 3;
for($i=0;$i<pow(2,$length);$i++){
$bin = decbin($i);
_add($bin, $length);
_out($bin, $length);
}
function _out($str, $length){
for ($i=0; $i<$length; $i++)
echo ($str[$i] == 0 ? 'F' : 'T')."\t";
echo "\n";
}
function _add(&$bin, $length){
$add = '';
if (strlen($bin) < $length){
for($j=0;$j<($length - strlen($bin));$j++){
$add.='0';
}
$bin = $add.$bin;
}
}
вывод выглядит следующим образом
F F F
F F T
F T F
F T T
T F F
T F T
T T F
T T T
Сделал небольшую функцию для этого:
function printTruth($vars) {
$rows = array();
$max = pow(2, $vars);
// Lines and Letters
$arr = array();
for($i=97;$i<(97+$vars);$i++) { $arr[] = chr($i); }
$rows[] = implode(' | ', $arr);
$rows[] = str_repeat('-', $vars*3);
// Variables
for($i=0;$i<$max;$i++) {
$oneRow = '';
for ($j=0;$j<$vars;$j++) {
if($j>0)
$oneRow .= " | ";
$oneRow .= (($i & pow(2,$j)) != 0 ? 'T' : 'F');
}
$rows[] = strrev($oneRow);
}
return implode("<br>", $rows);
}
echo printTruth(3);
Сначала вы должны вычислить количество комбинаций, например, если у вас есть 4 переменные, то у вас будет 16 комбинаций. Для каждой переменной взять свой индекс, назовите его i. В столбце этой переменной у вас будет (n / (2^i)) время переключения групп T и F, где n - суммарные комбинации возможных значений переменных.
<?php
$numberOfVariables = 5;
$totalCombinations = pow(2, $numberOfVariables);
for ($i = 0; $i < $numberOfVariables; $i++)
{
$subGroupCount = $totalCombinations / pow(2, $i);
$lettersPerGroup = $totalCombinations / $subGroupCount;
$toggler = true;
for($j=0; $j<$subGroupCount; $j++)
{
for($k=0; $k < $lettersPerGroup; $k++)
$array[$i][$j*$lettersPerGroup + $k] = ($toggler ? "T" : "F");
$toggler = !$toggler;
}
}
echo("<table border='1' bgcolor='yellow'>");
for ($k=0;$k<$totalCombinations;$k++)
{
echo("<tr>");
for ($j=$numberOfVariables-1;$j>=0;$j--)
echo("<td>".$array[$j][$k]."</td>");
echo("</tr>");
}
echo("</table>");
?>
Попробуйте этот код. generate($numberOfVariables) возвращает массив с таблицей истинности. Каждый элемент представляет собой строку, которую вы можете перебрать
<?
function generate($var=3){
$number= pow(2,$var)-1;
$array=array();
while($number>=0){
$str=decbin($number);
$number--;
while(strlen($str)<$var)
{
$str="0".$str;
}
array_push($array,$str);
}
return $array;
}?>
<pre>
<?print_r(array_reverse(generate(3)));
?>
</pre>
Вы можете сделать что-то вроде этого:
echo "<table>";
for($a=0; $a<=1; $a++){
for($b=0; $b<=1; $b++){
for($c=0; $c<=1; $c++){
for($d=0; $d<=1; $d++){
echo "<tr><td>$a</td><td>$b</td><td>$c</td><td>$d</td></tr>";
}
}
}
}
echo "</table>";
может быть, немного излишним, но это работает...