Hola amigos bueno se me ha pasado por la cabeza la idea de crear una forma de cifrar letras y números :D pues bueno ya he creado la forma de cifrado pero no quise quedarme en esto nada mas pensé en hacer un programa que me pidiera una palabra o una frase y que la ecripte, pero los únicos comandos que me sabia eran los de la Shell X'D pues decidí leer un poco sobre C++, Bueno y de lectura en lectura me ayudaron con lo siguiente :
#include<stdio.h>
#include<conio.h>
#include<dos.h>
void main(void)
{
printf("Ingrese la palabra: ");
scanf("%s");
char cad[10]="%s";
clrscr();
for(int a=0;a<10;a++)
{
printf("%c\n",cad[a]);
}
getch();
}
Pero el código no realiza la función que yo quiero (aclaro hay lo único que hace es pedirle al usuario que ingrese una palabra y la separe letra por letra) pero esa no es mi meta si alguien podría ayudarme con esto le agradecería sinceramente.
Saludos. :laugh:
#include<stdio.h>
int main()
{
char Palabra[256];
int i = 0;
fgets (Palabra,255,stdin);
for (i = 0; i < strlen(Palabra);i++)
printf("%c \n",Palabra[i]);
getchar(); // pausa
getchar();
return 0;
}
Has cumplido varias violaciones a la norma (void main, librería conio,dos y uso de getch()).
Cita de: Royca3 link=topic=392162.msg1859787#msg1859787
Pero el código no realiza la función que yo quiero (aclaro hay lo único que hace es pedirle al usuario que ingrese una palabra y la separe letra por letra) pero esa no es mi meta si alguien podría ayudarme con esto le agradecería sinceramente.
¿Y que más querías que hiciera?. El código hace ni más ni menos lo que tú le has implementado y, lo único que has hecho es imprimir una cadena introducida por teclado, ¿crees que te adivina el pensamiento y "él" añade el resto del código?. Y aún el caso de imprimir la cadena es aventurado, toda vez que presupones que el usuario va a introducir una cadena de justitos diez caracteres. Mejor sería:#include<stdio.h>
#include<stdlib.h>
int main(void)
{
int i=0;
char cadena[10];
printf("Ingrese la palabra: ");
scanf("%10s",&cadena);
for(i=0;cadena[i]!='\0';i++)
printf("%c ",cadena[i]);
getchar();
return 0;
}
De esta manera te aseguras "tomar" sólo diez caracteres.
Además de lo comentado por amchacon te aconsejaría leer algo sobre el tema de cifrado.
Si estas empezando te diría que te mires el cifrado de César (http://es.wikipedia.org/wiki/Cifrado_C%C3%A9sar), así para empezar con uno sencillo.
Saluditos! .... ..(http://st.forocoches.com/foro/images/smilies/aaaaa.gif)
P.D: amchacon ,te faltó en el código la librería string.h para hacer uso de strlen. Sí, ya sé que son las prisas. A esta hora y ya estamos como motos ;) ;)
aca tienes un ejemplo en c++ :D
cifrar.cpp
#include <iostream>
#include <string>
using namespace std;
int main()
{
string frase;
cout << "escriba una frase" << endl;
getline(cin,frase);
int n=1;
for(int tamano=0;tamano<frase.size();tamano++)
{
char letra=frase[tamano]+n;
cout << letra;
n++;
if(n>10)
{
n=1;
}
}
descifrar.cpp
#include <iostream>
#include <string>
using namespace std;
int main()
{
string frase;
cout << "escriba una frase cifrada" << endl;
getline(cin,frase);
int n=1;
for(int tamano=0;tamano<frase.size();tamano++)
{
char letra=frase[tamano]-n;
cout << letra;
n++;
if(n>10)
{
n=1;
}
}
}
Cita de: daryo en 10 Junio 2013, 16:56 PM
aca tienes un ejemplo en c++ :D
Y aquí uno en C:
#include<stdio.h>
int main()
{
char Palabra[256];
int i = 0;
puts ("\Introduzca palabra o frase a cifrar:");
fgets (Palabra,255,stdin);
puts ("\nLa frase es:");
for (i = 0;Palabra[i]!='\0';i++)
printf("%c ",Palabra[i]);
puts ("\nCiframos la frase");
for (i = 0; Palabra[i]!='\0';i++)
{
Palabra[i]+=i;
printf("%c ",Palabra[i]);
}
puts ("\n\nDesiframos la frase");
for (i = 0; Palabra[i]!='\0';i++)
{
Palabra[i]-=i;
printf("%c ",Palabra[i]);
}
puts ("\n\nPresione ENTER para salir");
getchar();
return 0;
}
Saluditos! .... ..(http://st.forocoches.com/foro/images/smilies/simba2.gif)
Cita de: leosansan en 10 Junio 2013, 18:56 PM
Y aquí uno en C:
#include<stdio.h>
int main()
{
char Palabra[256];
int i = 0;
puts ("\Introduzca palabra o frase a cifrar:");
fgets (Palabra,255,stdin);
puts ("\nLa frase es:");
for (i = 0;Palabra[i]!='\0';i++)
printf("%c ",Palabra[i]);
puts ("\nCiframos la frase");
for (i = 0; Palabra[i]!='\0';i++)
{
Palabra[i]+=i;
printf("%c ",Palabra[i]);
}
puts ("\n\nDesiframos la frase");
for (i = 0; Palabra[i]!='\0';i++)
{
Palabra[i]-=i;
printf("%c ",Palabra[i]);
}
puts ("\n\nPresione ENTER para salir");
getchar();
return 0;
}
Saluditos! .... ..(http://st.forocoches.com/foro/images/smilies/simba2.gif)
Muchas gracias a todos y perdón por mi ignorancia.... :-X
Bueno Te agradezco a ti leosansan Este Programa era el que buscaba...
;D
Saludos.
(Excelente Foro ;)
Cita de: daryo en 10 Junio 2013, 20:03 PM
si que lo es ;D
;-) ;-) ;-)
Soy nuevo y ya me encanto xD
Aquí otro ejemplo caesar:
#include <iostream>
using namespace std;
int main() {
string palabra = ""; //Creamos e inicalizamos variables
int numero = 0;
int TamanyoPalabra = 0;
cout << "Palabra? >> "; cin >> palabra; // 1 i 2 paso
cout << "Veces a correr >> "; cin >> numero; // 3 paso
TamanyoPalabra = palabra.length(); // 4 paso
for (int i = 0; i < TamanyoPalabra ; i++) { // 4 paso
for (int z = 0; z < numero; z++) { palabra[i]++; } // 5 paso
} // 6 paso
cout << palabra; // Mostrar palabra cifrada
return 0; //Salir
}
FUENTE Y MÁS INFORMACIÓN DEL CODE: http://www.infiniterware.net/tutoriales/programacion-general/Caesar-cipher-C++.html
Saluuuuuuu2
[quote author=Puntoinfinito link=topic=392162.msg1859969#msg1859969 date=1370890822]
Aquí otro ejemplo caesar:
[/quote]
Pues no, ni el tuyo ni el de daryo ni el mío con cifrados tipo César, al menos como yo veo los códigos.
Yo lo que hice en el post fue sugerirle que lo mirará pero no dije que en mi código lo aplicará. Me fui, como daryo, a lo fácil que es incrementar la variable letra sencillamente, mientras que en el cifrado César hay que tener en cuenta que si se llega a la las letras finales del abecedario se cuentan las últimas y, si faltan, se sigue la suma por las primeras.
Más información en Cifrado César cortesía de Wikipedia (http://es.wikipedia.org/wiki/Cifrado_C%C3%A9sar).
Saluditos! .... ..(http://st.forocoches.com/foro/images/smilies/ciao.gif)
Gracias amigo pero me confundí yo utilizo C :) Pero bueno, oye Como le hago para que coja la palabra que el usuario ingresa y la cambie por otras letras (especificadas por mi) Gracias
Saludos.
Aquí te dejo el que hice yo en su día, que trabaja con ficheros de texto. Se puede seleccionar el paso del cifrado, y tiene en cuenta que que cuando se llega a la Z se tiene que volver a empezar por la A.
También está en esta dirección:
http://cypascal.blogspot.com.es/2013/04/cifrado-cesar-en-c.html (http://cypascal.blogspot.com.es/2013/04/cifrado-cesar-en-c.html)
/*CIFRADO CESAR EN C/C++*/
/*Este programa se va a usar para llevar a cabo una codificación de un fichero
de texto, mediante el codificado CESAR, que es uno de los métodos más simples
de cifrar.
La variable trans==translación de cifrado, es la que indica cuantas letras
se va a correr el abecedario hacia la derecha. Ejemplo:
trans=3
c1=Caracter decodificado
c2=Caracter codificado
c2=c1+trans
Entonces en nuestro caso si c1='A'
c2='A'+3='D'
El programa solo codifica letras, tanto mayúsculas como minúsculas:
del ASCII 65 ('A') al 90 ('Z') y del 97 ('a') al 122 ('z').
*/
#include <stdio.h>
#include <string.h>
int main(void)
{
FILE *dat1,*dat2;
char nombre1[50],nombre2[50],c;
int trans;
printf("Seleccione translacion de encriptacion: "); scanf("%d", &trans);
printf("Nombre del fichero a codificar: "); scanf("%s", &nombre1);
printf("Nombre del fichero donde guardar codificado: "); scanf("%s", &nombre2);
if (fopen(nombre1,"r")==NULL)
printf("\nEl fichero no existe.\n\n");
else
{
dat1=fopen(nombre1,"r");/*Se abre en modo lectura*/
dat2=fopen(nombre2,"w");/*Se abre en modo escritura*/
fscanf(dat1,"%c",&c);
while (!feof(dat1))
{
/*Se codifica el caracter*/
if ((c>='A')&&(c<='Z'))
{
c=((c+trans-'A')%('Z'-'A'+1))+'A';
}
else if ((c>='a')&&(c<='z'))
{
c=(c+trans-'a')%('z'-'a'+1)+'a';
}
/*Se escribe el caracter en el codificado*/
fprintf(dat2,"%c",c);
/*Se lee un nuevo caracter*/
fscanf(dat1,"%c",&c);
}
fclose(dat1);
fclose(dat2);/*Se cierran ambos ficheros*/
}
}
Salu10 y bienvenido
Cita de: cypascal en 10 Junio 2013, 22:46 PM
Aquí te dejo el que hice yo en su día, que trabaja con ficheros de texto. Se puede seleccionar el paso del cifrado, y tiene en cuenta que que cuando se llega a la Z se tiene que volver a empezar por la A.
También está en esta dirección:
http://cypascal.blogspot.com.es/2013/04/cifrado-cesar-en-c.html (http://cypascal.blogspot.com.es/2013/04/cifrado-cesar-en-c.html)
/*CIFRADO CESAR EN C/C++*/
/*Este programa se va a usar para llevar a cabo una codificación de un fichero
de texto, mediante el codificado CESAR, que es uno de los métodos más simples
de cifrar.
La variable trans==translación de cifrado, es la que indica cuantas letras
se va a correr el abecedario hacia la derecha. Ejemplo:
trans=3
c1=Caracter decodificado
c2=Caracter codificado
c2=c1+trans
Entonces en nuestro caso si c1='A'
c2='A'+3='D'
El programa solo codifica letras, tanto mayúsculas como minúsculas:
del ASCII 65 ('A') al 90 ('Z') y del 97 ('a') al 122 ('z').
*/
#include <stdio.h>
#include <string.h>
int main(void)
{
FILE *dat1,*dat2;
char nombre1[50],nombre2[50],c;
int trans;
printf("Seleccione translacion de encriptacion: "); scanf("%d", &trans);
printf("Nombre del fichero a codificar: "); scanf("%s", &nombre1);
printf("Nombre del fichero donde guardar codificado: "); scanf("%s", &nombre2);
if (fopen(nombre1,"r")==NULL)
printf("\nEl fichero no existe.\n\n");
else
{
dat1=fopen(nombre1,"r");/*Se abre en modo lectura*/
dat2=fopen(nombre2,"w");/*Se abre en modo escritura*/
fscanf(dat1,"%c",&c);
while (!feof(dat1))
{
/*Se codifica el caracter*/
if ((c>='A')&&(c<='Z'))
{
c=((c+trans-'A')%('Z'-'A'+1))+'A';
}
else if ((c>='a')&&(c<='z'))
{
c=(c+trans-'a')%('z'-'a'+1)+'a';
}
/*Se escribe el caracter en el codificado*/
fprintf(dat2,"%c",c);
/*Se lee un nuevo caracter*/
fscanf(dat1,"%c",&c);
}
fclose(dat1);
fclose(dat2);/*Se cierran ambos ficheros*/
}
}
Salu10 y bienvenido
Este code lo que hace es buscar un archivo y "encriptarlo" verdad?
Lo que yo necesito es un programa que pida una palabra o frase y que cambie las letras ejemplo A por H b por X y asi entiendes?
Pero igual Gracias
saludos.
Ups!, perdón, pensaba que estábamos con Cesar.
Pero no me queda claro, ¿Qé método de cifrado quieres utilizar?
Salu10
Buenas amigos!!
Aqui otra versión del cifrado cesar donde tu puedes indicar el desplazamiento que quieras
#include <stdio.h>
#include <stdlib.h>
int main()
{
int des, i = 0, j = 0;
char cadena[30] = {0};
printf("Introduce una cadena: ");
scanf("%s", cadena);
printf("Ingrese desplazamiento: ");
scanf(" %d", &des);
for(; cadena[i]!='\0'; i++)
cadena[i]+=des;
printf("\nEncriptado es: %s \n", cadena);
for(; cadena[j]!='\0'; j++)
cadena[j]-=des;
printf("\nDesencriptado es: %s \n", cadena);
return 0;
}
Saludos ;D
for(; cadena[i]!='\0'; i++)
cadena[i]+=des;
printf("\nEncriptado es: %s \n", cadena);
for(; cadena[j]!='\0'; j++)
cadena[j]-=des;
printf("\nDesencriptado es: %s \n", cadena);
Y digo yo, que necesidad hay de hacer el caso y el inverso:
char cadena_aux[30];
for(; cadena[i]!='\0'; i++)
{
cadena[i]+=des;
cadena_aux[i] = cadena[i];
}
cadenaaux[i] = '\0';
printf("\nEncriptado es: %s y descifrado es %s\n", cadena,cadena_aux);
Bueno Alguien que me explique algo :(
El código de leosansan pero necesito que en lugar de ese método lo cambie por este:
A = X
B = J
C = O
D = P
E = Z
F = W
G = A
H = U
I = Ñ
J=L
K = D
L=Q
M=S
N = F
Ñ=R
O = B
P=M
Q=V
R=H
S=K
T = N
U=Y
V= T
W=I
X = E
Y= G
Z = C
Cita de: amchacon en 10 Junio 2013, 23:32 PM
Y digo yo, que necesidad hay de hacer el caso y el inverso:
Pues para descifrar no y ademas la forma que propones no hace lo deseado ya que la cadena
ya la habias modificado previamente y luego asignas los valores atuales que son los cifrados a
la cadena auxiliar
Saludos :silbar:
usa case
salu10
Cita de: CCross en 11 Junio 2013, 00:04 AM
Pues para descifrar no y ademas la forma que propones no hace lo deseado ya que la cadena
ya la habias modificado previamente y luego asignas los valores atuales que son los cifrados a
la cadena auxiliar
Saludos :silbar:
Huy es verdad :rolleyes:
Habría que hacer la igualación antes ;-)
Cita de: Royca3 en 11 Junio 2013, 00:00 AM
Bueno Alguien que me explique algo :(
El código de leosansan pero necesito que en lugar de ese método lo cambie por este:
A = X
B = J
C = O
D = P
.....................
Creo que seria mas o menos asi:
#include<stdio.h>
int main()
{
char caracter[2][28] = { {"ABCDEFGHIJKLMNÑOPQRSTUVWXYZ"},
{"XJOPZWAUÑLDQSFRBMVHKNYTIEGC"} };
char cadena[30] = {0};
int i, j = 0, k = 0;
printf("Ingrese una cadena en Mayuscula: ");
scanf("%s", cadena);
for (; cadena[k]!='\0'; k++){
for (i = 0; i < 27; i++)
if (cadena[j] == caracter[0][i]){
cadena[j] = caracter[1][i];
j++;
}
}
printf("\n%s", cadena);
return 0;
}
Saludos ::)
Osea que? No se puede? o que debo hacer?
:-X
Cita de: Royca3 en 11 Junio 2013, 01:13 AM
Osea que? No se puede? o que debo hacer?
:-X
El método de Ccros funciona ;)
Bueno compilo el código y tiene cero errores ;-)
Pero a lo que lo ejecuto e ingreso la palabra a cifrar
se sale automáticamente, luego ingreso nuevamente y me aparecen los resultados pero solo cambia la primer letra las demás las deja iguales.
Saludos.
Cita de: Royca3 en 11 Junio 2013, 01:37 AM
Bueno compilo el código y tiene cero errores ;-)
Pero a lo que lo ejecuto e ingreso la palabra a cifrar
se sale automáticamente, luego ingreso nuevamente y me aparecen los resultados pero solo cambia la primer letra las demás las deja iguales.
He editado el codigo de arriba ahora hace los que deseas
Saludos ;D
Cita de: CCross en 11 Junio 2013, 01:52 AM
He editado el codigo de arriba ahora hace los que deseas
Saludos ;D
Bueno mi amgo :D Gracias por tu interes...
Ahora me aprarece un error en
printf("\n%s", cadena);
return 0;
}
Intentare compilarlo con el Dev-c++
Saludos.
Cita de: Royca3 en 11 Junio 2013, 02:30 AM
Bueno mi amgo :D Gracias por tu interes...
Ahora me aprarece un error en
printf("\n%s", cadena);
return 0;
}
Intentare compilarlo con el Dev-c++
Saludos.
Es la línea 18 y es porque falta un punto y coma.
Te falta soltura en el lenguaje, yo te recomiendo que te lo mires entero.
En el cifrado de Cesar para encontrar los sustitutos puedes utilizar el operador de modulo (ya hay ejemplos) o bien se pueden colocar todos los caracteres en un array, con ello la operación termina como una suma (o resta si la operación es descifrar).
Por ejemplo:
#include <stdio.h>
#include <stdlib.h>
#define ABC "abcdefghijklmnopqrstuvwxyz"
int main(void)
{
char ch[] = ABC ABC;
size_t max_despl = sizeof(ABC) - 1;
unsigned despl;
unsigned i;
printf("Desplazamiento: ");
fflush(stdout);
if (scanf("%u", &despl) != 1 || despl > max_despl)
return EXIT_FAILURE;
for (i = 0; i < max_despl; i++)
printf("%c ==> %c\n", ch[i], ch[i + despl]);
return EXIT_SUCCESS;
}
----
Cita de: Royca3 en 10 Junio 2013, 23:13 PMLo que yo necesito es un programa que pida una palabra o frase y que cambie las letras ejemplo A por H b por X y asi entiendes?
Para eso debes colocar todos los caracteres a buscar en un array y sus sustituciones en otro, mas o menos así:
char in[] = "Ab";
char out[] = "HX";
Buscas cada carácter en el array "in" para conocer su indice, una vez lo obtienes (el indice, claro) lo sustituyes por el carácter correspondiente (mismo indice) del array "out".
La solución ya la tienes en el programa de
CCross, una opción ligeramente distinta es utilizando la función strchr para realizar la búsqueda:
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int main(void)
{
char in[] = "ABCDEFGHIJKLMNÑOPQRSTUVWXYZ";
char out[] = "XJOPZWAUÑLDQSFRBMVHKNYTIEGC";
char cadena[30];
char *p;
int i;
int j;
printf("Ingrese una cadena en Mayuscula: ");
fflush(stdout);
if (scanf("%s", cadena) != 1)
return EXIT_FAILURE;
for (i = 0; cadena[i] != '\0'; i++){
p = strchr(in, cadena[i]);
putchar(p == NULL ? cadena[i] : out[p - in]);
}
putchar('\n');
return EXIT_SUCCESS;
}
Un saludo
Disculpa por el detallito lo que causa codedear a prisas, ya lo correji
Saludos ;-)
rir3760 Muchas gracias el código es lo que buscaba pero obtengo el mismo error anterior el programa se me cierra antes de mostrar los resultados :-( alguien sabe el porque?
(Gracias a TODOS)
Saludos.
Cita de: Royca3 en 11 Junio 2013, 03:48 AM
rir3760 Muchas gracias el código es lo que buscaba pero obtengo el mismo error anterior el programa se me cierra antes de mostrar los resultados :-( alguien sabe el porque
Prueba con poner un
getchar(); antes de:
return EXIT_SUCCESS;
Buscate un buen libro de C te recomiendo el libro de kernighan y Ritchie que es muy bueno
para empezar
Saludos
Cita de: CCross en 11 Junio 2013, 03:56 AM
Prueba con poner un getchar(); antes de:
return EXIT_SUCCESS;
Buscate un buen libro de C te recomiendo el libro de kernighan y Ritchie que es muy bueno
para empezar
Saludos
Igual se sale :'( Pues si de echo ya tengo unos manuales pero apenas comienzo :)
Pero porque se sigue saliendo del programa sin mostrar resultados?
Saludos.
Cita de: Royca3 en 11 Junio 2013, 04:57 AM
Igual se sale :'( Pues si de echo ya tengo unos manuales pero apenas comienzo :)
Pero porque se sigue saliendo del programa sin mostrar resultados?
Saludos.
Pon 2 getchar().
La razón por la que se sale esque el programa termina al finalizar el main, si que muestra los resultados pero va tan rápido que ni los ves.
Por cierto:
http://foro.elhacker.net/programacion_cc/iquestpor_que_no_deberias_usar_el_devcpp-t385150.0.html
Cita de: amchacon en 11 Junio 2013, 10:17 AM
Pon 2 getchar().
La razón por la que se sale esque el programa termina al finalizar el main, si que muestra los resultados pero va tan rápido que ni los ves.
Por cierto:
http://foro.elhacker.net/programacion_cc/iquestpor_que_no_deberias_usar_el_devcpp-t385150.0.html
Listo amigo Todo en orden, el programa cumple la función requerida ;-) ;-) ;-) ;-) Les agradezco a CCross, rir3760, a leosansan aunque le doy gracias a todos por su buena voluntad próxima mente subiré un tutorial :laugh:
Saludos.
Cita de: CCross en 10 Junio 2013, 23:28 PM
Buenas amigos!!
Aquí otra versión del cifrado cesar donde tu puedes indicar el desplazamiento que quieras
No es el cifrado César, te muestro una salida al código que planteas:CitarIntroduce una cadena: wikipedia la enciclopedia libre
Ingrese desplazamiento:
cifrado es: ┘╦═╦ÊÃã╦├
descifrado es: wikipedia
Como ves, en el cifrado se "va más allá" de la z/Z, cuando debería empezar a contar desde la a.
Una solución "a pura fuerza bruta", es decir aplicando estrictamente lo que plantea el cifrado César :
/*En criptografía, el cifrado César, también conocido como cifrado
por desplazamiento, código de César o desplazamiento de César, es
una de las técnicas de codificación más simples y más usadas.
Es un tipo de cifrado por sustitución en el que una letra en el
texto original es reemplazada por otra letra que se encuentra un
número fijo de posiciones más adelante en el alfabeto. Por
ejemplo, con un desplazamiento de 6.*/
#include <stdio.h>
#include<stdlib.h>
#include <string.h>
int main (void)
{
int i,n=0,aux;
char frase [256];
char cadena []="WIKIPEDIA, LA ENCICLOPEDIA LIBRE \n\
Wikipedia la Enciclopedia Libre f F x X";
puts("WIKIPEDIA, LA ENCICLOPEDIA LIBRE \n\
Wikipedia la Enciclopedia Libre f F x X");
for (i=0;cadena[i]!='\0';i++)
{
if ((cadena [i]>='u' && cadena [i]<='z') || (cadena [i]>='U' && cadena [i]<='Z' )) cadena [i]-=20;
else if (cadena [i]>='a' && cadena [i]<='t' || (cadena [i]>='A' && cadena [i]<='T' )) cadena [i]+=6;
}
puts("\nY ahora codificada con desplazamiento 6:");
printf ("%s\n",cadena);
puts("\nY ahora descodificada: ");
for (i=0;cadena[i]!='\0';i++)
{
if ((cadena [i]<='f' && cadena [i]>='c') || (cadena [i]<='F' && cadena [i]>='C' )) cadena [i]+=20;
else if (cadena [i]>'f' && cadena [i]<='z' || (cadena [i]>'F' && cadena [i]<='Z' )) cadena [i]-=6;
}
printf ("%s\n",cadena);
/******* AHORA CON ENTRADA DESDE TECLADO *******/
while (1)
{
n=0;
puts ("\n\nIntroduce una frase: ");
fgets (frase,255,stdin);
do
{
printf ("\nIndica el desplazamiento (1-25): ");
scanf ("%d",&n);
while (getchar()!='\n');
}while (n<1 || n>25);
for (i=0;frase[i]!='\0';i++)
{
if ((frase [i]>('z'-n) && frase [i]<='z') || (frase [i]>('Z'-n) && frase [i]<='Z' ))
frase [i]=frase [i]-26+n;
else if (frase [i]>='a' && frase [i]<=('z'-n) || (frase [i]>='A' && frase [i]<=('Z'-n) ))
frase [i]+=n;
}
printf("\nCodificada con desplazamiento %d \n",n);
printf ("%s\n",frase);
puts("\nY ahora descodificada: ");
for (i=0;frase[i]!='\0';i++)
{
if ((frase [i]<('a'+n) && frase [i]>=('a')) || (frase [i]<('A'+n) && frase [i]>=('A') ))
frase [i] =frase [i]+26-n;
else if ((frase [i]>=('a'+n) && frase [i]<=('z') )|| (frase [i]>=('A'+n) && frase [i]<=('Z') ))
frase [i]-=n;
}
printf ("%s\n",frase);
}
/*************** :-) OTRA FORMA :-) **********/
return 0;
}
Con una salida como esta:
WIKIPEDIA, LA ENCICLOPEDIA LIBRE
Wikipedia la Enciclopedia Libre f F x X
Y ahora codificada con desplazamiento 6:
COQOVKJOG, RG KTIOIRUVKJOG ROHXK
Coqovkjog rg Ktioiruvkjog Rohxk l L d D
Y ahora descodificada:
WIKIPEDIA, LA ENCICLOPEDIA LIBRE
Wikipedia la Enciclopedia Libre f F x X
Introduce una frase:
leosansan Abc Xyz
Indica el desplazamiento (1-25): 6
Codificada con desplazamiento 6
rkuygtygt Ghi Def
Y ahora descodificada:
leosansan Abc Xyz
¡Vamos!, a ver si se animan a plantear códigos diferentes, que sean más ingeniosos y/o eficaces. Ya hay una media propuesta de cypascal usando la operación módulo -ver post anteriores-, aunque está incompleto, ya que le falto el descifrado. ;) ;-) ;)Saluditos!. (http://st.forocoches.com/foro/images/smilies/aaaaa.gif)
Y que sucede si en mi cadena tengo algun caracter estilo " @#$*{}", la tabla ascii no es solo el abecedario.
Un saludo.
leosansan, Excelente código, Gracias
(LA PROGRAMACIÓN ES MÁGICA ;D ;D )
pd. Ya inicie a leer unas guías para aprender C
Saludos.
Cita de: leosansan en 12 Junio 2013, 16:58 PM
No es el cifrado César, te muestro una salida al código que planteas:
Como ves, en el cifrado se "va más allá" de la z/Z, cuando debería empezar a contar desde la a.
Aqui va una versión mas eficiente
#include <stdio.h>
#include <ctype.h>
int main()
{
int i, j, des, a, z;
char cadena[30] = "";
printf("Ingrese una cadena: ");
scanf("%s", cadena);
printf("Ingrese desplazamiento: ");
scanf(" %d", &des);
for(i = 0; cadena[i]!='\0'; i++)
{
a = 97;
if(isupper(cadena[i])) a = 65;
cadena[i] =((cadena[i]-a+des)%26)+a;
}
printf("\nCifrado es: %s \n", cadena);
for(j = 0; cadena[j]!='\0'; j++)
{
z = 122;
if(isupper(cadena[j])) z = 90;
cadena[j]= ((cadena[j]-z-des)%26)+z;
}
printf("\nDecifrado es: %s \n", cadena);
return 0;
}
Saludos ::)
Son unos programadores grosos!
Sucedía lo mismo de siempre, no mostraba el resultado si no que se cerraba le agregue esto y listo...
getchar();
getchar();
}
:D eliminando el return 0;
Saludos.
Cita de: Royca3 en 12 Junio 2013, 19:49 PM
Son unos programadores grosos!
Sucedía lo mismo de siempre, no mostraba el resultado si no que se cerraba le agregue esto y listo...
getchar();
getchar();
}
:D eliminando el return 0;
Saludos.
No elimines el return 0, vas en contra del estándar.
Cita de: amchacon en 12 Junio 2013, 20:15 PM
No elimines el return 0, vas en contra del estándar.
Pero si no ha pasado nada el programa corre normalmente!!!?
Intente con el
getchar();
return 0;
getchar();
y no me funciono así que borre el return 0;
;D ?
getchar();
getchar();
return 0;
Cita de: amchacon en 12 Junio 2013, 21:09 PM
getchar();
getchar();
return 0;
Lo intentare de esa manera..
Cita de: amchacon en 12 Junio 2013, 21:09 PM
getchar();
getchar();
return 0;
El return es OBLIGADO en el estandar C , toda vez que de acuerdo dicho estandar la función¡ón main ha de ser declarada con de tipo int y esto conconllleba a un return de un valor int,. Cositas del estandar. Otra cosa es que sin él te funcione ya que algunos compiladores son muy listos y "corrigen" ciertos aspectos del código.
Así que sigue las indicaciones de
amchacon.
Saluditos!. (http://st.forocoches.com/foro/images/smilies/aaaaa.gif)
Cita de: cpu2 en 12 Junio 2013, 18:59 PM
Y que sucede si en mi cadena tengo algun caracter estilo " @#$*{}", la tabla ascii no es solo el abecedario.
Estamos hablando del cifrado César y hay que colocar las cosas en su contexto histórico y de acuerdo a ello proceder,. ¿O es que en la época de Julio César existían los símbolos que mencionas'. Ni existía Internet, y por tanto @ ni los móviles con sus almohadillas # ni el dólar $, etc.
Así que para hacer códigos comparables, siñemosnos al estricto cifrado César.Saluditos!. (http://st.forocoches.com/foro/images/smilies/aaaaa.gif)
Cita de: CCross en 12 Junio 2013, 19:23 PM
Aqui va una versión mas eficiente
Lamentablemente se queda "corto ya que no permite introducir frases. Una pequeña mejora, pero reconozco que está ingenioso.
Saluditos!. (http://st.forocoches.com/foro/images/smilies/aaaaa.gif)
Cita de: leosansan en 12 Junio 2013, 22:01 PM
El return es OBLIGADO en el estandar C , toda vez que de acuerdo dicho estandar la función¡ón main ha de ser declarada con de tipo int y esto con nlleba a un return de un valor int,. Cositas del estandar. Otra cosa es que sin él te funcione ya que algun9s compiladores son muy listos y "corrigen" ciertos aspectos del código.
así que sigue las indicaciones de amchacon.
Saluditos!.
(http://st.forocoches.com/foro/images/smilies/aaaaa.gif)
Ok gracias No sabia eso! :P
El cifrado César lo único que hace para cifrar es sumar 3 a cada carácter y para descifrar restar, por lo que he leido.
Aqui esta el mio pero en ASM.
_C.0:
incl %edx
scasb
jne _C.0
_C.1:
decl %edi
addb $0x3, (%edi)
decl %edx
jne _C.1
movl %edi, %eax
ret
Cita de: leosansan en 12 Junio 2013, 22:09 PM
Así que para hacer códigos comparables, siñemosnos al estricto cifrado César.
Ese código es el cifrado César, solo para el abecedario lo veo una tonteria.
Un saludo.
Cita de: cpu2 en 12 Junio 2013, 23:23 PM
El cifrado César lo único que hace para cifrar es sumar 3 a cada carácter y para descifrar restar, por lo que he leido.
Aqui esta el mio pero en ASM.
_C.0:
incl %edx
scasb
jne _C.0
_C.1:
decl %edi
addb $0x3, (%edi)
decl %edx
jne _C.1
movl %edi, %eax
ret
Ese código es el cifrado César, solo para el abecedario lo veo una tonteria.
Un saludo.
Perdón mi ignorancia pero como ejecuto un code en ASM?
(ademas en el titulo dice C/C++)
Saludos.
Creo que no hace falta compilarlo, con la ventana de comandos de windows probablemente baste.
Edito: Si que hace falta por lo menos un ensamblador.
Salu10
Cita de: cypascal en 13 Junio 2013, 00:06 AM
Creo que no hace falta compilarlo, con la ventana de comandos de windows probablemente baste.
Edito: Si que hace falta por lo menos un ensamblador.
Salu10
Y como lo compilo y ejecuto?
:-X
Cita de: Royca3 en 13 Junio 2013, 00:24 AM
Y como lo compilo y ejecuto?
:-X
no te lies con asm aun
Cita de: Royca3 en 13 Junio 2013, 00:24 AM
Y como lo compilo y ejecuto?
:-X
Creo que esto bastará, no programo en windows.
char *cesar (char *ptr)
{
__asm
{
xor eax, eax
xor ecx, ecx
mov edi, ebx
_C.0:
inc ecx
scasb
jne _C.0
_C.1:
dec edi
add byte ptr [edi], 0x3
dec ecx
jne _C.1
mov eax, edi
}
}
La llamas como si fuera una función de C, y esta te devuelve un puntero, haber si te funciona.
Un saludo.
La verdad es que no se NADA sobre ASM u__u y tienes razón no me meteré aun con ASM ;D
Saludos.
Cita de: cpu2 en 12 Junio 2013, 23:23 PM
El cifrado César lo único que hace para cifrar es sumar 3 a cada carácter y para descifrar restar, por lo que he leido.
Pues no, no es añadir estrictamente 3 sino un valor que el usuario indique. Vamos que es variable.Cita de: cpu2 en 12 Junio 2013, 23:23 PM
Ese código es el cifrado César, solo para el abecedario lo veo una tonteria.
Creo que no has pillado la "intención" de la propuesta. Se trata de ver el "potencial" de los intervinientes en el foro y que en base a este tema hagan propuestas ingeniosasy/o "raritas" con la única finalidad de exprimir el ingenio, plasmar los conocimientos y ayudar/enseñar unos de otros.
Tu propuesta, por ejemplo, sería clasificada como "rarita", ya que se sale del contexto habitual y,como tal propuesta "rarita" yo la aplaudo. Eso sí, echo en falta que expliques a los neófitos en el tema del ensamblador qué herramientas serían necesarias para poder ejecutar el código que planteas. Ya que haces una propuesta fuera del contexto C/C++ una breve explicación no nos vendría mal a algunos.Saluditos!
(http://st.forocoches.com/foro/images/smilies/aaaaa.gif)
Un "collage" con las propuestas de cypascal y CCross reunidas pero con función:#include <stdio.h>
#include <ctype.h>
void cesar (char *cadena , int letra, int des);
int main()
{
int des=0, letra=0;
char cadena[256]={0};
puts("Ingrese una cadena: ");
gets (cadena);
//fgets (cadena,255,stdin);
do
{
puts("\nIndica el desplazamiento: ");
scanf ("%d",&n);
while (getchar()!='\n');
}while (n<0 || n>25);
cesar (cadena,97,des);
printf("\nCifrado es: %s \n", cadena);
cesar (cadena,122,-des);
printf("\nDecifrado es: %s \n", cadena);
return 0;
}
void cesar (char cadena[] , int letra, int des)
{
int i,letra1=0;
for(i = 0; cadena[i]; i++)
{
if ( cadena[i]==' ')
continue;
letra1=letra;
if(isupper(cadena[i])!=0)
letra1-=32;
cadena[i] =((cadena[i]-letra1+des)%26)+letra1;
}
}
Saluditos!
(http://st.forocoches.com/foro/images/smilies/aaaaa.gif)
P.D:Cita de: cpu2 en 13 Junio 2013, 01:13 AM
Creo que esto bastará, no programo en windows.
char *cesar (char *ptr)
{
__asm
..............
}
¿Una función char que no retorna nada?.
Cita de: leosansan en 14 Junio 2013, 11:11 AM
P.D:
¿Una función char que no retorna nada?.
ret
Cita de: cpu2 en 14 Junio 2013, 12:22 PM
ret
¿ret?, en C/C++ lo que yo conozco es return. Y aún así no funciona. ¿Podrías indicarnos que programa usas para programar en ensamblador?.
Y puestos a buscar soluciones "curiosas" me dio por encontrar una solución en que sólo se sume y reste el desplazamiento y sin más librería que stdio. Eso sí, he tenido que usar un "señor" array, pero conseguí el propósito mencionado:#include <stdio.h>
int main(void)
{
char frase[256] = {0};
int i=0,j=0,desplazamiento=0;
char abc[] = {"ABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyzabcdefghijklmnopqrstuvwxyzabcdefghijklmnopqrstuvwxyz"};
while(1)
{
printf("Ingrese una cadena: ");
fgets (frase,255,stdin);
do
{
puts("Ingrese desplazamiento: (1-25) ");
scanf ("%d",&desplazamiento);
while (getchar()!='\n');
}while (desplazamiento<0 || desplazamiento>25);
for(i = 0; frase[i]; i++)
{
for(j = 0; j<26; j++)
{
if ( frase[i]==' ')
break;
if (frase[i]==abc[j])
{
frase[i] = abc[j+ desplazamiento];
break;
}
}
for(j = 78; j<104; j++)
{
if ( frase[i]==' ')
break;
if (frase[i]==abc[j])
{
frase[i] = abc[j+ desplazamiento];
break;
}
}
}
printf("Cadena cifrada: %s\n\n", frase);
for(i = 0; frase[i]; i++)
{
for(j = 52;j<78 ; j++)
{
if ( frase[i]==' ')
break;
if (frase[i]==abc[j])
{
frase[i] = abc[j- desplazamiento];
break;
}
}
for(j = 130;j<156 ; j++)
{
if ( frase[i]==' ')
break;
if (frase[i]==abc[j])
{
frase[i] = abc[j- desplazamiento];
break;
}
}
}
printf("Cadena descifrada: %s\n", frase);
}
return 0;
}
Saluditos!
(http://i1280.photobucket.com/albums/a497/leosansan/leosan1/leones%20peques/leonpersonajepeque_zps25f3622f.jpg)
El código es similar al anterior
Cita de: leosansan en 16 Junio 2013, 19:37 PM
¿ret?, en C/C++ lo que yo conozco es return. Y aún así no funciona. ¿Podrías indicarnos que programa usas para programar en ensamblador?.
GAS, ld y cualquier editor de texto.
Cita de: leosansan en 16 Junio 2013, 19:37 PM
Y puestos a buscar soluciones "curiosas" me dio por encontrar una solución en que sólo se sume y reste el desplazamiento y sin más librería que stdio. Eso sí, he tenido que usar un "señor" array, pero conseguí el propósito mencionado
Código demasiado largo y poco eficiente, para algo tan simple como el cifrado de César, y sin hablar del array.
printf("Ingrese una cadena: ");
fgets (frase,255,stdin);
puts("Ingrese desplazamiento: (1-25) ");
scanf ("%d",&desplazamiento);
Podrías utilizar los parametros argv, te ahorras ese fgets y esa aberración llamada scanf.
while (getchar()!='\n');
Venga hombre.
Un saludo.
P.D: Yo dejo el tema, desde mi punto de vista un desplazamiento de bits o un xor, es mucho mejor que el de César, yo no daría más importancia a esto de verdad.
Cita de: cpu2 en 17 Junio 2013, 08:47 AMPodrías utilizar los parametros argv, te ahorras ese fgets y esa aberración llamada scanf.
Perdiendo facilidad y sencillez...
Personalmente, si eres un poco peresozo y no quieres usar teclado. Simplemente pones esto al ejecutar el programa
programa < fichero.txt
Y en fichero pones todos los parámetros del teclado.
Cita de: cpu2 en 17 Junio 2013, 08:47 AMwhile (getchar()!='\n');
Venga hombre.
Es un método tan válido como cualquier otro para "limpiar" el buffer del teclado.
Cita de: cpu2 en 17 Junio 2013, 08:47 AMP.D: Yo dejo el tema, desde mi punto de vista un desplazamiento de bits o un xor, es mucho mejor que el de César, yo no daría más importancia a esto de verdad.
Bueno es curiosidad educativa :silbar:
Cita de: cpu2 en 17 Junio 2013, 08:47 AM
Código demasiado largo y poco eficiente, para algo tan simple como el cifrado de César, y sin hablar del array.
Ya,ya, pero para mejor eficiencia ya están los dos posteados con anterioridad, el de CCross/cypscal/mío con una función y el uso de la librería ctype y el anterior mío sin más uso que el de el caracter ascii. Aún espero una propuesta "completa" por parte tuya, donde se pueda introducir la frase y el desplazamiento, no tan solo mover tres posiciones los caracteres sin tener en cuenta que si se llega al final se vuelve al comienzo del abecedario e incluir tanto minúsculas como mayúsculas e incluya sus printf y toda esa retaila.De todas formas olvidas lo fundamental:Cita de: leosansan en 13 Junio 2013, 16:48 PM
Creo que no has pillado la "intención" de la propuesta. Se trata de ver el "potencial" de los intervinientes en el foro y que en base a este tema hagan propuestas ingeniosasy/o "raritas" con la única finalidad de exprimir el ingenio, plasmar los conocimientos y ayudar/enseñar unos de otros.
Cita de: cpu2 en 17 Junio 2013, 08:47 AM
printf("Ingrese una cadena: ");
fgets (frase,255,stdin);
puts("Ingrese desplazamiento: (1-25) ");
scanf ("%d",&desplazamiento);
Podrías utilizar los parametros argv, te ahorras ese fgets y esa aberración llamada scanf.
Sí hombre, y volver cuarenta años atrás a programar tipo MS-DOS metiendo todo a mano y a base de comandos. No, lo siento, pero yo ya pasé esa época, tanto que por líneas de comandos no hagoun código ni de coñ*. Como ya he menciona en alguna ocasión, en mi caso al menos, un pasito atrás ni "pá" coger impulso.
Cita de: cpu2 en 17 Junio 2013, 08:47 AM
while (getchar()!='\n');
Venga hombre.
¿Alguna sugerencia diferente para limpiar el buffer?. No creo que seas de los que usan fflush (stdin). Reconozco humildemente que mis conocimientos en este mundillo del C/C++ son aún escasos, pero ¿alguna propuesta alternativa que nos ilumine a los neófitos?Cita de: cpu2 en 17 Junio 2013, 08:47 AM
Un saludo.
Yo un "Saludito", como se dice por mi tierra. Lamento que hallas "agriado" un poco el tema con tu actitud ¿prepotente?, cuando mi única finalidad era que el personal que nos sigue se animase al "juego" de proponer alternativas al tema del cifrado César, como podría haber sido cualquier otro. Era sólo un reto que, por lo que veo y leo, tú te lo has tomado como algo personal, cosa que no entiendo. Y menos aún el que, repito/insisto no hallas hecho una propuesta alternativa en toda regla. Para criticar por criticar y deshagorte creo que existen otros foros más apropiados.
Cita de: cpu2 en 17 Junio 2013, 08:47 AM
P.D: Yo dejo el tema, desde mi punto de vista un desplazamiento de bits o un xor, es mucho mejor que el de César, yo no daría más importancia a esto de verdad.
Pues ya ves, esa es una sugerencia, que no aportación, que no caerá en saco roto, aunque intuyendo tu preparación habría esperado más de tí.Como diría amchacon:Cita de: amchacon en 17 Junio 2013, 11:37 AM
Perdiendo facilidad y sencillez...
Personalmente, si eres un poco peresozo y no quieres usar teclado. Simplemente pones esto al ejecutar el programa
programa < fichero.txt
Y en fichero pones todos los parámetros del teclado.
Es un método tan válido como cualquier otro para "limpiar" el buffer del teclado.
Bueno es curiosidad educativa :silbar:
En fin, que si se te hace muy largo el array del último post, y cuya única finalidad era la que era- releete el post para que lo entiendas bien- te dejo "otro" código con dos arrays, aunque más cortitos que el anterior , eso sí sin usar librerías, ni la operación módulo ni tan siquiera el código ascii, a"a pelo" vamos.....¡y que siga el juego! :#include <stdio.h>
int main(void)
{
char frase[256] = {0};
int i=0,j=0,desplazamiento=0;
char ABC[] = {"ABCDEFGHIJKLMNOPQRSTUVWXYZ"};
char abc[] = {"abcdefghijklmnopqrstuvwxyz"};
while(1)
{
printf("Ingrese una cadena: ");
fgets (frase,255,stdin);
do
{
puts("Ingrese desplazamiento: (1-25) ");
scanf ("%d",&desplazamiento);
while (getchar()!='\n');
}while (desplazamiento<0 || desplazamiento>25);
for(i = 0; frase[i]; i++)
{
for(j = 0; j<26; j++)
{
if ( frase[i]==' ')
break;
if (frase[i]==abc[j] && j+ desplazamiento<26)
{
frase[i] = abc[j+ desplazamiento];
break;
}
else if (frase[i]==abc[j] && j+ desplazamiento>=26)
{
frase[i] = ABC[j+ desplazamiento-26];
break;
}
else if (frase[i]==ABC[j] && j+ desplazamiento<26)
{
frase[i] = ABC[j+ desplazamiento];
break;
}
else if (frase[i]==ABC[j] && j+ desplazamiento>=26)
{
frase[i] = abc[j+ desplazamiento-26];
break;
}
}
}
printf("Cadena cifrada: %s\n\n", frase);
for(i = 0; frase[i]; i++)
{
for(j = 0; j<26; j++)
{
if ( frase[i]==' ')
break;
if (frase[i]==abc[j] && j- desplazamiento>=0)
{
frase[i] = abc[j- desplazamiento];
break;
}
else if (frase[i]==abc[j] && j- desplazamiento<0)
{
frase[i] = ABC[j-desplazamiento+26];
break;
}
else if (frase[i]==ABC[j] && j- desplazamiento>=0)
{
frase[i] = ABC[j- desplazamiento];
break;
}
else if (frase[i]==ABC[j] && j- desplazamiento<=0)
{
frase[i] = abc[j-desplazamiento+26];
break;
}
}
}
printf("Cadena descifrada: %s\n\n", frase);
}
return 0;
}
Saluditos!
(http://i1280.photobucket.com/albums/a497/leosansan/leosan1/leones%20peques/lion14peque_zps1d213b80.jpg)
emm! Sera que alguien puede pasarme un manual de C++??
Bien explicado y completo? en los que hayan aprendido ustedes
saludos
(tengo unos pero, no entiendo claramente)
La verdad es que, en principio y sólo en principio, el cifrado César es muy endeble ante un simple ataque por "fuerza bruta". Bastaría ir probando con un desplazamiento de 0 a 26 y ver cuando obtenemos "algo" razonable/entendible.
Como muestra esta imagen en que se observa el cifrado y el "ataque" para descifrarlo:(http://i1280.photobucket.com/albums/a497/leosansan/GRAFICOS1/descifrado_zps9055047e.jpg)
Cosa que consigo con este simple código:
#include <stdio.h>
int main (void)
{
int i,n=0,k=0;
char frase [256];
while (1)
{
n=0;
puts ("\nIntroduce una frase: ");
fgets (frase,255,stdin);
do
{
printf ("\nIndica el desplazamiento: ");
scanf ("%d",&n);
while (getchar()!='\n');
}while (n<0 || n>25);
for (i=0;frase[i];i++)
{
if ((frase [i]>('z'-n) && frase [i]<='z') || (frase [i]>('Z'-n) && frase [i]<='Z' ))
frase [i]=frase [i]-26+n;
else if (frase [i]>='a' && frase [i]<=('z'-n) || (frase [i]>='A' && frase [i]<=('Z'-n) ))
frase [i]+=n;
}
printf("\nCodificada con desplazamiento %d \n",n);
printf ("%s\n",frase);
puts("\nY ahora descodificada: ");
for (k=0;k<26;k++)
{
for (i=0;frase[i];i++)
{
if ((frase [i]<('a'+k) && frase [i]>=('a')) || (frase [i]<('A'+k) && frase [i]>=('A') ))
frase [i] =frase [i]+26-k;
else if ((frase [i]>=('a'+k) && frase [i]<=('z') )|| (frase [i]>=('A'+k) && frase [i]<=('Z') ))
frase [i]-=k;
}
printf ("%s\n\n",frase);
system ("pause");
}
}
return 0;
}
Pero, pero .... con una pequeña variante lo podríamos hacer "casi" indescifrable, según comenta la Wikipedia:"El cifrado Vigenère usa el cifrado César con un
desplazamiento diferente en cada posición del texto;
el valor del desplazamiento se define usando una
palabra clave repetitiva. Si la palabra clave fuera
escogida al azar y tan larga como el mensaje
(para que no se repita), el sistema resultante sería,
en teoría, indescifrable."
No creo que sea para tanto, pero cualquiera le lleva la contraria a Wikipedia. Como ya comenté en otro post anterior, más información en Cifrado César cortesía de Wikipedia (http://es.wikipedia.org/wiki/Cifrado_C%C3%A9sar).Por si acaso, yo lo he intentado usando un valor aleatorio como desplazamiento para cada letra, con lo que el ataque por fuerza bruta anterior no sirve de nada, tendría que ser algo más refinado y respaldado con "potencia".
Como muestra otra imagen:(http://i1280.photobucket.com/albums/a497/leosansan/GRAFICOS1/cifrado1_zps131723c9.jpg)
Pueden observar que la letra "a", que en el cifrado normal sería siempre la misma letra desplazada, ahora cambia debido a que he usado un array aleatorio para cifrar la frase y sin ese array se haría muy "cuesta arriba" el descifrado ya que incluso el análisis por frecuencia fallaría, por la aleatoriedad mencionada.
Y he aquí la pequeña, pero "potente", variación:#include <stdio.h>
#include <ctype.h>
#include <time.h>
void cesar (char *cadena , int letra, int *key);
int main()
{
int i=0,des=0, letra=0;
srand((unsigned) time(NULL));
char cadena[0x100]={0};
puts("Ingrese una cadena: ");
gets (cadena);
int key[strlen(cadena)],_key[strlen(cadena)];
for (i = 0;cadena[i]; i++)
{
key[i] = rand() % 0x1A+1 ;
_key[i] = -key[i];
}
cesar (cadena,0x61,key);
printf("\nCifrado es: %s \n", cadena);
cesar (cadena,0x7A,_key);
printf("\nDecifrado es: %s \n", cadena);
return 0;
}
void cesar (char cadena[] , int letra, int *des)
{
int i,letra1=0;
for(i = 0; cadena[i]; i++)
{
if ( cadena[i]==' ')
continue;
letra1=letra;
if(isupper(cadena[i])!=0)
letra1-=0x20;
cadena[i] =((cadena[i]-letra1+des[i] )%0x1A)+letra1;
}
}
Como veis he sustituido los números decimales por sus correspondientes hexadecimales, sólo por darle un aspecto más críptico al tema.
Y el código propuesto tiene un valor añadido y es que cada mensaje usa un array aleatorio diferente en cada caso, por lo que la "interseptación" de dos mensajes no ayudaría en nada a los desencriptadores.
¡Anímate!, e intenta el descifrado de:
Cifrado es: Uhmvhvcjsakoo Mqoug ctz Xwrvdm Gycgs
Saluditos!
(http://st.forocoches.com/foro/images/smilies/aaaaa.gif)
Muy interesante Leo ;-)
En mis tiempos hize un programa para descifrar palabras cifradas con César:
https://dl.dropboxusercontent.com/u/69551225/Desencriptador%20Secuencias%20%28WINDOWS%29.rar
https://dl.dropboxusercontent.com/u/69551225/Desencriptador%20Secuencias%20%28SOURCE%29.zip
Tienes que representar las letras con números y el programa busca palabras que sigan ese patrón. Por ejemplo:
CitarAmanda
Introduciríamos en el programa:
Citar121341
(cada letra se corresponde con un número).
El programa entonces, detectaría que la primera letra,tercera y quinta letra son iguales. Entonces buscaría en su base de datos palabras que sigan ese patrón. Los nombres obtenidos se guardan en un txt (deseché mostrarlos por pantalla por si la lista fuese muyy larga).
Cita de: amchacon en 18 Junio 2013, 22:44 PM
Muy interesante Leo ;-)
Gracias, gracias e idem por el tuyo.
Ya para finalizar el tema no quería quedarme sin "actualizar" el cifrado César, que por motivos históricos se reducía al abecedario, y permitir el cifrado de buena parte del código ascii. Vamos que al menos contuviera símbolos tan actuales como $, ~, #, @ <, >, " ¿, ¡ etc. Una salida como ejemplo:
Ingrese una cadena:
"leosan@san.com! ¿san? #leo+san|=\12345!==$$
Cifrado es: Vêt├ôíêc£x}enªºn ╣░®╗M C├ç┼k¢qçðv|3xKfT=pYxx
Decifrado es: "leosan@san.com! ¿san? #leo+san|=\12345!==$$
Y esta vez no voy a colgar el código, creo que si han seguido los posts anteriores no tendrán ninguna dificultad en obtenerlo. Sólo pretendía poner de manifiesto que, como ya comenté, el cifrado César "actual" va más allá del simple abecedario original.
Saluditos!
(http://i1280.photobucket.com/albums/a497/leosansan/leosan1/leones%20peques/leonpersonajepeque_zps25f3622f.jpg)
(He corregido algun error que había en alguna cadena. Ahora si que creo que está todo bien)
He añadido comprobación de parámetros en la función vigenere, que es donde deberían estar si se quiere añadir a una librería.
Ahora, en lugar de definir un nombre de fichero temporal mediante una constante simbolica, utilizo tmpnam() para asegurarme de que el nombre no se corresponda con ningún archivo existente.
¡Buenas!
Como estáis recopilando códigos de cifrado, aquí os dejo el del cifrado de Vigenere.
El código es un podo largo. El cifrado de Vigenere está al principio. Hay repeticion de código, pero lo he hecho así para evitar introducir demasiadas condiciones en el codigo de cifrado, ya que para archivos extensos el algoritmo sería muy lento. El codigo que hay en main es basicamente para leer y clasificar los datos de la linea de comandos (por eso es tan largo). Al final os dejo la sintaxis de llamada desde la linea de comandos y algunos ejemplos:
/*
* Cifrado de Vigenere
*
* Linea de comandos: nombre_programa -a alfabeto -[c|f] cadena|fichero -k clave -s [fichero_salida | std]
*
* Cifra los caracteres de la cadena (parametros -c "Cadena") o el fichero (parametros -f nombre_del_fichero)
* que coincidan con los del alfabeto (- que se le pasa, utilizando la clave dada (parametros -k clave)
*
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define MAX_STRLEN 256
/*
* Las siguientes constantes sirven para decidir si la funcion vigenere actuara sobre una cadena o un fichero
*/
#define IN_CADENA 0
#define IN_FICHERO 1
/*
* Constantes para decidir si la funcion vigenere cifrara o descifrara la entrada
*/
#define ACCION_CIFRAR 0
#define ACCION_DESCIFRAR 1
/*
* vigenere(tipo entrada (cadena o fichero), cadena con texto o nombre de fichero con el input,
* alfabeto, clave, cifrar|descifrar, fichero de salida (stdin o nombre de arcivo))
*
* valor de retorno: 0 si algo falla, 1 si todo va bien.
*/
unsigned long flen(FILE *f)
{
unsigned long pos,len;
pos = ftell(f);
fseek(f,0,SEEK_END);
len = ftell(f);
fseek(f,pos,SEEK_SET);
return len;
}
int vigenere(int modo_input, char *cadena, char *alfabeto, char *clave, char accion, FILE *salida)
{
FILE *entrada = NULL;
unsigned long longitud,tam_bloque,tam_clave,tam_alfabeto;
char *lector,*busqueda;
int *desplazamiento,i,indice = 0;
/* comprobamos que en alfabeto no haya letras repetidas */
for(i = 0 ; alfabeto[i] ; i++)
if(strchr(alfabeto + i + 1,alfabeto[i]))
return 0;
/* comprobamos que los caractreres de la clave esten en el alfabeto */
for(i = 0 ; clave[i] ; i++)
if(!strchr(alfabeto,clave[i]))
return 0;
if(modo_input == IN_CADENA)
longitud = strlen(cadena);
else if(modo_input == IN_FICHERO)
{
if(!(entrada = fopen(cadena,"rb")))
return 0;
longitud = flen(entrada);
}
else
return 0;
tam_alfabeto = strlen(alfabeto);
tam_clave = tam_bloque = strlen(clave);
if(!(lector = malloc(tam_bloque * sizeof(char))))
return 0;
if(!(desplazamiento = malloc(tam_bloque * sizeof(int))))
{
free(lector);
return 0;
}
for(i = 0 ; clave[i] ; i++)
{
if(!(busqueda = strchr(alfabeto,clave[i])))
{
free(lector);
free(desplazamiento);
return 0;
}
if(accion == ACCION_CIFRAR)
desplazamiento[i] = busqueda - alfabeto;
else if(accion == ACCION_DESCIFRAR)
desplazamiento[i] = alfabeto - busqueda; /* el desplazamiento de descifrado es el opuesto del de cifrado */
else
{
free(lector);
free(desplazamiento);
return 0;
}
}
if(modo_input == IN_CADENA)
{
while(longitud)
{
if(longitud < tam_bloque)
tam_bloque = longitud;
memcpy(lector,cadena,tam_bloque * sizeof(char));
for(i = 0 ; i < tam_bloque ; i++)
{
if(busqueda = strchr(alfabeto,lector[i]))
{
lector[i] = alfabeto[((busqueda - alfabeto) + desplazamiento[indice] + tam_alfabeto) % tam_alfabeto];
indice = (indice + 1) % tam_clave;
}
}
fwrite(lector , tam_bloque*sizeof(char) , 1 , salida);
if((tam_bloque == tam_clave) && (longitud > tam_bloque))
cadena += tam_bloque;
longitud -= tam_bloque;
}
}
else if(modo_input == IN_FICHERO)
{
while(longitud)
{
if(longitud < tam_bloque)
tam_bloque = longitud;
fread(lector,tam_bloque * sizeof(char),1,entrada);
for(i = 0 ; i < tam_bloque ; i++)
{
if(busqueda = strchr(alfabeto,lector[i]))
{
lector[i] = alfabeto[((busqueda - alfabeto) + desplazamiento[indice] + tam_alfabeto) % tam_alfabeto];
indice = (++indice) % tam_clave;
}
}
fwrite(lector , tam_bloque*sizeof(char) , 1 , salida);
longitud -= tam_bloque;
}
}
else
{
if(entrada)
fclose(entrada);
free(lector);
free(desplazamiento);
return 0;
}
if(entrada)
fclose(entrada);
free(lector);
free(desplazamiento);
return 1;
}
int main(int argc, char *argv[])
{
char error[] = "%s [+|-]c (cifrar|descifrar) -a alfabeto -[s|f] [cadena|fichero] -k clave -o [fichero_salida|std]\n";
char alfabeto[MAX_STRLEN] = "", fentrada[MAX_STRLEN] = "", clave[MAX_STRLEN] = "", fsalida[MAX_STRLEN] = "";
char *texto = NULL,*ftemp = NULL;
int sobreescribir = 0, modo = ACCION_CIFRAR + ACCION_DESCIFRAR + 1, i, j;
FILE *f;
if(argc != 10)
{
fprintf(stderr,error,argv[0]);
return 1;
}
for(i = 1 ; i < 10 ; i += 2)
{
if(!strcmp(argv[i],"-a"))
{
if(strlen(alfabeto))
{
fprintf(stderr,"-a: Parametro repetido\n");
return 2;
}
if(!strlen(argv[i + 1]))
{
fprintf(stderr,"-a: Alfabeto vacio no permitido\n");
return 3;
}
for(j = 0 ; argv[i + 1][j] ; j++)
{
if(strchr(argv[i + 1] + j + 1, argv[i + 1][j]))
{
fprintf(stderr,"-a: No se permiten caracteres repetidos en el alfabeto\n");
return 4;
}
}
if(strlen(argv[i + 1]) >= MAX_STRLEN)
{
fprintf(stderr,"-a: El alfabeto es demasiado largo. Se truncara a partir del caracter %d.\n",MAX_STRLEN);
strncpy(alfabeto, argv[i + 1], 255);
alfabeto[255] = '\0';
}
else
strcpy(alfabeto,argv[i + 1]);
}
else if(!strcmp(argv[i],"-s"))
{
if(texto || strlen(fentrada))
{
fprintf(stderr,"-s: Parametro -[s|f] repetido\n");
return 5;
}
texto = argv[i + 1];
}
else if(!strcmp(argv[i],"-f"))
{
if(texto || strlen(fentrada))
{
fprintf(stderr,"-f: Parametro -[s|f] repetido\n");
return 6;
}
if(strlen(argv[i + 1]) >= MAX_STRLEN)
{
fprintf(stderr,"-f: El nombre del fichero de entrada no debe exceder los %d caracteres\n",MAX_STRLEN);
return 7;
}
if(!strlen(argv[i + 1]))
{
fprintf(stderr,"-f: El nombre del fichero de entrada no puede estar vacio\n");
return 8;
}
strcpy(fentrada,argv[i + 1]);
}
else if(!strcmp(argv[i],"-k"))
{
if(strlen(clave))
{
fprintf(stderr,"-k: Parametro -k repetido\n");
return 9;
}
if(!strlen(argv[i + 1]))
{
fprintf(stderr,"-k: No se permiten claves vacias");
return 10;
}
if(strlen(argv[i + 1]) >= MAX_STRLEN)
{
fprintf(stderr,"-k: La clave no debe exceder los %d caracteres\n",MAX_STRLEN);
return 11;
}
if(!strlen(argv[i + 1]))
{
fprintf(stderr,"-k: La clave no puede ser vacia\n");
return 12;
}
strcpy(clave,argv[i + 1]);
}
else if(!strcmp(argv[i],"-o"))
{
if(strlen(fsalida))
{
fprintf(stderr,"-o: Parametro -o repetido\n");
return 13;
}
if(strlen(argv[i + 1]) >= MAX_STRLEN)
{
fprintf(stderr,"-o: El nombre del fichero de salida no debe exceder los %d caracteres\n",MAX_STRLEN);
return 14;
}
if(!strlen(argv[i + 1]))
{
fprintf(stderr,"-o: El nombre del fichero de salida no puede estar vacio\n");
return 15;
}
strcpy(fsalida,argv[i + 1]);
}
else if(!strcmp(argv[i],"-c"))
{
if(!(modo ^ ACCION_CIFRAR) || !(modo ^ ACCION_DESCIFRAR))
{
fprintf(stderr,"-c: Parametro [+|-]c repetido\n");
return 16;
}
modo = ACCION_DESCIFRAR;
i--; /* este argumento no tiene asociado un parametro */
}
else if(!strcmp(argv[i],"+c"))
{
if(!(modo ^ ACCION_CIFRAR) || !(modo ^ ACCION_DESCIFRAR))
{
fprintf(stderr,"+c: Parametro [+|-]c repetido\n");
return 17;
}
modo = ACCION_CIFRAR;
i--; /* este argumento no tiene asociado un parametro */
}
else
{
fprintf(stderr,error,argv[0]);
return 18;
}
}
//comprobar que los caracteres de la clave estan en el alfabeto
for(i = 0 ; clave[i] ; i++)
{
if(!strchr(alfabeto,clave[i]))
{
fprintf(stderr,"La clave contiene caracteres que no aparecen en el alfabeto\n");
return 19;
}
}
if(!strcmp(fsalida,"std"))
{
f = stdout;
}
else
{
if(f = fopen(fsalida,"r"))
{
char opcion;
do{
printf("El fichero de salida ya existe. Deseas sobreescribirlo? (S/N) ");
opcion = getchar();
if(opcion > 'Z')
opcion += 'S' - 's';
while(getchar() != '\n');
}while(opcion!= 'S' && opcion!='N');
if(opcion == 'S')
{
fclose(f);
ftemp = tmpnam(NULL);
if(!(f = fopen(ftemp,"wb")))
{
fprintf(stderr,"%s: Error al abrir el fichero de salida\n",argv[0]);
return 20;
}
sobreescribir = 1;
}
else
{
fclose(f);
return 0;
}
}
else
f = fopen(fsalida,"wb");
}
//si fsalida == std llamar a vigenere con los parametro y fsalida stdout
if(texto)
{
if(!vigenere(IN_CADENA,texto,alfabeto,clave,modo,f))
{
fprintf(stderr,"%s no ha podido cifrar la informacion dada\n",argv[0]);
return 21;
}
}
else
{
if(!vigenere(IN_FICHERO,fentrada,alfabeto,clave,modo,f))
{
fprintf(stderr,"%s no ha podido cifrar la informacion dada\n",argv[0]);
return 21;
}
}
if(strcmp(fsalida,"std"))
fclose(f);
if(sobreescribir)
{
remove(fentrada);
rename(ftemp,fentrada);
remove(ftemp);
}
return 0;
}
Sintaxis (mi programa se llama vigenere.exe):
vigenere [+|-]c (cifrar|descifrar) -a alfabeto -[s|f] [cadena|fichero] -k clave -o [fichero_salida|std]
+c cifrar
-c descifrar
-a "alfabeto de cifrado": Los caracteres que se consideraran para cifrar.
-s "cadena": La entrada es la cadena que sigua a -s
-f "fichero": La entrada se tomara del nombre de fichero que siga a -f
-k "clave": Indicara el desplazamiento de los caracteres del texto en claro
-o "fichero": La salida se producira en el fichero indicado.
-o std: La salida se producira por pantalla.
Ejemplos:
vigenere +c -a "abcdefghijklmnopqrstuvwxyz .:,;-+" -s "el simbolo de la suma es + y el de la resta -" -k "ejemplo de clave" -o std
iu;;xxphostfp +e;.yypesl,dt ee+;hn;xpe-;vx :j
vigenere +c -a "abcdefghijklmnopqrstuvwxyz .:,;-+" -s "el simbolo de la suma es + y el de la resta -" -k "ejemplo de clave" -o prueba.txt
type prueba.txt
iu;;xxphostfp +e;.yypesl,dt ee+;hn;xpe-;vx :j
vigenere -c -a "abcdefghijklmnopqrstuvwxyz .:,;-+" -s "iu;;xxphostfp +e;.yypesl,dt ee+;hn;xpe-;vx :j" -k "ejemplo de clave" -o std
el simbolo de la suma es + y el de la resta -
vigenere -c -a "abcdefghijklmnopqrstuvwxyz .:,;-+" -f prueba.txt -k "ejemplo de clave" -o std
el simbolo de la suma es + y el de la resta -
El orden de los argumentos no tiene que ser el que he utilizado yo, puede ser cualquier otro.
¡Saludos!
PD:
El codigo ensablador que ha dejado cpu2 sigue la sintaxis Intel. GCC utiliza sintaxis AT&T (creo).
cpu2, ¿podrías traducir tu código de Intel a AT&T?
No soy tan experto como uds pero siguiendo la regla del cifrado de Cesar logre hacer mi codigo asi:
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
int main(){
char oracion[100];
char codificado[100];
int i, desp;
for(i = 0; i < 100; i++)
codificado[i] = '\0';
printf("Ingrese una oracion para cifrar:\n");
fgets(oracion, 100, stdin);
do{
printf("Ingrese el numero de desplazamientos en el abecedario: ");
scanf("%d", &desp);
if(desp > 27)
printf("El desplazamiento no puede ser mayor a 27\n\n");
}
while(desp > 27);
for(i = 0; i < strlen(oracion); i++){
if(oracion[i] >= 65 && oracion[i] <= 90){
if(oracion[i] + desp > 90)
codificado[i] = (oracion[i] - 90) + desp + 64;
else
codificado[i] = oracion[i] + desp;
}
else{
if(oracion[i] >= 97 && oracion[i] <=122){
if(oracion[i] + desp > 122)
codificado[i] = (oracion[i] - 122) + desp + 96;
else
codificado[i] = oracion[i] + desp;
}
else
codificado[i] = oracion[i];
}
}
printf("La oracion codificada es: %s", codificado);
getchar();
return 0;
}
EDIT: Olvide inicializar el array "codificado" ;D