|
Este aviso fue puesto el 25 de junio de 2017. |
El Algoritmo de Schonhage-Strassen es un algoritmo que consiste en la multiplicación de matrices. Es asintóticamente más rápido que el algoritmo de multiplicación de matrices estándar, pero más lento que el algoritmo más rápido conocido, y es útil en la práctica para matrices grandes.
Historia
El Algoritmo de Schonhage-Strassen fue el primer algoritmo en bajar en tiempo de ejecución de ε=1 , este fue descubierto en 1969 , por Volker Strassen de aquí el nombre del algoritmo. Este algoritmo logró alcanzar ε=0,808 en su tiempo de ejecución .
Desde ese momento muchos matemáticos han intentado hacer a este algoritmo más rápido para su mejora a nivel matemático y para ganar fama . Un gran salto fue obtenido en 1981 por Schönhage que logró ε=0,548 ,; Desde entonces la evolución de este es debida principalmente a Schönhage y a Strassen , ya que con sus investigaciones durante muchos años fueron reduciendo este tiempo y dando un gran paso para las matemáticas.
Análisis
El método clásico requiere n²(2n-1) operaciones aritméticas para multiplicar dos matrices (n x n). Strassen (1969) publicó un método que solo requería operaciones. Se ha realizado mucho trabajo para intentar reducir el exponente 2,81, actualmente el mejor tiempo conocido es operaciones (Virginia Vassilovska Williams). Hay todavía un largo camino por recorrer, pues la mejor cota inferior conocida es .
En lo que es la multiplicación es una de las operaciones que debemos repetir varias veces en enteros grandes, por lo que no está de más comprobar que la multiplicación por el método estándar de dos enteros binarios de longitudes k y precisa un máximo k' suma de un bit(Operaciones bit), por lo tanto el producto de dos números enteros con a lo sumo r cifras decimales requiere a lo más log² r operaciones. Es decir, es de orden O(log² r) y por lo tanto es una operación eficiente. Actualmente hay algoritmos de multiplicación más rápidos, el más conocido es el método de Karatsuba (cuya descripción es completamente accesible) que permite multiplicar dos enteros de longitud binaria menor o igual que k en O(k¹·59) operaciones bit. El método más rápido se debe a Schonhage y Strassen y su complejidad temporal es O(k log log log k).
Algoritmo
Utilizando el algoritmo de Strassen definimos las siguientes matrices:
M1=(A1,2 - A2,2).(B2,1 + B2,2)
M2=(A1,1 + A2,2).(B1,1 + B2,2)
M3=(A1,1 - A2,1).(B1,1 + B2,1)
M4=(A1,1 + A1,2).B2,2
M5=A1,1 .(B1,2 -B2,2)
M6=A2,2.(B2,1 - B1,1)
M7=(A2,1 + A2,2).B1,1
Algoritmo en C
Ya que todos estos procesos son de una complejidad alta , habría que realizarlos en un computador. Por lo tanto habría que hacerlo con un lenguaje de programación como puede ser C, Java, Python... En este caso nos centramos en C++ por lo que el algoritmo es el siguiente:
void strassen ( vector < vector < int > > Y Un ,
vector < vector < int > > y B ,
vector < vector < int > > y C , sin firmar int tam );
unsigned int nextPowerOfTwo ( int n );
void strassenR ( vector < vector < int > > Y Un ,
vector < vector < int > > y B ,
vector < vector < int > > y C ,
int tam );
void suma ( vectorial < vector < int > > Y Un ,
vector < vector < int > > y B ,
vector < vector < int > > y C , int tam );
vacío de resta ( vector < vector < int > > Y Un ,
vector < vector < int > > y B ,
vector < vector < int > > y C , int tam );
void printMatrix ( vector < vector < int > > matriz , int n );
vacío de lectura ( cadena de nombre de archivo , vector < vector < int > > Y Un , vector < vector < int > > y B );
void ikjalgorithm ( vector < vector < int > > A ,
vector < vector < int > > B ,
vector < vector < int > > y C , int n ) {
a ( int i = 0 ; i < n ; i ++ ) {
a ( int k = 0 ; k < n ; k ++ ) {
a ( int j = 0 ; j < n ; j ++ ) {
C [ i ] [ j ] + = a [ i ] [ k ] * B [ k ] [ j ];
}
}
}
}
void strassenR ( vector < vector < int > > Y Un ,
vector < vector < int > > y B ,
vector < vector < int > > y C , int tam ) {
si ( tam <= leafsize ) {
ikjalgorithm ( A , B , C , tam );
volver ;
}
// Otros casos se tratan aquí:
otra cosa {
int newTam = TAM / 2 ;
vector < int > interior ( newTam );
vector < vector < int > >
A11 ( newTam , interior ), A12 ( newTam , interior ), A21 ( newTam , interior ), A22 ( newTam , interior ),
B11 ( newTam , interior ), B12 ( newTam , interior ), b21 ( newTam , interior ), B22 ( newTam , interior ),
C11 ( newTam , interior ), C12 ( newTam , interior ), C21 ( newTam , interior ), C22 ( newTam , interior ),
P1 ( newTam , interior ), P2 ( newTam , interior ), p3 ( newTam , interior ), p4 ( newTam , interior ),
p5 ( newTam , interior ), P6 ( newTam , interior ), p7 ( newTam , interior ),
aResult ( newTam , interior ), bResult ( newTam , interior );
int i , j ;
// dividiendo las matrices en 4 submatrices:
para ( i = 0 ; i < newTam ; i ++ ) {
a ( j = 0 ; j < newTam ; j ++ ) {
a11 [ i ] [ j ] = A [ i ] [ j ];
A12 [ i ] [ j ] = A [ i ] [ j + newTam ];
a21 [ i ] [ j ] = A [ i + newTam ] [ j ];
a22 [ i ] [ j ] = A [ i + newTam ] [ j + newTam ];
b11 [ i ] [ j ] = B [ i ] [ j ];
b12 [ i ] [ j ] = B [ i ] [ j + newTam ];
b21 [ i ] [ j ] = B [ i + newTam ] [ j ];
b22 [ i ] [ j ] = B [ i + newTam ] [ j + newTam ];
}
}
// Cálculo de P1 a P7:
suma ( a11 , a22 , aResult , newTam ); // A11 + a22
suma ( b11 , b22 , bResult , newTam ); // B11 + B22
strassenR ( aResult , bResult , p1 , newTam ); // P1 = (a11 + a22) * (B11 + B22)
suma ( a21 , a22 , aResult , newTam ); // A21 + a22
strassenR ( aResult , b11 , p2 , newTam ); // P2 = (a21 + a22) * (b11)
restar ( b12 , b22 , bResult , newTam ); // B12 - B22
strassenR ( A11 , bResult , p3 , newTam ); // P3 = (a11) * (b12 - b22)
restar ( b21 , b11 , bResult , newTam ); // B21 - b11
strassenR ( a22 , bResult , p4 , newTam ); // P4 = (a22) * (b21 - b11)
suma ( a11 , a12 , aResult , newTam ); // A11 + a12
strassenR ( aResult , B22 , p5 , newTam ); // P5 = (a11 + a12) * (b22)
restar ( a21 , a11 , aResult , newTam ); // A21 - a11
suma ( b11 , b12 , bResult , newTam ); // + B11 b12
strassenR ( aResult , bResult , P6 , newTam ); // P6 = (A21-A11) * (B11 + B12)
restar ( a12 , a22 , aResult , newTam ); // A12 - a22
suma ( b21 , b22 , bResult , newTam ); // B21 + B22
strassenR ( aResult , bResult , p7 , newTam ); // P7 = (A12-A22) * (B21 + B22)
// Cálculo de c21, c21, c11 c22 e:
sum ( p3 , p5 , c12 , newTam ); // C12 = p3 + p5
suma ( p2 , p4 , c21 , newTam ); // C21 = p2 + p4
sum ( P1 , P4 , aResult , newTam ); // P1 + p4
suma ( aResult , p7 , bResult , newTam ); // P1 + P4 + p7
restar ( bResult , p5 , c11 , newTam ); // C11 = P1 + P4 - P5 + p7
sum ( p1 , p3 , aResult , newTam ); // P1 + p3
suma ( aResult , P6 , bResult , newTam ); // P1 + p3 + P6
restar ( bResult , p2 , c22 , newTam ); // C22 = p1 + p3 - p2 + p6
// La agrupación de los resultados obtenidos en una sola matriz:
para ( i = 0 ; i < newTam ; i ++ ) {
a ( j = 0 ; j < newTam ; j ++ ) {
C [ i ] [ j ] = c11 [ i ] [ j ];
C [ i ] [ j + newTam ] = c12 [ i ] [ j ];
C [ i + newTam ] [ j ] = c21 [ i ] [ j ];
C [ i + newTam ] [ j + newTam ] = c22 [ i ] [ j ];
}
}
}
}
unsigned int nextPowerOfTwo ( int n ) {
retorno pow ( 2 , int ( ceil ( log2 ( n ))));
}
void strassen ( vector < vector < int > > Y Un ,
vector < vector < int > > y B ,
vector < vector < int > > y C , sin firmar int n ) {
// unsigned int n = tam;
unsigned int m = nextPowerOfTwo ( n );
vector < int > interior ( m );
vector < vector < int > > APrep ( m , interior ), BPrep ( m , interior ), CPrep ( m , interior );
para ( sin firmar int i = 0 ; i < n ; i ++ ) {
a ( unsigned int j = 0 ; j < n ; j ++ ) {
APrep [ i ] [ j ] = A [ i ] [ j ];
BPrep [ i ] [ j ] = B [ i ] [ j ];
}
}
strassenR ( APrep , BPrep , CPrep , m );
para ( sin firmar int i = 0 ; i < n ; i ++ ) {
a ( unsigned int j = 0 ; j < n ; j ++ ) {
C [ i ] [ j ] = CPrep [ i ] [ j ];
}
}
}
void suma ( vectorial < vector < int > > Y Un ,
vector < vector < int > > y B ,
vector < vector < int > > y C , int tam ) {
int i , j ;
para ( i = 0 ; i < tam ; i ++ ) {
a ( j = 0 ; j < tam ; j ++ ) {
C [ i ] [ j ] = A [ i ] [ j ] + B [ i ] [ j ];
}
}
}
vacío de resta ( vector < vector < int > > Y Un ,
vector < vector < int > > y B ,
vector < vector < int > > y C , int tam ) {
int i , j ;
para ( i = 0 ; i < tam ; i ++ ) {
a ( j = 0 ; j < tam ; j ++ ) {
C [ i ] [ j ] = A [ i ] [ j ] - B [ i ] [ j ];
}
}
}
int getMatrixSize ( string nombre ) {
string línea ;
ifstream archivoentrada ;
archivoentrada . abierta ( nombre de archivo . c_str ());
getline ( archivoentrada , línea );
volver recuento ( line . iniciar (), la línea . final (), '\ t' ) + 1 ;
}
vacío de lectura ( cadena de nombre de archivo , vector < vector < int > > Y Un , vector < vector < int > > y B ) {
string línea ;
ARCHIVO * matrixfile = freopen ( nombre de archivo . C_str (), "r" , la entrada estándar );
si ( matrixfile == 0 ) {
cerr << "No se pudo leer el archivo" << nombre de fichero << endl ;
volver ;
}
int i = 0 , j , una ;
mientras que ( getline ( cin , línea ) && ! line . vacía ()) {
istringstream ISS ( línea );
j = 0 ;
mientras que ( ISS >> una ) {
A [ i ] [ j ] = una ;
j ++ ;
}
I ++ ;
}
i = 0 ;
mientras que ( getline ( cin , línea )) {
istringstream ISS ( línea );
j = 0 ;
mientras que ( ISS >> una ) {
B [ i ] [ j ] = una ;
j ++ ;
}
I ++ ;
}
fclose ( matrixfile );
}
anulará printMatrix ( vector < vector < int > > matriz , int n ) {
a ( int i = 0 ; i < n ; i ++ ) {
a ( int j = 0 ; j < n ; j ++ ) {
si ( j ! = 0 ) {
cout << " \ t " ;
}
Cout << matriz [ i ] [ j ];
}
Cout << endl ;
}
}
int principal ( int argc , char * argv []) {
string nombre ;
si ( argc < 3 ) {
nombre de archivo = "2000.in" ;
} Demás {
nombre de archivo = argv [ 2 ];
}
si ( argc < 5 ) {
leafsize = 16 ;
} Demás {
leafsize = atoi ( argv [ 4 ]);
}
int n = getMatrixSize ( nombre de archivo );
vector < int > interior ( n );
vector < vector < int > > A ( n , interior ), B ( n , interior ), C ( n , interior );
leer ( nombre de archivo , A , B );
strassen ( A , B , C , N );
printMatrix ( C , n );
volver 0 ;
}
Bibliografía