Club Delphi  
    Paypal   FTP   CCD     Buscar   Trucos   Trabajo   Foros

Retroceder   Foros Club Delphi > Otros entornos y lenguajes > C++ Builder
Registrarse FAQ Miembros Calendario Guía de estilo Buscar Temas de Hoy Marcar Foros Como Leídos

 
 
Herramientas Buscar en Tema Desplegado
  #5  
Antiguo 27-04-2015
Avatar de aguml
aguml aguml is offline
Miembro
 
Registrado: may 2013
Posts: 885
Poder: 14
aguml Va por buen camino
Lo conseguí
Os pongo la prueba que hice para que la veais y ver si se puede mejorar de alguna manera. Tengo una funcion que divide Hexas enormes y otra que hace lo mismo con enteros. El limite me lo da el divisor ya que no se si sería posible partir la operacion en partes para que el divisor fuese mas grande (mi nivel de matematicas no da para mas ).
Aquí el código:
Código PHP:
//---------------------------------------------------------------------------

#include <vcl.h>
#include <limits>
#pragma hdrstop

#include "Unit1.h"
//---------------------------------------------------------------------------
#pragma package(smart_init)
#pragma resource "*.dfm"
TForm1 *Form1;

bool dividirHex(AnsiString dividendoAnsiString divisorAnsiString *cocienteAnsiString *restoint *error);
bool dividirInt(AnsiString dividendoAnsiString divisorAnsiString *cocienteAnsiString *restoint *error);

enum Error{SinError=0OverflowDivForZero};
//---------------------------------------------------------------------------
__fastcall TForm1::TForm1(TComponentOwner)
   : 
TForm(Owner)
{
}
//---------------------------------------------------------------------------

//Funcion para comprobar que un valor no sobrepase el maximo permitido para un Int64 sin signo
//El primer parametro es el valor a comprobar y el segundo parametro es para indicar si el valor
//es hexadecimal o no
bool IsOverflow(AnsiString valorbool IsHexa)
{
   
bool overflow=false;
   
AnsiString valorLimite std::numeric_limits<unsigned long long>::max();
   if(
IsHexa == true)
      
valorLimite AnsiString().sprintf("%I64X"valorLimite);

   if(
valorLimite.Length() == valor.Length()){
      for(
int i=1<=valor.Length(); i++)
      {
         if(
valorLimite[i] > valor[i]){
            break;
         }else if (
valorLimite [i] < valor[i]){
            
overflow=true;
            break;
         }
      }
   }else if(
valorLimite.Length() > valor.Length()){
      
overflowfalse;
   }else{
      
overflow=true;
   }
   return 
overflow;
}
//---------------------------------------------------------------------------

//Funcion para dividir valores hexadecimales extragrandes
//Practicamente el limite nos lo da el valor del divisor ya que el dividendo puede ser
//tan grande como el maximo menos 1 de AnsiString
bool dividirHex(AnsiString dividendoAnsiString divisorAnsiString *cocienteAnsiString *restoint *error)
{
   
unsigned __int64 auxDividendoauxDivisorauxCocienteauxResto;
   
bool salir=false;
   
bool retval=false;
   
bool primeraPasada=true;
   
int largoDividendolargoPorcion=1;

   
auxDivisor StrToInt64(AnsiString("0x" divisor));
   if(
auxDivisor == 0){
      
retval false;
      *
error DivForZero;
   }else{
      if(!
IsOverflow (divisortrue)){
         *
cociente="";
         *
resto="";

         do {
            
largoDividendo=dividendo.Length ();

            if(
largoDividendo != 0){
               if(
primeraPasada == true){
                  
auxDividendo StrToInt64("0x" dividendo.SubString(1,largoPorcion));
                  while(
auxDivisor auxDividendo && salir != true){
                     
largoPorcion++;
                     if(
largoPorcion largoDividendo){
                        *
cociente "0";
                        *
resto dividendo;
                        *
error 0;
                        
salir true;
                        
retval true;
                     }
                     if(!
salir)
                        
auxDividendo StrToInt64("0x" dividendo.SubString(1,largoPorcion));
                  }
                  
primeraPasada=false;
               }else{
                  
largoPorcion=1;
                  
auxDividendo StrToInt64("0x" + *resto dividendo.SubString(1,largoPorcion));
               }

               if(!
salir){
                  
dividendo dividendo.SubString(largoPorcion+1,largoDividendo);
                  
auxResto=auxDividendo auxDivisor;
                  
auxCociente=auxDividendo auxDivisor;

                  if(
auxCociente 0){
                     *
cociente=*cociente AnsiString ().sprintf("%I64X"auxCociente);
                     *
resto AnsiString().sprintf("%I64X"auxResto);
                  }else{
                     *
cociente=*cociente "0";
                     *
resto AnsiString().sprintf("%I64X"auxResto);
                  }
               }
            }else{
               
salir=true;
               if(
primeraPasada == true){
                  *
cociente="0";
                  *
resto="0";
               }
               
retval=true;
               *
error=0;
            }
         }while(
salir != true);
      }else{
         *
error=Overflow;
         
retval=false;
      }
   }
   return 
retval;
}
//---------------------------------------------------------------------------

//Funcion para dividir valores enteros extragrandes
//Practicamente el limite nos lo da el valor del divisor ya que el dividendo puede ser
//tan grande como el maximo menos 1 de AnsiString
bool dividirInt(AnsiString dividendoAnsiString divisorAnsiString *cocienteAnsiString *restoint *error)
{
   
unsigned __int64 auxDividendoauxDivisorauxCocienteauxResto;
   
bool salir=false;
   
bool retval=false;
   
bool primeraPasada=true;
   
int largoDividendolargoPorcion=1;

   
auxDivisor StrToInt64(divisor);
   if(
auxDivisor == 0){
      
retval false;
      *
error=DivForZero;
   }else{
      if (!
IsOverflow (divisorfalse)){
         *
cociente="";
         *
resto="";

         do {
            
largoDividendo=dividendo.Length ();

            if(
largoDividendo != 0){
               if(
primeraPasada == true){
                  
auxDividendo StrToInt64(dividendo.SubString(1,largoPorcion));
                  while(
auxDivisor auxDividendo){
                     
largoPorcion++;
                     if(
largoPorcion largoDividendo){
                        *
cociente "0";
                        *
resto dividendo;
                        *
error 0;
                        
salir true;
                        
retval true;
                     }
                     if(!
salir)
                        
auxDividendo StrToInt64(dividendo.SubString(1,largoPorcion));
                  }
                  
primeraPasada=false;
               }else{
                  
largoPorcion=1;
                  
auxDividendo StrToInt64(*resto dividendo.SubString(1,largoPorcion));
               }
               if(!
salir){
                  
dividendo dividendo.SubString(largoPorcion+1,largoDividendo);
                  
auxResto=auxDividendo auxDivisor;
                  
auxCociente=auxDividendo auxDivisor;

                  if(
auxCociente 0){
                     *
cociente=*cociente AnsiString(auxCociente);
                     *
resto AnsiString(auxResto);
                  }else{
                     *
cociente=*cociente "0";
                     *
resto AnsiString(auxResto);
                  }
               }
            }else{
               
salir=true;
               if(
primeraPasada == true){
                  *
cociente="0";
                  *
resto="0";
               }
               
retval=true;
               *
error=0;
            }
         }while(
salir != true);
      }else{
         *
error=Overflow;
         
retval=false;
      }
   }
   return 
retval;
}
//---------------------------------------------------------------------------

void __fastcall TForm1::ButtonCalcularClick(TObject *Sender)
{
   
AnsiString Cociente,Resto;
   
int error;

   if(
RadioGroupBase->ItemIndex == 0)
      
dividirInt(EditDividendo->TextEditDivisor->Text, &Cociente, &Resto, &error);
   else
      
dividirHex(EditDividendo->TextEditDivisor->Text, &Cociente, &Resto, &error);
   
EditCociente->Text Cociente;
   
EditResto->Text Resto;

   switch(
error)
   {
      case 
0:
         
StatusBar->SimpleText "Operacion realizada satisfactoriamente";
         break;

      case 
1:
         
StatusBar->SimpleText "Hay overflow en el divisor";
         break;

      case 
2:
         
StatusBar->SimpleText "No se puede dividir por 0";
         break;
   }
}
//---------------------------------------------------------------------------

void __fastcall TForm1::EditFiltradoKeyPress(TObject *Senderchar &Key)
{
   if(
RadioGroupBase->ItemIndex == 1){
      if((
Key '0' || Key '9') && (Key 'a' || Key 'f') && (Key 'A' || Key 'F') && Key != '\b')
         
Key 0;
   }else{
      if((
Key '0' || Key '9') && Key != '\b')
         
Key 0;
   }
}
//---------------------------------------------------------------------------

void __fastcall TForm1::RadioGroupBaseClick(TObject *Sender)
{
   if(
RadioGroupBase->ItemIndex == 1)
   {  
//Lo paso a Hexa
      
if(!IsOverflow(EditDividendo->Text,false) &&
         !
IsOverflow(EditDivisor->Text,false) &&
         !
IsOverflow(EditCociente->Text,false) &&
         !
IsOverflow(EditResto->Text,false)){

            if(
EditDividendo->Text != "" &&
               
EditDivisor->Text != "" &&
               
EditCociente->Text != "" &&
               
EditResto->Text != "")
            {
               
unsigned __int64 aux;
               
aux StrToInt64(EditDividendo->Text);
               
EditDividendo->Text AnsiString().sprintf("%I64X"aux);
               
aux StrToInt64(EditDivisor->Text);
               
EditDivisor->Text AnsiString().sprintf("%I64X"aux);
               
aux StrToInt64(EditCociente->Text);
               
EditCociente->Text AnsiString().sprintf("%I64X"aux);
               
aux StrToInt64(EditResto->Text);
               
EditResto->Text AnsiString().sprintf("%I64X"aux);
               
StatusBar->SimpleText "";
            }else{
               
StatusBar->SimpleText "Hay algún valor no válido";
            }
      }else{
         
EditDividendo->Text "0";
         
EditDivisor->Text "0";
         
EditCociente->Text "0";
         
EditResto->Text "0";
         
StatusBar->SimpleText "Overflow";
      }
   }else{ 
//Lo paso a Int
      
if(!IsOverflow(EditDividendo->Text,true) &&
         !
IsOverflow(EditDivisor->Text,true) &&
         !
IsOverflow(EditCociente->Text,true) &&
         !
IsOverflow(EditResto->Text,true)){

            if(
EditDividendo->Text != "" &&
               
EditDivisor->Text != "" &&
               
EditCociente->Text != "" &&
               
EditResto->Text != "")
            {
               
EditDividendo->Text StrToInt64("0x" EditDividendo->Text);
               
EditDivisor->Text StrToInt64("0x" EditDivisor->Text);
               
EditCociente->Text StrToInt64("0x" EditCociente->Text);
               
EditResto->Text StrToInt64("0x" EditResto->Text);
               
StatusBar->SimpleText "";
            }else{
               
StatusBar->SimpleText "Hay algún valor no válido";
            }
      }else{
         
EditDividendo->Text "0";
         
EditDivisor->Text "0";
         
EditCociente->Text "0";
         
EditResto->Text "0";
         
StatusBar->SimpleText "Overflow";
      }
   }
}
//--------------------------------------------------------------------------- 
Responder Con Cita
 


Herramientas Buscar en Tema
Buscar en Tema:

Búsqueda Avanzada
Desplegado

Normas de Publicación
no Puedes crear nuevos temas
no Puedes responder a temas
no Puedes adjuntar archivos
no Puedes editar tus mensajes

El código vB está habilitado
Las caritas están habilitado
Código [IMG] está habilitado
Código HTML está deshabilitado
Saltar a Foro

Temas Similares
Tema Autor Foro Respuestas Último mensaje
Instalar componentes de 32 bits en una PC de win7 a 64 bits uper Windows 10 29-01-2016 18:08:30
compilar programas 32 bits con lazarus 64 bits anubis Lazarus, FreePascal, Kylix, etc. 3 30-10-2013 18:08:44
Como ejecutar aplicativos dbExpress de 32 bits en 64 bits rolandoj Conexión con bases de datos 0 30-07-2010 19:39:07
Aplicaciones de 32 bits en Windows de 64 bits Gabo Debates 9 25-09-2008 20:49:32
Porque GetLocaleFormatSettings siempre retorna dd.MM.yyyy QuarkBcn API de Windows 2 11-10-2007 13:43:05


La franja horaria es GMT +2. Ahora son las 19:02:37.


Powered by vBulletin® Version 3.6.8
Copyright ©2000 - 2026, Jelsoft Enterprises Ltd.
Traducción al castellano por el equipo de moderadores del Club Delphi
Copyright 1996-2007 Club Delphi