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 - Fedex15

#1
Programación C/C++ / Re: cadena de caracteres
27 Abril 2017, 18:41 PM
Cita de: MAFUS en 27 Abril 2017, 12:01 PM
Este último código tiene pérdida de memoria al perder la referencia a cadena. Hay que pensar que es memoria dinámica y hay que liberarla a mano.

Aca lo mejore un poco, ahora le paso un char** para poder modificar la original, primero liberandola y poniendo la cadena nueva. Tambien saque las cadenas temporales y lo hice directamente, las habia usado para ir dandome cuenta que tenia que hacer.



#include <stdio.h>
#include <stdlib.h>
#include <string.h>
// Verifica si la cadena find se encuentra dentro cadena. En caso de ser
// verdad devuelve 1, en caso contrario devuelve 0;

int string_contiene (char* cadena, char* find) {
size_t longCadena = strlen (cadena);
size_t longFind = strlen (find);
if (longCadena == 0) {
return 0;
}
if ((longFind > 0) && (longFind <= longCadena)) {
int count = 0;
for (int i=0; ((i<longCadena) && (count<longFind)); i++) {
count = 0;
if (cadena[i] == find[0]) {
for (; ((count<longFind) && (cadena[i] == find[count])); i++, count++);
}
}
if (count == longFind) {
return 1;
}
}
return 0;
}

// Devuelve la posicion donde comienza la cadena find en la cadena.
// Si la cadena find no se encuentra en cadena, devuelve -1.

int string_posicion (char* cadena, char* find) {
if (string_contiene (cadena, find)) {
size_t longCadena = strlen (cadena);
size_t longFind = strlen (find);
int aux, count = 0;
for (int i=0; ((i<longCadena) && (count<longFind)); i++) {
count = 0;
aux = i;
if (cadena[i] == find[0]) {
for (; ((count<longFind) && (cadena[i] == find[count])); i++, count++);
}
}
if (count == longFind) {
return aux;
}
}
return -1;
}

// Recibe una cadena, una cadena a buscar y una cadena a reemplazar.
// Si la cadena find esta en cadena, reemplaza la primera ocurrencia de find en
// cadena por la cadena str y devuelve la nueva cadena, sino, devuelve la misma
// cadena sin modificar.

void string_reemplazar (char** cadena, char* find, char* str) {
int posicion = string_posicion (cadena[0], find);
if (posicion != -1) {
size_t longCadena = strlen (cadena[0]);
size_t longFind = strlen (find);
size_t longStr = strlen (str);
if (longFind <= longStr) {
char* cadenaModificada = (char*) malloc (sizeof (char) * longCadena + (longStr - longFind) );
}
char* cadenaModificada = (char*) malloc (sizeof (char) * longCadena - (longFind - longStr) );
for (int i=0; i<posicion; i++){
cadenaModificada[i] = cadena[0][i];
cadenaModificada[i+1] = '\0';
}
int j = posicion;
for (int i=0; i<longStr; j++, i++){
cadenaModificada[j] = str[i];
}
for (int i=posicion+longFind; i<longCadena; j++, i++){
cadenaModificada[j] = cadena[0][i];
}
                cadenaModificada[j] = '\0';
free (cadena[0]);
cadena[0] = cadenaModificada;
}
}

int main(){
char* cadena = (char*) malloc (sizeof (char) * 100);
char* find = (char*) malloc (sizeof (char) * 50);
char* str = (char*) malloc (sizeof (char) * 50);
printf ("Ingrese una cadena:\n");
gets (cadena);
printf ("Ingrese una palabra a buscar:\n");
gets (find);
printf ("Ingrese el reemplazo:\n");
gets (str);
string_reemplazar (&cadena, find, str);
printf ("La cadena es:\n%s", cadena);
       free(cadena);
       free(find);
       free(str);
return 0;
}


#2
Programación C/C++ / Re: cadena de caracteres
27 Abril 2017, 02:11 AM
Cita de: brayan en 24 Abril 2017, 04:30 AM
mira lo que me piden es esto
por ejemplo el usuario ingresa"hola mundo"
ingresa una palabra a buscar"hola" y una a reemplazar "nada"
al final el programa debe mostrar "nada mundo"
pero que pasa con lo que yo eh realizado solo sirve si la palabra a reemplazar es de menor o igual tamaño a la palabra a buscar ejemplo"hola" para buscar y "nadas" para reemplazar me muestra "nada mundo"
este es el codigo
Código (cpp) [Seleccionar]
#include <iostream>
#include <string.h>
using namespace std;

int reempla(char *frase,char *palabra,char *elim,char *frase1){
int cont1=0;
int ind = 0;
int letra = 0;
int z=0;
int j=0;


for(int x=0; x<strlen(frase); x++){

if(frase[x] == '\0'){ // si esto se cumple llego al final de la oracion
break;
}

if(frase[x] == palabra[0]){
[color=red][/color]
ind = 0;

letra = 0;
for( j = x; j<x+strlen(palabra); j++){

if(frase[j] == palabra[ind]){

frase1[j]=32;
frase1[j]=elim[z];


z++;


letra++;
ind++;

}

}

if (letra == strlen(palabra)) {
cont1++;
}

}

}


return cont1;
}

int main() {

int gran=0;
char frase[500];
char frase1[500];
char palabra[50];
char elim[50];


cout<<"Ingrese la cadena de caracter: "<<endl;
fflush(stdin);
gets(frase);


for(int z=0;frase[z]!='\0';z++){
frase1[z]=frase[z];
}


cout<<"Ingrese palabra a buscar: "<<endl;
fflush(stdin);
gets(palabra);

cout<<"Ingrese palabra a reemplazar: "<<endl;
fflush(stdin);gets(elim);

reempla(frase,palabra,elim,frase1);

cout<<frase<<endl<<palabra<<endl<<elim<<endl<<frase1<<endl;

return 0;
}



· Los códigos deben ir en etiquetas GeSHi

>aquí las reglas del foro
-Engel Lex


Hola que tal, mira, no conozco mucho de c++, yo concozco mas de c, pero te hice aca un programa que funciona como vos queres. Si tenes alguna duda me decis, no creo que sea muy dificil pasarlo a c++. Saludos


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

// Verifica si la cadena find se encuentra dentro cadena. En caso de ser
// verdad devuelve 1, en caso contrario devuelve 0;

int string_contiene (char* cadena, char* find) {
size_t longCadena = strlen (cadena);
size_t longFind = strlen (find);
if (longCadena == 0) {
return 0;
}
if ((longFind > 0) && (longFind <= longCadena)) {
int count = 0;
for (int i=0; ((i<longCadena) && (count<longFind)); i++) {
count = 0;
if (cadena[i] == find[0]) {
for (; ((count<longFind) && (cadena[i] == find[count])); i++, count++);
}
}
if (count == longFind) {
return 1;
}
}
return 0;
}

// Devuelve la posicion donde comienza la cadena find en la cadena.
// Si la cadena find no se encuentra en cadena, devuelve -1.

int string_posicion (char* cadena, char* find) {
if (string_contiene (cadena, find)) {
size_t longCadena = strlen (cadena);
size_t longFind = strlen (find);
int aux, count = 0;
for (int i=0; ((i<longCadena) && (count<longFind)); i++) {
count = 0;
aux = i;
if (cadena[i] == find[0]) {
for (; ((count<longFind) && (cadena[i] == find[count])); i++, count++);
}
}
if (count == longFind) {
return aux;
}
}
return -1;
}

// Recibe una cadena, una cadena a buscar y una cadena a reemplazar.
// Si la cadena find esta en cadena, reemplaza la primera ocurrencia de find en
// cadena por la cadena str y devuelve la nueva cadena, sino, devuelve la misma
// cadena sin modificar.

char* string_reemplazar (char* cadena, char* find, char* str) {
int posicion = string_posicion (cadena, find);
if (posicion != -1) {
size_t longCadena = strlen (cadena);
size_t longFind = strlen (find);
size_t longStr = strlen (str);
if (longFind <= longStr) {
char* cadenaModificada = (char*) malloc (sizeof (char) * longCadena + (longStr - longFind) );
}
char* cadenaModificada = (char*) malloc (sizeof (char) * longCadena - (longFind - longStr) );
char* temp1 = (char*) malloc (sizeof (char) * longCadena);
char* temp2 = (char*) malloc (sizeof (char) * longCadena);
temp1[0] = '\0';
temp2[0] = '\0';
for (int i=0; i<posicion; i++){
temp1[i] = cadena[i];
temp1[i+1] = '\0';
}
for (int i=0, j=posicion+longFind; j<longCadena; i++, j++) {
temp2[i] = cadena[j];
temp2[i+1] = '\0';
}
size_t longT1 = strlen (temp1), longT2 = strlen (temp2);
strcpy (cadenaModificada, temp1);
int i=longT1;
for (int j=0; j<longStr; j++, i++){
cadenaModificada[i] = str[j];
}
for (int j=0 ;j<=longT2; j++, i++){
cadenaModificada[i] = temp2[j];
}
free (temp1);
free (temp2);
return cadenaModificada;
}
return cadena;
}

int main(){
char* cadena = (char*) malloc (sizeof (char) * 100);
char* find = (char*) malloc (sizeof (char) * 50);
char* str = (char*) malloc (sizeof (char) * 50);
printf ("Ingrese una cadena:\n");
gets (cadena);
printf ("Ingrese una palabra a buscar:\n");
gets (find);
printf ("Ingrese el reemplazo:\n");
gets (str);
cadena = string_reemplazar (cadena, find, str);
printf ("La cadena es:\n%s", cadena);
return 0;
}
#3
Cita de: CalgaryCorpus en 24 Abril 2017, 01:41 AM
Cuando estas insertando, en estos momentos, con el codigo que presentas, indicas explicitamente el tipo de dato que esta siendo guardado., por ejemplo:

dato_insertar(general, ENTERO, &b);
dato_insertar(general2, DOUBLE, &num);
dato_insertar(general3, CADENA, cadena);

luego te obligas a hacer un switch con ese tipo.

Tengo una solucion que elimina el switch posterior, aunque aun tienes que indicar el tipo de dato que usas al insertar.

Sugiero crear un nuevo campo que guarde un puntero a una funcion que sepa imprimir el dato que estas agregando.

typedef struct _variante{
int tipo;
void* valor;
        void (*f)(void *);
       
} variante;

definir las funciones que saben imprimir, por ejemplo:

void tipo_entero(void * pTipo) {
  printf("%d", *(int *)pTipo );
}


Las otras funciones son similares, pero usan otros %d en el printf y otro cast.

luego en dato_insertar simplemente se copia el puntero a funcion

void dato_insertar (dato datos, int tipo, void* d, void (*f)() ){
datos->tipo = tipo;
datos->valor = d;
        datos->f = f;
}

y esta funcion se invoca usando el nombre de la funcion a usar para imprimr. Nota que es solo el nombre de la funcion, No como siempre usas las funciones, con parentesis.

dato_insertar(general, &b, tipo_entero );
   
y finalmente cuando quieras imprimir el dato, solo haces

dato.f(dato.valor);


y esto imprimira de acuerdo a la funcion que usaste.

Suerte!


Gracias por la respuesta, voy a ver que me sale. Ahora estoy haciendo un menu que me pidieron usando listas generales para crear listas de enteros.
#4
Fijate este codigo si te sirve.


int cuenta_ocurrencias (char* str1, char* str2) {
if ((str1[0] == '\0') || (str2[0] == '\0')) return 0;
int str1Long = strlen(str1), str2Long = strlen(str2), iter = 0;
int i, j;
for (i = 0; i < str1Long; i++){
for (j = 0; ((j < str2Long) && (str2[j] == str1[j+i])); j++);
if (str2[j] == '\0'){ // Salio del for y si llego al final
iter++;             // Es porque la cadena 2 estaba en la cadena 1
}
}
return iter;
}
#5
Mira, con char palabra te va a tirar error porque es un char, no es una cadena.

Acordate que gets() es una funcion peligrosa, si le pasas una cadena de mayor tamaño que la memoria reservada sigue escribiendo en otras partes de la memoria, lo que podria probocar algun segmentation fault.

Acordate que gets es es :

Char* gets (char* _s). Recibe un puntero a char y devuelve un puntero a char.

Cual es el error que te esta tirando, es de compilacion o cuando ejecutas el programa?.
#6
Cita de: MAFUS en 21 Abril 2017, 11:08 AM
He intentado compilar el código y tiene muchos errores. Podría resolverlos pero, la verdad, me ha dado pereza. Si pudieras corregir el código para que no haya mas errores que el lógico que intentas resolver sería de agradecer.

Gracias por responder, aca te dejo todo el codigo en un solo archivo:


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

typedef struct _GNodo {
void *dato;
struct _GNodo *sig;
} GNodo;

typedef GNodo* GList;

typedef void (*FuncionVisitante) (void* dato);

GList glist_crear() {
return NULL;
}

void glist_destruir(GList lista) {
GList nodoAEliminar;
while (lista != NULL) {
nodoAEliminar = lista;
lista = lista->sig;
free (nodoAEliminar);
}
}

GList glist_agregar_inicio(GList lista, void* dato) {
GList nuevoNodo = malloc (sizeof (GNodo));
nuevoNodo->dato = dato;
nuevoNodo->sig = lista;
return nuevoNodo;
}

GList glist_agregar_final (GList lista, void* dato) {
GList nuevoNodo = malloc (sizeof (GNodo));
nuevoNodo->dato = dato;
nuevoNodo->sig = NULL;
if (lista == NULL) {
return nuevoNodo;
}
GList nodo = lista;
for (; nodo->sig != NULL; nodo = nodo->sig);
nodo->sig = nuevoNodo;
return lista;
}

void glist_recorrer (GList lista, FuncionVisitante visit) {
for (GList nodo = lista; nodo != NULL; nodo=nodo->sig){
visit (nodo->dato);
}
}

static void imprimir_entero(void* dato) {
 printf("%d ", (*(int*)dato));
}
// Lo que yo quiero hacer es algo asi, pero no se como hacerlo:
// Si es entero:
// printf ("%d ", (*(int*)dato));
// Si es double:
//  printf ("%lf ", (*(double*)dato));
// Si es cadena:
//  printf ("%s ", (*(char*)dato));

// Como tendria que hacer para saber de alguna forma cual es el tipo
// que esta apuntando el puntero.

int main(){
int a = 3;
char c = 'a';
double d = 2.3;
GList lista = glist_crear();
lista = glist_agregar_inicio (lista, &a);
lista = glist_agregar_inicio (lista, &c);
lista = glist_agregar_inicio (lista, &d);
lista = glist_agregar_final (lista, &a);
printf ("%lf %c %d\n", (*(double*)lista->dato),(*(char*)lista->sig->dato),(*(int*)lista->sig->sig->sig->dato) );
glist_recorrer (lista, imprimir_entero);
glist_destruir (lista);
return 0;
}


yo lo compilo usando gcc listas.c -o lista, asi me funciona bien.

PD: Aca te dejo unas modificaciones que estuve haciendo, haber si voy por buen camino.


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

#define ENTERO 0
#define DOUBLE 1
#define CADENA 2

typedef struct _variante{
int tipo;
void* valor;
} variante;

typedef variante* dato;

typedef struct _GNodo {
void *dato;
struct _GNodo *sig;
} GNodo;

typedef GNodo* GList;

typedef void (*FuncionVisitante) (dato d);

dato dato_crear (){
dato nuevoDato = (variante*) malloc (sizeof(variante));
return nuevoDato;
}

void dato_insertar (dato datos, int tipo, void* d){
datos->tipo = tipo;
datos->valor = d;
}


void glist_destruir(GList lista) {
GList nodoAEliminar;
while (lista != NULL) {
nodoAEliminar = lista;
lista = lista->sig;
free (nodoAEliminar);
}
}

GList glist_crear() {
return NULL;
}

GList glist_agregar_inicio(GList lista, void* dato) {
GList nuevoNodo = malloc (sizeof (GNodo));
nuevoNodo->dato = dato;
nuevoNodo->sig = lista;
return nuevoNodo;
}

GList glist_agregar_final (GList lista, void* dato) {
GList nuevoNodo = malloc (sizeof (GNodo));
nuevoNodo->dato = dato;
nuevoNodo->sig = NULL;
if (lista == NULL) {
return nuevoNodo;
}
GList nodo = lista;
for (; nodo->sig != NULL; nodo = nodo->sig);
nodo->sig = nuevoNodo;
return lista;
}

void glist_recorrer (GList lista, FuncionVisitante visit) {
for (GList nodo = lista; nodo != NULL; nodo=nodo->sig){
visit (nodo->dato);
}
}

static void imprimir_entero(dato d) { 
  switch (d->tipo){
case (0):
printf ("%d ", *(int*)d->valor);
break;
case (1):
printf ("%lf ", *(double*)d->valor);
break;
case (2):
printf ("%s ", (char*)d->valor);
break;
}
}
// Lo que yo quiero hacer es algo asi, pero no se como hacerlo:
// Si es entero:
// printf ("%d ", (*(int*)dato));
// Si es double:
//  printf ("%lf ", (*(double*)dato));
// Si es cadena:
//  printf ("%s ", (*(char*)dato));

// Como tendria que hacer para saber de alguna forma cual es el tipo
// que esta apuntando el puntero.

int main(){
int b = 3;
double num = 6.7;
char cadena[50] = "mundo ";
dato general = dato_crear();
dato general2 = dato_crear();
dato general3 = dato_crear();
dato_insertar(general, ENTERO, &b);
dato_insertar(general2, DOUBLE, &num);
dato_insertar(general3, CADENA, cadena);
GList lista = glist_crear();
lista = glist_agregar_inicio (lista, general);
lista = glist_agregar_inicio (lista, general2);
lista = glist_agregar_inicio (lista, general3);
//~ lista = glist_agregar_final (lista, &a);
glist_recorrer (lista, imprimir_entero);
glist_destruir (lista);
return 0;
}
#7
Programación C/C++ / Listas generales en C
21 Abril 2017, 00:25 AM
Hola, estoy tratando de resolver un problema que se me pide con listas. Debo usar listas siguiendo la sig estructura:


typedef struct _GNodo {
void *dato;
struct _GNodo *sig;
} GNodo;


Tengo 3 archivos, el glist.h, glist.c, y main_g.c. Las funciones que tengo son las basicas, que me permiten crear, destruir, agregar final, agregar inicio, y recorrer.


#include "glist.h"
#include <stdlib.h>

GList glist_crear() {
return NULL;
}

void glist_destruir(GList lista) {
GList nodoAEliminar;
while (lista != NULL) {
nodoAEliminar = lista;
lista = lista->sig;
free (nodoAEliminar);
}
}

GList glist_agregar_inicio(GList lista, void* dato) {
GList nuevoNodo = malloc (sizeof (GNodo));
nuevoNodo->dato = dato;
nuevoNodo->sig = lista;
return nuevoNodo;
}

GList glist_agregar_final (GList lista, void* dato) {
GList nuevoNodo = malloc (sizeof (GNodo));
nuevoNodo->dato = dato;
nuevoNodo->sig = NULL;
if (lista == NULL) {
return nuevoNodo;
}
GList nodo = lista;
for (; nodo->sig != NULL; nodo = nodo->sig);
nodo->sig = nuevoNodo;
return lista;
}

void glist_recorrer (GList lista, FuncionVisitante visit) {
for (GList nodo = lista; nodo != NULL; nodo=nodo->sig){
visit (nodo->dato);
}
}


El problema que estoy teniendo, es que ahora no estoy trabajando con listas donde el dato es un int, estoy trabajando con listas donde el dato es un puntero a void. Yo se que el puntero a void puede apuntar a cualquier tipo de dato, pero para poder desreferenciarlo tengo que aplicar el cast correspondiente al tipo de dato que guarda.


int main(){
int a = 3;
void *ptr;
ptr = &a;
printf ("%d", *((int*)ptr)); //convierto ptr a int* y lo desreferencio con *
return 0;
}


Este es el main que tengo:


#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "glist.h"


static void imprimir_entero(void* dato) {
  printf("%d ", (*(int*)dato));
}

int main(){
int a = 3;
char c = 'a';
double d = 2.3;
GList lista = glist_crear();
lista = glist_agregar_inicio (lista, &a);
lista = glist_agregar_inicio (lista, &c);
lista = glist_agregar_inicio (lista, &d);
lista = glist_agregar_final (lista, &a);
printf ("%lf %c %d\n", (*(double*)lista->dato),(*(char*)lista->sig->dato),(*(int*)lista->sig->sig->sig->dato) );
//~ printf ("\n %lf", (*(double*)(lista->dato)));
glist_recorrer (lista, imprimir_entero);
glist_destruir (lista);
return 0;
}


Como podria hacer para poder saber cual es el tipo de dato al que apunta void. Como podria hacer lo que esta en la linea 20

printf ("%lf %c %d\n", (*(double*)lista->dato),(*(char*)lista->sig->dato),(*(int*)lista->sig->sig->sig->dato) );

de forma general con la funcion imprimir_entero. El primer caso seria con un entero, un double y un char o cadena. Gracias