C++: Az n-királynő probléma brute force megoldása

Programozással kapcsolatos cikkek / C, C++ (1618 katt)

A feladat

Az n-királynő probléma a következő: helyezzünk el n db királynőt egy (általánosított) n x n-es sakktáblán úgy, hogy a királynők ne üssék egymást. Ez a sakk szabályai szerint azt jelenti, hogy két királynő nem állhat a tábla azonos során vagy oszlopán, illetve azonos átlón.

A feladat megoldása jelentheti azt, hogy keressünk egy konkrét jó elhelyezést, illetve azt is, hogy keressük meg az összes lehetséges elhelyezést.

Reprezentáció

Az n-királynő problémát többféle módon is lehet reprezentálni.

Az első módszer szerint egy n x n-es kétdimenziós tömböt használunk, ahol a tomb[i, j] helyen található 1 érték jelzi azt, hogy az i. sor j. oszlopában van királynő, a 0 pedig azt, hogy nincs. Ebben az esetben ha minden lehetséges állást meg szeretnénk vizsgálni, akkor nagyon nagy számot kapnánk. A 8 királynő esetén például ( 64 8 ) = 4 426 165 368 ( 64 alatt a 8 ) esetet kellene végignézni.

Ha felhasználjuk azt a tényt, hogy két királynő sosem állhat egy oszlopban, akkor egy egydimenziós, n elemű tömb is megfelel, amelyben az egyes számok azt jelentik, hogy a tömb indexének megfelelő oszlop hányadik sorában áll egy királynő. Ha minden esetet meg szeretnénk vizsgálni a 8 királynő esetén, akkor már csak 8^8 = 16 777 216 eset jönne szóba. Ez az ábrázolási mód a visszalépéses keresés esetén használható, mely esetben az algoritmus jelentősen csökkenti a megvizsgált állások számát.

A harmadik lehetséges módszer esetében azt a tényt is figyelembe vesszük, hogy két királynő sem azonos oszlopban, sem azonos sorban nem állhat. Ekkor egy egydimenziós, n elemű tömböt használunk, amelyben az egyes számok az előző esethez hasonlóan azt jelentik, hogy a tömb indexének megfelelő oszlop hányadik sorában áll egy királynő, viszont a tömb mindig az 1..n számok egy permutációját tartalmazza. Ha az összes esetet meg szeretnénk vizsgálni, amely a mai számítógépek esetén kis n esetén elfogadható időn belül megvalósítható, akkor ezt az ábrázolást célszerű használni. A 8 királynő esetén a megvizsgált állások száma így már csak 8! = 40 320 darab lesz.

A helyes megoldások száma

A program futtatása során az n különböző értékeire a következő megoldás darabszámokat kaptam:


Királynők száma (n) Megoldások száma
1 1
2 0
3 0
4 2
5 10
6 4
7 40
8 92
9 352
10 724
11 2680



Az elkészített program

A feladat megoldására a Microsoft Visual Studio 2015-ös verzióját és a C++ nyelvet használtam. A program az összes lehetséges permutáció vizsgálatát végzi el. Ebben az esetben szükség van a szóba jövő permutációk előállítására. A C++ nyelv használata esetén erre létezik kész megoldás (algorithm, next_permutation), amit fel is használtam a programban.

A program az indítása után bekéri a felhasználótól az n értékét, amely a képernyőre kiírt határok közé eshet. Ezután az adott programban megvalósított módszerrel megkeresi a feladat összes lehetséges megoldását, és ezeket kiírja a képernyőre.

A kiírás formátuma:

1 3 5 2 4

Ez azt jelenti, hogy az első oszlop első sorában, majd a második oszlop harmadik sorában, az ötödik oszlop ötödik sorában stb. áll egy-egy királynő.

Miután befejeződött a keresés, a program megkérdezi a felhasználót, hogy akar-e újabb keresést végezni. Ha i-vel válaszolunk, akkor egy újabb n értéket adhatunk meg, egyébként a program kilép.

A program forráskódja

#include <iostream>
#include <algorithm>
#include <string>

using namespace std;

const int MAXN = 11;

typedef int Tabla[MAXN];

void Koszonto()
{
cout << "Az N királynő probléma megoldása az összes eset vizsgálatával"
<< endl;
cout << "-------------------------------------------------------------"
<< endl;
cout << "Készítette: Török Viktor" << endl;
cout << endl;
}

int MeretBekerese()
{
int n = 0;

do
{
cout << "Kérem, adja meg n értékét (1 - " << (MAXN - 1) << "): ";
cin >> n;

if (cin.fail())
{
cin.clear();
cin.ignore(numeric_limits<int>::max(), '\n');
}
} while (n < 1 || n >= MAXN);

return n;
}

bool KovetkezoEset()
{
string kov;
cout << "Következő eset (i / n): ";
cin >> kov;

return kov == "i" || kov == "I";
}

void TablaInicializalasa(Tabla tabla, int n)
{
for (int i = 0; i < n; i++)
{
tabla[ i ] = i;
}
}

void TablaKiirasa(Tabla tabla, int n)
{
for (int i = 0; i < n; i++)
{
cout << (tabla[ i ] + 1) << " ";
}

cout << endl;
}

bool Megoldas(Tabla tabla, int n)
{
for (int i = 0; i < n - 1; i++)
{
for (int j = i + 1; j < n; j++)
{
if ((j - i) == abs(tabla[j] - tabla[ i ]))
{
return false;
}
}
}

return true;
}

void MegoldasokKeresese(Tabla tabla, int n)
{
int darab = 0;

cout << "A megoldások:" << endl;

do
{
if (Megoldas(tabla, n))
{
TablaKiirasa(tabla, n);
darab++;
}
} while (next_permutation(tabla, tabla + n));

cout << "---------------------------" << endl;
cout << "A megoldások száma " << n << " királynő esetén: " << darab
<< endl;
}

int main()
{
setlocale(LC_ALL, "");

Tabla tabla;

Koszonto();

do
{
int n = MeretBekerese();

TablaInicializalasa(tabla, n);
MegoldasokKeresese(tabla, n);
} while (KovetkezoEset());

return 0;
}


Előző oldal Kapitány