Lo que necesito es que dado un array de largo 80, cree otro que re ordene los datos del anterior según la clase de los datos.
En otras palabras, entra un código de letras y numero, y necesito que en el nuevo arreglo, en cada casilla se guarden los números y letras por separado según se lean.
Adjunto una imagen para que me entiendan mejor, el primero es el código que entra, y lo de abajo es como necesito que quede el otro arreglo.
(http://i1113.photobucket.com/albums/k503/ZedGe1505/asd12_zpscb7034d1.png)
Este es el código erróneo que tengo hasta ahora:
#include <ctype.h>
#include <stdio.h>
#include <stdlib.h>
#include <iostream>
#include <stdio.h>
#include <string.h>
using namespace std;
char* organizar(char *operacion);
int main()
{
char dato_entrada[80];
cout << "Ingrese Codigo: ";
cin >> dato_entrada;
char* operacion = organizar(dato_entrada);
printf(operacion);
return 0;
}
char* organizar (char* operacion){
int largo=0; //largo total
int largo2=0; //recorre aux2
int largo3=0; //recorre el codigo inicial
while (operacion[largo]!='\0') largo++;//cuenta el largo del codigo
char* aux2 = '\0'; //array que tendra la separaciones
while(largo3 < largo){
if(operacion[largo3] != '0' || operacion[largo3] != '1' || operacion[largo3] != '2'){
aux2[largo2] = strcpy(aux2[largo2],operacion[largo3]); //si encuentra un 0 1 o 2 lo guarda
printf(aux2[largo2]);
largo3++;
}
else if (operacion[largo3] != 'a' || operacion[largo3] != 'b' || operacion[largo3] != 'c'){
aux2[largo2] = strcpy(aux2[largo2],&operacion[largo3]); //si encuentra un a b o c lo guarda
printf(aux2[largo2]);
largo3++;
}
else largo2++;
}
return aux2;
}
No he leído tu código, pero has incluido "stdio.h" dos veces.
Más fácil todavía. Primero miras el tamaño de la char, todo y que sabemos su tamaño.
dato_entrada.size();
Ahora con un ciclo vas mirando una por una diciendo si es letra o número;
... !variable > Z ...
Esto lo verifica, entonces haces que vaya almacenando contenido a una misma posición que ya iras controlando con una variable y con append vas añadiendo, y en caso de que esto cambie (haces un else puro y duro) lo vaya almacenando a la siguiente posición y vaya ordenándolo. Y listo.
Cita de: Puntoinfinito en 2 Abril 2013, 21:03 PM
Esto lo verifica, entonces haces que vaya almacenando contenido a una misma posición que ya iras controlando con una variable y con append vas añadiendo, y en caso de que esto cambie (haces un else puro y duro) lo vaya almacenando a la siguiente posición y vaya ordenándolo. Y listo.
use el append pero me da problemas al querer trabajar con char* :/
Si muestras parte del código te puedo ayudar mucho mejor ;)
Ahora que caigo append es un parametro de la librería string, pero de todas maneras creo que se puede utilizar con char. La utilización tendría que ser;
char variable[4] = {'H', 'o','l'}; //creamos una sequencia char con un espacio libre
append(variable[3],'a'); //lo rellenamos
Esto recuerda que se puede hacer de manera automática;
for (int i = 0;i<variable.size();i++) {
append(variable[i],'loquesea');
}
Si continuas teniendo error, utiliza strcpy, que es el que utilizaba yo;
char a[12] = "World"; /* 12 = strlen("Hello World") + 1 */
strcpy(a, "Hello World");
Saludos!! :D
Me sigue dando errores al usar append o strcpy,que no puedo pasar de char a const char, o derrepente sale de char a char*
El tema de const char* a char* o vice versa, es un tema de 'casting', se soluciona haciendo casting y listo, puesto que 'const' es tan sólo una protección del compilador a nivel código, para el programador, para que se remarquen que ciertas "cosas" son constantes. Y Siendo así, deberías o bien respetar esto o bien saber muy bien por qué habrías de cambiarlo..
Sube código
El codigo es ese que subi en el post :S
Si se trata de escribir una cadena alfanumérica y hacer que la misma se divida en bloques separados de letras o números, entonces en ese caso escribí un código simple que hace eso, el tema es que justo me dieron ganas de no hacer nada XD y lo dejé funcionando pero sin terminar ya que no llegué a implementar memoria dinámica para que quede terminado.
En lugar de eso existe un array de cadenas de un máximo aceptable de 20 cadenas, pero para ver como se hace te va a servir.
Tu código dijiste que estaba mal por eso te muestro este otro código.
Como te dije, está sin terminar pero por eso mismo si te interesa lo podés terminar, implementando memoria dinámica para el array de cadenas que vas a usar para guardar todos los bloques.
#include<stdlib.h>
#include<stdio.h>
#include<string.h>
//http://www.cplusplus.com/reference/cstdlib/realloc/
int main(){
char cadena[512];
//char** array_de_cadenas_r = 0;
//char** array_de_cadenas = (char**)malloc(sizeof(char)*1);
//array_de_cadenas[0] = (char*)malloc(sizeof(char)*128);
char array_de_cadenas[20][128] = {{0}};
printf("ingrese la cadena..\n");
scanf("%s", cadena);
int modo=0;
int pos=-1;
int let;
int num;
int i=0;
int j;
while(cadena[i]){
if(pos == 20) break;
if(cadena[i] >= 'a' && cadena[i]<='z'){
let++;
num=0;
}
else if(cadena[i] >= 'A' && cadena[i]<='Z'){
let++;
num=0;
}
else if(cadena[i] >= '0' && cadena[i]<='9'){
num++;
let=0;
}
else
{
let=0;
num=0;
}
if((modo==1&&let)||(modo==2&&num));
else
{
pos++;
j=0;
//int a=0;
//char backup[512];
//strcpy(backup, array_de_cadenas[a]);
//array_de_cadenas_r = (char**)realloc(array_de_cadenas, sizeof(char)*(pos+1));
//array_de_cadenas[pos] = (char*)malloc(sizeof(char)*128);
}
if(let){
array_de_cadenas[pos][j] = cadena[i];
j++;
modo=1;
}
else if(num){
array_de_cadenas[pos][j] = cadena[i];
j++;
modo=2;
}
else
{
modo=0;
}
i++;
}
for(int b=0; b<pos+1; b++){
printf(array_de_cadenas[b]);
putchar('\n');
}
//for(int c=0; c<pos; c++) free(array_de_cadenas[c]);
//free(array_de_cadenas);
system("pause");
return 0;
}
gracias amigo lo adaptare a lo otro que necesito :P muchas gracias....
pero podrías explicarme algo??
Citarif((modo==1&&let)||(modo==2&&num));
else
{
pos++;
j=0;
}
para que es esa parte del codigo?
Ak está el code, avisenme cualquier koza si algún detaye se habrá escapado XD
Eso que remarcastes significa lo siguiente, son todos los casos en los cuales se continúa copiando char por char hasta completar un bloque.
un bloque se forma o sólo por letras o sólo por números.
Entonces esa comprobación comprueba el estado de un par de flags o agrupación de flags, que informan el paso de un bloque a otro, o mejor dicho el final de un bloque.
Si se encuentra ante el final de un bloque, entonces se incrementa la posición en el array dinámico, y se resetea el contador usado para copiar cadenas.
#include<stdlib.h>
#include<stdio.h>
#include<string.h>
//http://www.cplusplus.com/reference/cstdlib/realloc/
int main(){
char cadena[512];
char** array_de_cadenas = 0;
char** array_de_cadenas_r = (char**)malloc(sizeof(char)*1);
array_de_cadenas_r[0] = (char*)malloc(sizeof(char)*128);
memset(array_de_cadenas_r[0],0,sizeof(char)*128);
// char array_de_cadenas[20][128] = {{0}};
printf("ingrese la cadena..\n");
scanf("%s", cadena);
int modo=0;
int pos=-1;
int let=0;// contador usado como flag
int num=0;// contador usado como flag
int i=0;
int j=0;
while(cadena[i]){
//if(pos == 20) break;
if(j==(128-2)) {
printf("uno de los bloques sobrepasa char=128-2\n");
system("pause");
return 0;
}
if(cadena[i] >= 'a' && cadena[i]<='z'){
let++;
num=0;
}
else if(cadena[i] >= 'A' && cadena[i]<='Z'){
let++;
num=0;
}
else if(cadena[i] >= '0' && cadena[i]<='9'){
num++;
let=0;
}
else
{
let=0;
num=0;
}
if((modo==1&&let)||(modo==2&&num));
else
{
pos++;
j=0;
array_de_cadenas_r = (char**)realloc(array_de_cadenas_r, sizeof(char)*(pos+1));
array_de_cadenas_r[pos] = (char*)malloc(sizeof(char)*128);
memset(array_de_cadenas_r[pos],0,sizeof(char)*128);
}
if(let){
array_de_cadenas_r[pos][j] = cadena[i];
j++;
modo=1;
}
else if(num){
array_de_cadenas_r[pos][j] = cadena[i];
j++;
modo=2;
}
else
{
modo=0;
}
i++;
}
array_de_cadenas = array_de_cadenas_r;
for(int b=0; b<pos+1; b++){
printf(array_de_cadenas[b]);
putchar('\n');
}
for(int c=0; c<pos; c++) free(array_de_cadenas[c]);
free(array_de_cadenas);
system("pause");
return 0;
}
Cita de: 85 en 4 Abril 2013, 18:39 PM
Si se trata de escribir una cadena alfanumérica y hacer que la misma se divida en bloques separados de letras o números
En C otra forma de realizar esa operación es mediante las funciones isdigit e isalpha mas punteros a funciones, estos para alternar entre los sets a procesar (digitos ==> alfabeticos ==> digitos ...).
Si ignoramos por un momento la validación de errores de malloc y realloc, un ejemplo de ello es:
#include <stdio.h>
#include <stdlib.h>
char **separar(char *p);
int main(void)
{
char **mat;
int i;
mat = separar("01ab210c");
for (i = 0; mat[i] != NULL; i++){
printf("%2d: %3s\n", i, mat[i]);
free(mat[i]);
}
free(mat);
return EXIT_SUCCESS;
}
#include <string.h>
#include <ctype.h>
char **separar(char *p)
{
int (*pfn[2])(int) = {isdigit, isalpha};
int set;
char **mat;
int i;
char *q;
mat = malloc((strlen(p) + 1) * sizeof *mat);
i = 0;
set = isalpha(*p) != 0;
while (pfn[set](*p)){
for (q = p + 1; pfn[set](*q); q++)
;
mat[i] = malloc(q - p + 1);
sprintf(mat[i], "%.*s", (int) (q - p), p);
i++;
p = q;
set = !set;
}
mat[i] = NULL;
mat = realloc(mat, ++i * sizeof *mat);
return mat;
}
La salida es la esperada:
0: 01
1: ab
2: 210
3: c
También pueden utilizarse otras funciones, por ejemplo sscanf.
Un saludo
aweee k nivel ;D
Creo que con un contador, bandera o flag y un array para guardar los parciales "va que chuta":
01ab210c213qwe31zxc213
01
ab
210
c
213
qwe
31
zxc
213
Presione una tecla para continuar . . .
#include<stdio.h>
int main(){
int i,j=0,k=0,cont=0;
char cadena[80]="01ab210c213qwe31zxc213";;
char array_de_cadena[40][6] = {{0}};
puts("01ab210c213qwe31zxc213\n");
//scanf("%s", cadena);
for (i=0;cadena[i]!='\0';i++){
if (cadena[i]>47 && cadena[i]<58 ){
if (cont==1){
array_de_cadena[j][k]='\0';j++;k=0;
}
array_de_cadena[j][k]=cadena[i];k++;cont=2;
}
else {
if (cont==2){
array_de_cadena[j][k]='\0';j++;k=0;
}
array_de_cadena[j][k]=cadena[i];k++;cont=1;
}
}
for (i=0;i<=j;i++)
printf ("%s \n",array_de_cadena[i]);
return 0;
}
Saluditos!. ...(http://st.forocoches.com/foro/images/smilies/aaaaa.gif)
eiiiii no vale poner varias cosas en una línea para que parezca más corto XD
Cita de: 85 en 5 Abril 2013, 21:19 PM
eiiiii no vale poner varias cosas en una línea para que parezca más corto XD
;-) ;-) ;-)
Lo "rectifico" y aún así, además de corto es "simple" siguiendo el principio de la navaja de Ockham: «en igualdad de condiciones, la explicación más sencilla suele ser la correcta»
#include<stdio.h>
int main(){
int i,j=0,k=0,cont=0;
char cadena[80]="01ab210c213qwe31zxc213";;
char array_de_cadena[40][6] = {{0}};
puts("01ab210c213qwe31zxc213\n");
for (i=0;cadena[i]!='\0';i++){
if (cadena[i]>47 && cadena[i]<58 ){
if (cont==1){
array_de_cadena[j][k]='\0';
j++;
k=0;
}
array_de_cadena[j][k]=cadena[i];
k++;
cont=2;
}
else {
if (cont==2){
array_de_cadena[j][k]='\0';
j++;
k=0;
}
array_de_cadena[j][k]=cadena[i];
k++;
cont=1;
}
}
for (i=0;i<=j;i++)
printf ("%s \n",array_de_cadena[i]);
return 0;
}
Un abrazo y Saluditos!. ....(http://smilies-gifs.com/emoticonos-grandes/21grandes.gif)
P.D: Sería más corto con el uso de sscanf, pero no sería más simple ni en igualdad de condiciones respecto a los propuestos.
Cita de: leosansan en 5 Abril 2013, 23:20 PM
Sería más corto con el uso de sscanf, pero no sería más simple ni en igualdad de condiciones respecto a los propuestos.
#include <stdio.h>
#include <stdlib.h>
int main(void)
{
char *cad = "01ab210c213qwe31zxc213";
char tok[40][6] = {{0}};
int i;
int j;
int nc;
puts("01ab210c213qwe31zxc213");
i = 0;
while (
sscanf(cad, "%[0123456789]%n", tok[i], &nc) == 1 ||
sscanf(cad, "%[^0123456789]%n", tok[i], &nc) == 1
){
cad += nc;
i++;
}
for (j = 0; j < i; j++)
printf ("tok[%d] == %s\n", j, tok[j]);
return 0;
}
A mi me parece aceptable (no hay nada realmente complicado).
----
Edito:
Por otra parte si la aproximación debe ser "simplicidad a rajatabla" (no me gusta ya que la biblioteca estándar esta para eso: para explotarla) podemos basarnos en una operación XOR:
#include <stdio.h>
#define IS_DIGIT(ch) ((ch) >= '0' && (ch) <= '9')
int main(void)
{
char cad[] = "01ab210c213qwe31zxc213";
char tok[40][6] = {{0}};
int i;
int j;
int k;
puts("01ab210c213qwe31zxc213");
j = 0;
k = 0;
for (i = 0; cad[i] != '\0'; i++){
tok[j][k] = cad[i];
if (IS_DIGIT(cad[i]) ^ IS_DIGIT(cad[i + 1])){
j++;
k = 0;
}else
k++;
}
for (i = 0; i < j; i++)
printf("tok[%d] == %s\n", i, tok[i]);
return 0;
}
Por supuesto con las limitaciones que tiene el uso del array para almacenar los tokens: resultados incorrectos si se sobrepasa la capacidad del elemento (token) o del array en si.
----
Edito 2:
Cambie la versión utilizando sscanf por una ultima mas corta (y por supuesto mas fea). Y no tienen que preguntarme: ya no es tan aceptable (esto ultimo hay que tomarlo con un poco de humor).
Un saludo
"Sin dudarlo un instante" me quedo con el segundo. Es una simplificación del mío al mirar un caracter y el siguiente y decidir en función de ello lo que hacer.
#include<stdio.h>
int main(){
int i,j=0,k=0,cont=0;
char cadena[80]="01ab210c213qwe31zxc213";
char array_de_cadena[40][6] = {{0}};
puts("01ab210c213qwe31zxc213\n");
for (i=0;cadena[i]!='\0';i++){
array_de_cadena[j][k]=cadena[i];
if ((cadena[i]>47 && cadena[i]<58) ^ (cadena[i+1]>47 && cadena[i+1]<58)){
j++;
k=0;
}
else
k++;
}
for (i=0;i<=j;i++)
printf ("%s \n",array_de_cadena[i]);
return 0;
}
Muy agudo rir.
Sólo me queda la duda de no introducir el caracter nulo al final de cada cadena.
Saluditos!. ...(http://st.forocoches.com/foro/images/smilies/ciao.gif)
P.D: Lo que no entiendo es a cuento de que vino el código que habías posteado anteriormente :silbar:
rir usando macros para hacer el código más legible? ;-)
Cita de: leosansan en 6 Abril 2013, 14:43 PMSólo me queda la duda de no introducir el caracter nulo al final de cada cadena.
No hay problema. Cuando se declara un array y se inicializa mediante una lista de valores:
char tok[40][6] = {{0}};
Los valores faltantes se toman como cero.
Cita de: leosansan en 6 Abril 2013, 14:43 PMP.D: Lo que no entiendo es a cuento de que vino el código que habías posteado anteriormente :silbar:
¿El primero? Me parecio un buen ejemplo del uso de un interruptor y punteros a funciones.
Cita de: 85 en 6 Abril 2013, 16:12 PM
rir usando macros para hacer el código más legible? ;-)
No la veo mal. La alternativa era isdigit pero esta no garantiza el resultado uno o cero, se debe forzar y con ello el condicional (con macro):
if (IS_DIGIT(cad[i]) ^ IS_DIGIT(cad[i + 1])){
Termina así (con isdigit):
if (!!isdigit(cad[i]) ^ !!isdigit(cad[i + 1])){
En mi opinión demasiado forzado, a falta de opciones no queda mas (remedio) que utilizar la macro.
Un saludo
Si eso estaba puesto para inicializar en cero..
Y con respecto a mi segundo código y el primero de rir, usaban memoria dinámica porque no se sabe cuántos bloques (de letras o números) van a resultar. Lo que pasa que asumimos un número así por decir 40 para hacerla más fácil.