Menú

Mostrar Mensajes

Esta sección te permite ver todos los mensajes escritos por este usuario. Ten en cuenta que sólo puedes ver los mensajes escritos en zonas a las que tienes acceso en este momento.

Mostrar Mensajes Menú

Mensajes - dijsktra

#1
Programación C/C++ / Re: Frustracion
25 Junio 2021, 21:21 PM
Aqui te va una propuesta de conteo de palabras con recursion...

(No es pa tanta frustración... Más se perdio en Cuba, que decía mi abuelo... ::) :D     )

Decir que es imprescindible llevar control de la primera posición de la linea - si queremos admitir lineas vacias - y del resultado provisional.





#include <ctype.h>
#include <stdio.h>
#include <stdlib.h>


// Immersion:
// ac = #i : 0 <= i < n : V[i]='nb' and (i>0 -> V[i-1]='b')
// (s act as index n, in C-ish style)
// 0 <= n <= len(p)
int num_wordG(const char *p,const char *s, int ac)
{
 if (*s)
   if (isspace(*s)) return num_wordG(p,++s,ac); // Rec.
   else return num_wordG(p,s+1,ac+(p==s || isspace(*(s-1)))); // Rec.
 return ac; // Basis case.
}


int main(int argc, char *args[])
{
 char *line = NULL;
 size_t len = 0 ;
 ssize_t nread;
 for ( ; (nread = getline(&line, &len, stdin)) != -1; )
   {
     line[nread-1]=0; // chop the \n
     printf("%s -> %i\n",line,num_wordG(line,line,0));
   }
 free(line);
 return 0;
}





Algunos casos de prueba...


En un lugar de la Mancha
En un lugar de la Mancha -> 6
son 3 palabras
son 3 palabras -> 3
una
una -> 1

-> 0

#2
Programación C/C++ / Re: Semaforos en c
25 Junio 2021, 17:48 PM
Cita de: RayR en 25 Junio 2021, 15:27 PM
¿¿¿??? Pues eso es lo que puse en mi mensaje:


Tienes Razon, RayR... Acabe cansado de implementarlo y no lei del todo tu respuesta.
#3
Programación C/C++ / Re: Semaforos en c
25 Junio 2021, 10:22 AM
Cita de: RayR en 16 Mayo 2021, 19:57 PM
... A cada hilo le podrías enviar como parámetro entero su número de índice. Cada hilo "n" espera por el semáforo n, e incrementa (sem_post) el semáforo n + 1, a excepción del último (2), que incrementará el primero (0).
El planteamiento es correcto. Pero es importante resaltar el valor inicial de los semáforos. Todos a 0, menos el primero, a 1.

Aqui va una implementación.


#include <pthread.h>
#include <stdlib.h>
#include <stdio.h>
#include <semaphore.h>
#include <unistd.h> // sleep thread.
#include <string.h> // memcopy

static const char values[3]={'A','B','C'};
static sem_t semaphore[3];
static pthread_t thread[3];

static void *routine(void *arg);

int main(int argc, char *args[])
{
    for (int i = 0; i < 3; i++)
if (sem_init(&(semaphore[i]), 0, (i==0)?1:0))
    {
perror("sem_init");
exit(-1);
    }
    for (int i = 0; i < 3; i++)
{
    int *local_idx;
    if ((local_idx= (int *)malloc(sizeof(int)))==NULL)
{
    perror("malloc");
    exit(-1);
}
    *local_idx=i;
    if  (pthread_create(&thread[i], NULL, routine, local_idx))
    {
        perror("pthread_create");
        exit(-1);
    }
}
    for (int i = 0; i < 3; i++)
      if (pthread_join(thread[i], NULL))
{
  perror("pthread_join");
  exit(-1);
}
}

static void *routine(void *arg)
{
    int local_id;
    memcpy(&local_id,arg,sizeof(int));
    free(arg);
    for( ; 1 ; )
{
    if (sem_wait(&semaphore[local_id]))
{
    perror("sem_wait");
    exit(-1);
}
    printf("%c",values[local_id]);
    //sleep(1);
    if (sem_post(&semaphore[(local_id+1)%3]))
{
    perror("sem_post");
    exit(-1);
}
}
}



Ejemplo de salida

ABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABCABC...
#4
Programación C/C++ / Re: AYUDA PROBLEMA C++
17 Junio 2020, 21:56 PM
Cita de: YreX-DwX en 15 Junio 2020, 21:42 PM
...
Claro que teniendo desde el principio en cuenta la otra condición de unicidad de los dígitos se puede exprimir mucho más el problema y llegar a una solución como la que muestras (que cuando tenga más tiempo le daré unas vueltecillas porque a mí no se me habría ocurrido hacerlo así  :xD)

He arreglado la respuesta que compuse en https://foro.elhacker.net/programacion_cc/c_divisibilidad_por_primos_de_un_numero_por_partes-t505293.0.html;msg2223708#msg2223708

Echale un vistazo, esta mas clara,.
#5
Programación C/C++ / Re: AYUDA PROBLEMA C++
15 Junio 2020, 13:19 PM
Lo primero, El_lentejas, cambia el titulo por algo m'as sugerente en vez de AYUDA PROBLEMA C++...

Esto no ayuda a los buscadores a indexar la respuesta...

Pon , "digitos y divisibilidad por primos, por ejemplo@...


Cita de: YreX-DwX en 12 Junio 2020, 02:42 AM
...
Si te fijas, al principio del enunciado hay otro comentario:Es cierto que hay dos números que cumplen con las condiciones recuadradas y estos son:

Numero valido: 4130952867
Numero valido: 4130959493

...

EDIT


He decidido cambiar la respuesta, para editar un caso con mejor codigo y salida formateada.
He generalizado el problema a sacar todos los numeros especiales desde 0(10 veces)0 hasta 999(10 veces)9.
Sigue siendo de complejidad O(1), porque solo tratamos la base 10... para tratar en base N, con N arbitrariamente grande, tendriamos que hacernos con N-3 primos, cosa que, como se sabe, es muy costosa...Fundamento de  algoritmos de criptografia.

1406357289
d1d2d3 = 406 --> 406 / 2 = 203
d2d3d4 = 063 --> 063 / 3 = 21
d3d4d5 = 635 --> 635 / 5 = 127
d4d5d6 = 357 --> 357 / 7 = 51
d5d6d7 = 572 --> 572 / 11 = 52
d6d7d8 = 728 --> 728 / 13 = 56
d7d8d9 = 289 --> 289 / 17 = 17
----------
1430952867
d1d2d3 = 430 --> 430 / 2 = 215
d2d3d4 = 309 --> 309 / 3 = 103
d3d4d5 = 095 --> 095 / 5 = 19
d4d5d6 = 952 --> 952 / 7 = 136
d5d6d7 = 528 --> 528 / 11 = 48
d6d7d8 = 286 --> 286 / 13 = 22
d7d8d9 = 867 --> 867 / 17 = 51
----------
1460357289
d1d2d3 = 460 --> 460 / 2 = 230
d2d3d4 = 603 --> 603 / 3 = 201
d3d4d5 = 035 --> 035 / 5 = 7
d4d5d6 = 357 --> 357 / 7 = 51
d5d6d7 = 572 --> 572 / 11 = 52
d6d7d8 = 728 --> 728 / 13 = 56
d7d8d9 = 289 --> 289 / 17 = 17
----------
4106357289
d1d2d3 = 106 --> 106 / 2 = 53
d2d3d4 = 063 --> 063 / 3 = 21
d3d4d5 = 635 --> 635 / 5 = 127
d4d5d6 = 357 --> 357 / 7 = 51
d5d6d7 = 572 --> 572 / 11 = 52
d6d7d8 = 728 --> 728 / 13 = 56
d7d8d9 = 289 --> 289 / 17 = 17
----------
4130952867
d1d2d3 = 130 --> 130 / 2 = 65
d2d3d4 = 309 --> 309 / 3 = 103
d3d4d5 = 095 --> 095 / 5 = 19
d4d5d6 = 952 --> 952 / 7 = 136
d5d6d7 = 528 --> 528 / 11 = 48
d6d7d8 = 286 --> 286 / 13 = 22
d7d8d9 = 867 --> 867 / 17 = 51
----------
4160357289
d1d2d3 = 160 --> 160 / 2 = 80
d2d3d4 = 603 --> 603 / 3 = 201
d3d4d5 = 035 --> 035 / 5 = 7
d4d5d6 = 357 --> 357 / 7 = 51
d5d6d7 = 572 --> 572 / 11 = 52
d6d7d8 = 728 --> 728 / 13 = 56
d7d8d9 = 289 --> 289 / 17 = 17
----------

A falta de un criterio de divisbilidad para cualquier numero primo, no tenemos mas remedio que practicar la division y ver el resultado, lo que se traduce en una ténica de "fuerza bruta" o "backtraking".

El backtraking consiste en explorar todas las posibildades de un arbol de decision... Ensayamos en primer lugar el 0, depues cualquier numero menos el 0, que puedeser el {1,2,3,4,5,6,7,8,9}... Si cogemos el 1, entonces para la tercera cifra dispondremos de {2,3,4,5,6,7,8,9}. Pero si cogemos el 2 entonces tendermso {1,3,4,5,6,7,8,9}... Como se puede intuir, algo por lo menos exponencial. 10^10.
un 10 seguido de 10 ceros...

Se puede recortar computos marcando los digitos que ya se han usado, para asi no volverlos a usar... (podar el espacio de soluciones...) Si procedemos con una busqueda por profundidad, la recursion nos proporciona gratis la estructura de pila necesaria para hacer el recorrido...

Bueno,... miraros los esquemas de programacion "vuelta atras" y sabreis a lo que me refiero.

#include <stdio.h>
#include <stdlib.h>
#include <assert.h>

#define MAX 10
const int N = MAX ;   // i.e O(1)
static  int d[MAX];
static const  int primes[MAX-3]={2,3,5,7,11,13,17}; // N primes is very hard to compute.



void print_solution(const int d[])
{
  for(int n=0;n<10;n++) printf("%d",d[n]);
  printf("\n");
  for(int n=1;n<8;n++)
    {
      char s[4];
      sprintf(s,"%d%d%d",d[n],d[n+1],d[n+2]);
      int num=atoi(s);
      printf("d%dd%dd%d = %03d --> %03d / %d = %d\n",n,n+1,n+2,num,num,primes[n-1],num/primes[n-1]);
    }
  printf("----------\n");
  return;
}

/* O(1) Specialzer.*/
void primeG(const int k, const int part,const int N)
{
  static int free[MAX]={1,1,1,1,1,1,1,1,1,1};

  //  printf("%d\n",k);
  if (k==N)
    {
      print_solution(d);
      return;
    }
  else
    for(int c=0; c<10; c++)
    {
      if (!free[d[k]=c]) continue;
#define DDD (10*part + c)
      if ((k>2) && (DDD%primes[k-3])) continue;
      free[c]=0;
      primeG(k+1,DDD%100,MAX);
      free[c]=1;
    }
  return;
}





int main(int argc, char *args)
{
  // Call Specializer.
  primeG(0,0,MAX);
  return 0;
}



#6
Cita de: Loretz en 25 Mayo 2020, 20:09 PM
¿dijsktra?

¿Puede ser que estés haciendo tu tributo jugando con el nombre del homenajeado?

https://www.britannica.com/biography/Edsger-Dijkstra



El mismo. Una figura apasionante. Con el (y otros más=, la programación pasó a tener rando de disciplina científica.

¿ Te interesa saber algo de él?
Entre otras cosas, te diré que fue un poco "tecnofobo", todos sus artículos los escrbía a manuscrito, ideo un método de sintésis de programas sistemático, conocido como la "derivación de programas".... Se oponía  a la programación con GOTOS, siendo el primer apóstol de la promgramación estructurada.

Tampoco le gustaba mucho la programación orientada a objetos, que consideraba "una aberración propia de Sillicon Valley..."

Solía decir. "Si depurar es el acto de encontrar fallos en un programa, entonces programar debe consistir en el acto de ponerlos dentro"...

Una vida apasionante....

https://www.youtube.com/watch?v=nVyqbfzOGI8


#7
Mas sencillo.

#include <stdio.h>
#include <stdlib.h>


int main(int argc, char *args)
{
  int n,acum,many;
  many=acum=0;
  while (scanf("%d",&n)==1)
    {
      acum +=n;
      many++;
    }
  if (ferror(stdin))
    {
      perror("scanf");
      exit(1);
    }
  printf("\n%d %d\n",many,acum);
  return 0;
}


Pruebas
gcc stream.c -o main && ./main
10 20 30 40 50 n

5 150
#8
Bueno, ya va siendo hora que vaya dando una respuesta a este tema... ME VA A QUEDAR UN MENSAJE LARGO, pero a lo mejor a la posteridad le sirve de ayuda.
Recordemos que las condiciones para resolverlo eran

  • sin cambiar la codificacion de los parametros, esto es, entra const int **A, no const int *A
  • una sola variable de control, n, (aparte de los parametros, A, N)
  • Un solo bucle. No he podido, me he cansado!! (Se puede, no obstante)

Cita de: YreX-DwX en  6 Mayo 2020, 05:43 AM

i := 1
MIENTRAS i <= (N*(N-1)/2) and i != 0 ENTONCES
   i = (A[f(i)][g(i)] == 0) * (i+1)
FIN MIENTRAS
RETURN i

Como se puede apreciar falta definir las funciones f() y g() que determinen la fila y columna del siguiente elemento en función de i.Si alguien se quiere animar a intentar seguir o a exponer su razonamiento... :rolleyes: :rolleyes:

Ni mucho menos es tan simple. En https://en.wikipedia.org/wiki/Pairing_function y en http://szudzik.com/ElegantPairing.pdf se describen algunos esquemas de recorrido de una matriz, proyectando dos dimensiones en uno y viceversa. No viene la tuya, pero para otras mas sencillas quedan expresiones muy complejas que trascienden el uso basico de % y /.... Un horror, vamos porque nosotros queremos no pasar de % y /.

Yo sabia que se podía, porque un resultado clásico de computabilidad dice:

Citartodo lo que se puede computar con programa estilo "while" y N variables se puede hacer con 1 variable.

La contrapartida será, por supuesto,


  • la legibilidad, No tiene interes pr'actico, pero si t'eorico, pues se puede hacer lo mismo con una variable que con N, (tesis de turing-Church)
  • la eficiencia.Los programas consumen más tiempo de ejecución, aunque sin dejar de ser tratable polinomialmente (Tesis de Cook)  
Esta  es la estretegia:


En una variable se pueden codificar el estado de N variables, sin más que adoptar en esa variable el producto de los primeros N numeros primos elevados al valor de cada variable respectiva.


Eso quiere decir que para codificar


x,y = 3, 1


Damos el valor a n de.

n = (2^3)*(3^1)=8*3=24


Que por el teorema de la Aritmetica, denota univocamente unico numero o valor para n. La forma para recuperar x, (y) será el  exponente de la mayor potencia de 2, (3) que todavia haga divisible a n entre ella.


x = max { k | \exists z n=2^k*z } = 3, ya que 2^4=32 no divide a 24
y = max { k | \exists z n=3^k*z } = 1 ya que 3^2=9 no divide a 24


Pero recordemos que no podemos usar m'as que una variable, luego es ilegal llamar a funciones f(n) g(n), porque estas har'ian uso de pilas y variables, y eso es trampa... Da la casualidad de que esas funciones SI SE PUEDEN COMPUTAR SOLO CON % Y /.

Este es el codigo que quier implementar de partida.

int lowerTriangleDesc(const int **A, const int N)
{
 int i, j ;
 for(i = 0, j=i+1; i < N-1 && !A[i][j]; )
   if (j==N-1) { i++; j = i + 1 ;}
   else j++;
 return i==N-1;
}


Fijemonos en como se realizan las instrucciones post decremento post incremento


#define DEC(id)         n/=id
#define INC(id)         n*=id


Esto es asi, ya que sumar 1 a la n'esima variable equivalea multiplicar el numero por el n-esimo factor primo.

M'as dificil...
Como poder evaluar i < N - 1, cuando en nuestra unica variable solo esta almacenado n=2^i*3^j ?

La estrategia sera cargar en una variable auxiliar-virtual k,n=2^i*3^j*5^k el valor de N-1-i y ver si es mayor que cero

  •   el exponente k(5) tiene que ser  igual a N, lo  que solo se puede llevar a cabo al multiplicar tantas veces por 5 como unidades tenga N
  • Como esto solo se puede saber decrementando N una unidad hasta que queda 0...al punto que multiplicamos por 5, multiplicamos por 7 (otra variable virtual l) para no perder la referencia de N en posteriores computos (que rollo!)
  • A continuacion, restauramos N, haciendo la operacion inversa, dividiendo por 7, hasta que no se pueda mas. esto es n%7==0,  ycada vez se suma 1  a N
  • Despues restamos a k (5), 1, dividiendo por 5.
  • Despues restamos i(2) a k(5), con cuidado de guardar copia en l(7) de lo que vale i  (Otra vez Un rollo!!!)
  • Restauramos en i(2), previamente guardado en el exponente de l(7)
  • Finalmente, vemos el resultado k(5) a ver si es mayor que cero. Esto ocurre si el numero n%5==0, en otro caso, k(5) es igual a 0. Convencete con n=(2^1)*(3^3)*5(^1). Ves que el resto entre 5(k) es 0, lo que no pasa con n=(2^1)*(3^3)*5(^0), cuyo resto entre 5(k) no es 0

// Expressions
#define NON_ZERO(id)    !(n % id)
#define ZERO(id)        (n%id)


Con un poco de paciencia, se puede entender el razonamiento.


  • Como es tan tedioso y proclve a errores me he creado una especie de "compilador" con las directivas de preprocesador C.
  •  El monstruo ha quedado notablemente m'as complejo que el programa anterior, pero era lo esperado... ya que, como se puede ver, solo se usa UNA variable, n)



int debug(const char *s, int n, const int i)
{
#ifdef DEBUG  
 int c;
 for( c=0; !(n%i) ; n/=i) c++;
 printf("debug STATE %s : -> %d \n",s,c);
 return c;
#endif  
}


// N Constant
#define ASSIGN_N(id1,N,id2)  for( ;N;N--) n*=id1*id2
#define RESTORE_N(N,id) for( ; !(n%id);n/=id) N++

// Restoring
#define RESTORE(id2,id3)    for( ; !(n%id3);n/=id3)  n*=id2

// Ordinary vars.
#define DEC(id)         n/=id
#define INC(id)         n*=id
#define SUBS(id1,id2,id3)   for( ; !(n%id2);n/=id2) { n/=id1; n*=id3 ;}
#define RESET(id)       for( ; !(n%id);n/=id)
#define ASSIGN(id1,id2,id3)   for( ; !(n%id2);n/=id2) n*=id1*id3
#define NEG(id)           if (n%id) n*=id; else  for( ; !(n%id);n/=id)

// Expressions
#define NON_ZERO(id)    !(n % id)
#define ZERO(id)        (n%id)

// Indirect pointers;
#define INC_POINTER(A,id1,id2) for( ; !(n%id1); n/=id1 ) { A++; n*=id2 ; }
#define INC_POINTER_POINTER(A,id1,id2) for( ; !(n%id1); n/=id1 ) { (*A)++; n*=id2;  }

#define DEC_POINTER_POINTER(A,id1,id2) for( ; !(n%id1); n/=id1 ) { (*A)--; n*=id2;  }
#define DEC_POINTER(A,id1,id2)    for( ; !(n%id1); n/=id1 ) { A--; n*=id2 ; }

int lowerTriangleDesc(const int **A, int N)
{
 int n=1;
 // i,j,k : 2,3,5    l : 7
 n=3;  // i,j,k,l=0,1,0,0
 ASSIGN_N(5,N,7);
 RESTORE_N(N,7);
 DEC(5);
 SUBS(5,2,7);
 RESTORE(2,7);
 if (NON_ZERO(5))  // if (N-1-i>0), i.e i<N-1
   {
     INC_POINTER(A,2,7); // A[i]
     RESTORE(2,7) ;
     INC_POINTER_POINTER(A,3,7); // A[i][j]
     RESTORE(3,7) ;
     if (**A)
RESET(5);
     DEC_POINTER_POINTER(A,3,7); // A[i][j]
     RESTORE(3,7) ;
     DEC_POINTER(A,2,7);
     RESTORE(2,7) ;
   } // k = k && !A[i][j]
 debug("init i",n,2); debug("init j",n,3);
 debug("B: loop i < N-1 && !A[i][j]",n,5);
 for(; NON_ZERO(5); )
   {
     RESET(5);
     ASSIGN_N(5,N,7);
     RESTORE_N(N,7);
     DEC(5);
     SUBS(5,3,7); // N-1-j > 0 , j<N-1
     RESTORE(3,7);
     if (ZERO(5))
{
 INC(2); //i++
 RESET(3);
 ASSIGN(3,2,5);
 RESTORE(2,5);
 INC(3); //j++
}
     else INC(3);
     debug("step i",n,2); debug("step j",n,3);
     ASSIGN_N(5,N,7);
     RESTORE_N(N,7);
     DEC(5);
     SUBS(5,2,7);
     RESTORE(2,7);
     if (NON_ZERO(5))  // if (N-1-i>0), i.e i<N-1
{
 INC_POINTER(A,2,7); // A[i]
 RESTORE(2,7) ;
 INC_POINTER_POINTER(A,3,7); // A[i][j]
 RESTORE(3,7) ;
 if (**A)
   RESET(5);
 DEC_POINTER_POINTER(A,3,7); // A[i][j]
 RESTORE(3,7) ;
 DEC_POINTER(A,2,7);
 RESTORE(2,7) ;
} // k = k && !A[i][j]
     debug("B: loop i < N-1 && !A[i][j]",n,5);
   }
 ASSIGN_N(5,N,7);
 RESTORE_N(N,7);
 DEC(5);
 SUBS(5,2,7);
 RESTORE(2,7);
 NEG(5);
 debug("epilogue: i==N-1",n,5);
 return NON_ZERO(5);
}


Los mensajes debug, por supuesto, son opcionales y no forman parte del codigo. Es para poder seguir mejor la codificacion de las variables en una sola.

Ah!, las pruebas de que va bien!

El codigo de IO esta en
https://foro.elhacker.net/programacion_cc/algebra_de_matrices_matriz_triangular_superior_desdendiente_en_c-t504451.0.html;msg2220794#msg2220794

$ ./main
1
1
debug STATE init i : -> 0
debug STATE init j : -> 1
debug STATE B: loop i < N-1 && !A[i][j] : -> 0
debug STATE epilogue: i==N-1 : -> 1
lowerTriangleDesc: 1
2
1 0
1 1
debug STATE init i : -> 0
debug STATE init j : -> 1
debug STATE B: loop i < N-1 && !A[i][j] : -> 1
debug STATE step i : -> 1
debug STATE step j : -> 2
debug STATE B: loop i < N-1 && !A[i][j] : -> 0
debug STATE epilogue: i==N-1 : -> 1
lowerTriangleDesc: 1
2
1 1
1 1
debug STATE init i : -> 0
debug STATE init j : -> 1
debug STATE B: loop i < N-1 && !A[i][j] : -> 0
debug STATE epilogue: i==N-1 : -> 0
lowerTriangleDesc: 0
3
1 0 0
1 1 0
1 1 1
debug STATE init i : -> 0
debug STATE init j : -> 1
debug STATE B: loop i < N-1 && !A[i][j] : -> 2
debug STATE step i : -> 0
debug STATE step j : -> 2
debug STATE B: loop i < N-1 && !A[i][j] : -> 3
debug STATE step i : -> 1
debug STATE step j : -> 2
debug STATE B: loop i < N-1 && !A[i][j] : -> 1
debug STATE step i : -> 2
debug STATE step j : -> 3
debug STATE B: loop i < N-1 && !A[i][j] : -> 0
debug STATE epilogue: i==N-1 : -> 1
lowerTriangleDesc: 1
3
1 0 0
1 1 1
1 1 1

debug STATE init i : -> 0
debug STATE init j : -> 1
debug STATE B: loop i < N-1 && !A[i][j] : -> 2
debug STATE step i : -> 0
debug STATE step j : -> 2
debug STATE B: loop i < N-1 && !A[i][j] : -> 3
debug STATE step i : -> 1
debug STATE step j : -> 2
debug STATE B: loop i < N-1 && !A[i][j] : -> 0
debug STATE epilogue: i==N-1 : -> 0
lowerTriangleDesc: 0


P.D
Si aun no te convence que es una variable...  aplica la siguiente linea en el compilador
para observar que solo existe la varaible "n" en la rutina...

gcc -E source.c -o source.hcp


el resultado es.

int lowerTriangleDesc(const int **A, int N)
{
 int n=1;

 n=3;
 for( ;N;N--) n*=5*7;
 for( ; !(n%7);n/=7) N++;
 n/=5;
 for( ; !(n%2);n/=2) { n/=5; n*=7 ;};
 for( ; !(n%7);n/=7) n*=2;
 if (!(n % 5))
   {
     for( ; !(n%2); n/=2 ) { A++; n*=7 ; };
     for( ; !(n%7);n/=7) n*=2 ;
     for( ; !(n%3); n/=3 ) { (*A)++; n*=7; };
     for( ; !(n%7);n/=7) n*=3 ;
     if (**A)
for( ; !(n%5);n/=5);
     for( ; !(n%3); n/=3 ) { (*A)--; n*=7; };
     for( ; !(n%7);n/=7) n*=3 ;
     for( ; !(n%2); n/=2 ) { A--; n*=7 ; };
     for( ; !(n%7);n/=7) n*=2 ;
   }
 debug("init i",n,2); debug("init j",n,3);
 debug("B: loop i < N-1 && !A[i][j]",n,5);
 for(; !(n % 5); )
   {
     for( ; !(n%5);n/=5);
     for( ;N;N--) n*=5*7;
     for( ; !(n%7);n/=7) N++;
     n/=5;
     for( ; !(n%3);n/=3) { n/=5; n*=7 ;};
     for( ; !(n%7);n/=7) n*=3;
     if ((n%5))
{
 n*=2;
 for( ; !(n%3);n/=3);
 for( ; !(n%2);n/=2) n*=3*5;
 for( ; !(n%5);n/=5) n*=2;
 n*=3;
}
     else n*=3;
     debug("step i",n,2); debug("step j",n,3);
     for( ;N;N--) n*=5*7;
     for( ; !(n%7);n/=7) N++;
     n/=5;
     for( ; !(n%2);n/=2) { n/=5; n*=7 ;};
     for( ; !(n%7);n/=7) n*=2;
     if (!(n % 5))
{
 for( ; !(n%2); n/=2 ) { A++; n*=7 ; };
 for( ; !(n%7);n/=7) n*=2 ;
 for( ; !(n%3); n/=3 ) { (*A)++; n*=7; };
 for( ; !(n%7);n/=7) n*=3 ;
 if (**A)
   for( ; !(n%5);n/=5);
 for( ; !(n%3); n/=3 ) { (*A)--; n*=7; };
 for( ; !(n%7);n/=7) n*=3 ;
 for( ; !(n%2); n/=2 ) { A--; n*=7 ; };
 for( ; !(n%7);n/=7) n*=2 ;
}
     debug("B: loop i < N-1 && !A[i][j]",n,5);
   }
 for( ;N;N--) n*=5*7;
 for( ; !(n%7);n/=7) N++;
 n/=5;
 for( ; !(n%2);n/=2) { n/=5; n*=7 ;};
 for( ; !(n%7);n/=7) n*=2;
 if (n%5) n*=5; else for( ; !(n%5);n/=5);
 debug("epilogue: i==N-1",n,5);
 return !(n % 5);
}


SE PUEDE HACER CON UN SOLO BUCLE

DE NUEVO, EL COSTE A PAGAR SERA UN CONTROL MAS COMPLEJO,

AQUI LO DEJO.
#9
Cita de: NEBIRE en  6 Mayo 2020, 18:22 PM
...y ahora el pseudocódigo (que cualquiera puede traducir a cualquier lenguaje, ya que el problema no es un problema restrictivo a ningún lenguaje específico, es decir no es un problema de ningún lenguaje si no de algoritmia)...

Tengo que dejar claro que su idea es la que más me ha gustado.
Comentarios...

  • Mete  una variable mas, size, que no es m'as que ancho*ancho, luego se puede enmendar
  • No es programacion estructurada (return 0) pero eso es facilmente enmendable.Ojo!, no digo que no sea válida en C, digo que no es C-estructurada
  • La construccion "repeat until", no es la adecuada. De hecho para N=1, el algoritmo falla. Pero es facilmente enmendable

Con las ligeras enmiendas... El pseudocodigo de NEBIRE pasado a C, resulta

int upperTriangleDesc(const int *A, const int N)
{
 int i=N*N-N;
 while (i && !A[i])
   {
     i += N + 1;
     if (i > N*N)
i = N*N - (N*(1 + (i%N)));
   }
 return !i;
}

Y algunos resultados (correctos, por supuesto)
1
1
upperTriangleDesc: 1
2
1 1
0 1
upperTriangleDesc: 1
3
1 2 3
0 1 2
0 0 1
upperTriangleDesc: 1
3
1 2 3
0 1 2
1 0 1
upperTriangleDesc: 0

Es encomiable el esfuerzo por hallar la ley que aplica al indice "i". Realmente es una función compleja...

Sin embargo... Hay un detalle no menor que puede echarse al traste....

SE HA CAMBIADO EL TIPO DE LA MATRIZ DE BIDIMENSIONAL A UNIDIMENSIONAL


const int **A;
// Set by
if  ((A=(int**)malloc(N*sizeof(int *)))==NULL)
 {
   perror("malloc");
   exit(1);
 }
for(int n=0;n<N;n++)
 {
   if  ((A[n]=(int*)malloc(N*sizeof(int)))==NULL)
     {
perror("malloc");
exit(1);
     }
 }

por
const int *A;

if  ((A=(int*)malloc(N*N*sizeof(int)))==NULL)
 {
   perror("malloc");
   exit(1);
 }


EDIT
Esto supone que se han cambiado las condiciones del problema original por otro que era equivalente, que lo emula ,pero con condiciones distintas (los datos de entrada se presuponen codificados de otro modo). As'i, con N=6, el resultado no es correcto cuando se le pasa un valor const int **A, malloc(N), ya que A[30] , esta fuera de memoria. (si lo esta const int *A, malloc(N*N)...)

Todavia no tengo la mia... tengo algnuas cosas todavía sin madurar.
#10
Cita de: YreX-DwX en  5 Mayo 2020, 18:01 PM
Hmm... Veo demasiada similitud con el otro tema.

Tienes razon. YreX-DwX... No me ha quedado bien.... :-( :-( Era muy similar al otro.... Y la solucion que pones es implecable...

Cita de: YreX-DwX en  5 Mayo 2020, 18:01 PM

int lowerTriangleDesc(const int **A, const int N){
    int i, j = N;
    for(i = 0; i < N-1 && j == N; ++i)
        for(j = i+1; j < N && !A[i][j]; ++j);
    return j == N;
}


yo habia propuesto

int lowerTriangleDescBis(const int **A,
const int N)
{
  int i,j;
  for (i=N-1,j=N; i && (j==N); i-=(j==N))
    for(j=i; j<N && !A[i-1][j];j++);
  return !i;
}


Cita de: YreX-DwX en  5 Mayo 2020, 18:01 PM
PD: Sigo pensando que hay alguna solución oculta sorprendente. No dormiré tranquilo hasta ver la solución. :xD

  VAMOS A HACERLO DIFICIL, PERO DIFICILISMO !

Todo el mundo ve que hay dos bucles y dos variables

ESTE ES EL RETO:


PROGRAMAR LA SOLUCION USANDO SOLO UN BUCLE Y UNA SOLA VARIABLE DE CONTROL, A PARTE DE LAS QUE SE DAN POR PARAMETRO


(Por supuesto, el programa sera mas dificil de legible, pero sera igualmente correcto....)


No me pidais la solucion en un dia que no se si tendre tiempo... Se que se puede, y tengo que hacerla...
y tu, YreX-DwX, m'as vale que duermas un poco...que te puedes volver tarumba... :rolleyes: :rolleyes: