#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <math.h>
#define LENGTH(x) (sizeof(x) / sizeof(x[0]))
int hasPar ( char * ) ; //Überprüft, ob Klammern im String sind und gibt Position der ersten Klammer zurück
char * subStr ( char * , char * , int , int ) ; //Funktion, die Substring zurückgibt
double calcStr( char * ) ; //Wertet Terme ohne Klammern aus
double calc ( char * ) ; //Rekursive Berechnung des Terms
int main( /*int argc, char *argv[]*/ ) {
char * s/*, c*/ ;
double ergebnis;
do {
puts ( "Taschenrechner. Ignoriert alles, au\xe1 " "er 0-9,.,+,-,*,/,^,(,)" ) ;
scanf ( "%99[^\n ]" , s
) ; //[^\n] bedeutet, dass alle Zeichen außer Zeilenumbruch eingelesen werden sollen
printf ( "Erkannter Ausdruck:\n %s\n " , s
) ;
ergebnis= calc( s) ;
printf ( "Berechnetes Ergebnis:\n %f\n " , ergebnis
) ;
//if (getc(stdin) == 'c') {break;}
} while ( 0 ) ;
return 0 ;
}
/*************************************
* Überprüfen, ob Klammern im Term sind
* gibt Position der ersten Klammer
* oder -1 zurück
**************************************/
int hasPar ( char * s) {
for ( unsigned int k
= 0 ; k
< strlen ( s
) ; k
++ ) { //k verwendet, da mit i Interferenzen mit calc() aufgetreten sind if ( s[ k] == '(' ) {
return k;
}
}
return ( - 1 ) ;
}
/*****************************************************
* Funktion, die Substring zurückgibt
* s: Ursprungsstring
* offset: Position, an der Substring beginnt
* len: Länge des Substrings
* t: Variable, in der Substring gespeichert wird
******************************************************/
char * subStr ( char * dest, char * src, int offset, int len) {
if ( offset+ len > input_len ) { //Wenn Substring größer sein sollte als Usprungsstring oder Substring Null Zeichen enthalten soll
return NULL;
} else if ( len <= 0 ) {
dest[ 0 ] = '\0 ' ;
}
strncpy ( dest
, src
+ offset
, len
) ; //len Zeichen werden aus s ab offset in t kopiert dest[ len] = '\0 ' ;
return dest;
}
/*****************************************************
* Wertet Terme ohne Klammern aus
******************************************************/
double calcStr ( char * s) {
char * t, * t_first;
int len_s
= strlen ( s
) , len_first
= 0 ; t
= calloc ( len_s
, sizeof ( char ) ) ; //Kopie von s zum Arbeiten erstellen, Schritt 1 strcpy ( t
, s
) ; //Kopie von s zum Arbeiten erstellen, Schritt 2
//ACHTUNG: REIHENFOLGE WICHTIG FÜR KORREKTE ANWENDUNG VON RECHENREGELN
if ( t[ 0 ] != '+' ) { //+ als unärer Operator
t_first
= strtok ( t
, "+" ) ; //String auf + prüfen, Nach Ausführung von strtok: t_first: String bis exklusiv + len_first
= strlen ( t_first
) ; //Länge des Ergebnisstrings berechnen zum Vergleich mit Länge des Ursprungsstrings if ( len_first != len_s) { //Wenn Länge gleich, dann ist hier auch der Inhalt gleich, also kein Plus enthalten
switch ( t_first[ len_first- 1 ] ) { //Wenn + unär, also vor dem + ein anderes OpSym
case '+' :
t[ len_first- 1 ] = '\0 ' ;
return calcStr( t) + ( calcStr( s+ len_first) ) ;
case '-' :
t[ len_first- 1 ] = '\0 ' ;
return calcStr( t) - ( calcStr( s+ len_first) ) ;
case '*' :
t[ len_first- 1 ] = '\0 ' ;
return calcStr( t) * ( calcStr( s+ len_first) ) ;
case '/' :
t[ len_first- 1 ] = '\0 ' ;
return calcStr( t) / ( calcStr( s+ len_first) ) ;
case '^' :
t[ len_first- 1 ] = '\0 ' ;
return pow ( calcStr
( t
) , ( calcStr
( s
+ len_first
) ) ) ; default :
return calcStr( t_first) + ( calcStr( s+ len_first+ 1 ) ) ; //Rekursives Aufrufen der Strings links und rechts des Operationszeichens
}
}
}
strcpy ( t
, s
) ; //da t bei Überprüfung auf + verändert wurde, Wiederherstellung der Arbeitskopie aus Ursprungsstring if ( t[ 0 ] != '-' ) { //- als unärer Operator
t_first
= strtok ( t
, "-" ) ; //analog oben if ( len_first != len_s) {
switch ( t_first[ len_first- 1 ] ) {
case '+' :
t[ len_first- 1 ] = '\0 ' ;
return calcStr( t) + ( calcStr( s+ len_first) ) ;
case '-' :
t[ len_first- 1 ] = '\0 ' ;
return calcStr( t) - ( calcStr( s+ len_first) ) ;
case '*' :
t[ len_first- 1 ] = '\0 ' ;
return calcStr( t) * ( calcStr( s+ len_first) ) ;
case '/' :
t[ len_first- 1 ] = '\0 ' ;
return calcStr( t) / ( calcStr( s+ len_first) ) ;
case '^' :
t[ len_first- 1 ] = '\0 ' ;
return pow ( calcStr
( t
) , ( calcStr
( s
+ len_first
) ) ) ; default :
return calcStr( t_first) + ( calcStr( s+ len_first+ 1 ) ) ; //Rekursives Aufrufen der Strings links und rechts des Operationszeichens
}
}
}
if ( len_first != len_s) {
return calcStr( t_first) * ( calcStr( s+ len_first+ 1 ) ) ;
}
if ( len_first != len_s) {
return calcStr( t_first) * ( 1 / calcStr( s+ len_first+ 1 ) ) ;
}
if ( len_first != len_s) {
if ( t_first[ len_first- 1 ] == 'e' ) {
if ( t_first[ 0 ] == '-' ) {
return ( - 1 ) * exp ( calcStr
( s
+ len_first
+ 1 ) ) ; } else if ( t_first[ 0 ] == '+' ) {
return exp ( calcStr
( s
+ len_first
+ 1 ) ) ; } else {
return pow ( calcStr
( t_first
) , ( calcStr
( s
+ len_first
+ 1 ) ) ) ; }
}
}
return atof ( s
) ; //String ist bei keinem Operationszeichen zerfallen => String ist Zahl ; atof castet string zu double (aus stdlib.h)
}
/****************************************************
* Funktion zur Verschachtelten Berechnung des Terms
****************************************************/
double calc ( char * s) {
double result_d= 0 .;
char * t, * result_str;
int check= 1 , i= hasPar( s) , j= 0 ;
if ( i == ( - 1 ) ) {
return calcStr( s) ;
} else {
j= ++ i; //j=++i ist Position des Chars nach der ersten öffnenden Klammer
while ( check > 0 ) {
if ( s[ i] == '(' ) {
check++;
} else if ( s[ i] == ')' ) {
check--;
}
i++;
} //Bestimmen der Länge der "obersten" Klammer, i ist Position des ersten Zeichens nach der Klammer
t
= calloc ( strlen ( s
) , sizeof ( char ) ) ; //Hilfsstring, Länge: s[0 bis j-1] + Puffer für Nachkommastellen result_str
= calloc ( strlen ( s
) , sizeof ( char ) ) ; //Hilfsstring, Länge s[j bis i-2] + Puffer für Nachkommastellen result_d= calc( subStr( result_str, s, j, i- j- 1 ) ) ; //Trennung auf zwei Zeilen zur besseren Lesbarkeit der verschachtelten Funktionsaufrufe
sprintf ( result_str
, "%-.16f" , result_d
) ; //casten des errechneten Double-Ergebnisses zurück in *char !!!ACHTUNG!!! Präzision nicht auf 16 stellen, da sonst undefiniertes Verhalten erzeugt wird return calcStr
( strcat ( subStr
( t
, s
, 0 , j
- 1 ) , strcat ( result_str
, s
+ i
) ) ) ; //rekursiver Aufruf: calc( s[bis j-2] ## s[j bis i-2] ## s[i bis Ende]) //##: Konkatenation }
}
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <math.h>
#define LENGTH(x)  (sizeof(x) / sizeof(x[0]))

int hasPar (char *); //Überprüft, ob Klammern im String sind und gibt Position der ersten Klammer zurück
char* subStr (char* , char* , int , int ); //Funktion, die Substring zurückgibt
double calcStr(char *); //Wertet Terme ohne Klammern aus
double calc (char *); //Rekursive Berechnung des Terms


int main(/*int argc, char *argv[]*/) {
	char *s/*, c*/;
	double ergebnis;

	do {
		puts("Taschenrechner. Ignoriert alles, au\xe1""er 0-9,.,+,-,*,/,^,(,)");

		fflush(stdin);
		s= calloc(100,1);
		scanf("%99[^\n]", s); //[^\n] bedeutet, dass alle Zeichen außer Zeilenumbruch eingelesen werden sollen
		
		printf("Erkannter Ausdruck:\n%s\n", s);
		
		ergebnis= calc(s);	
		printf("Berechnetes Ergebnis:\n%f\n", ergebnis);
		

		puts("Erfolg!");;
		free(s);
		fflush(stdin);
		//if (getc(stdin) == 'c') {break;}
	} while (0);
	
	return 0;
}



/*************************************
* Überprüfen, ob Klammern im Term sind
* gibt Position der ersten Klammer
* oder -1 zurück
**************************************/
int hasPar (char *s) {
	for (unsigned int k=0; k<strlen(s); k++) { //k verwendet, da mit i Interferenzen mit calc() aufgetreten sind
		if ( s[k] == '(' ) {
			return k;
		}
	}

	return (-1);
}


/*****************************************************
* Funktion, die Substring zurückgibt
* s: Ursprungsstring
* offset: Position, an der Substring beginnt
* len: Länge des Substrings
* t: Variable, in der Substring gespeichert wird
******************************************************/
char* subStr (char* dest, char* src, int offset, int len) {
	int input_len = strlen (src);

	if ( offset+len > input_len ) { //Wenn Substring größer sein sollte als Usprungsstring oder Substring Null Zeichen enthalten soll
		return NULL;
	} else if (len <= 0) {
		dest[0]= '\0';
	}

	strncpy(dest, src + offset, len); //len Zeichen werden aus s ab offset in t kopiert
	dest[len]= '\0';
	return dest;
}


/*****************************************************
* Wertet Terme ohne Klammern aus
******************************************************/
double calcStr (char *s) {
	char *t, *t_first;
	int len_s= strlen(s), len_first=0;
	t= calloc(len_s, sizeof(char)); //Kopie von s zum Arbeiten erstellen, Schritt 1
	strcpy(t,s); //Kopie von s zum Arbeiten erstellen, Schritt 2

	//ACHTUNG: REIHENFOLGE WICHTIG FÜR KORREKTE ANWENDUNG VON RECHENREGELN

	if(t[0] != '+') { //+ als unärer Operator
		t_first= strtok(t,"+"); //String auf + prüfen, Nach Ausführung von strtok: t_first: String bis exklusiv +
		len_first=strlen(t_first); //Länge des Ergebnisstrings berechnen zum Vergleich mit Länge des Ursprungsstrings
		if (len_first != len_s) { //Wenn Länge gleich, dann ist hier auch der Inhalt gleich, also kein Plus enthalten
			switch ( t_first[len_first-1] ) { //Wenn + unär, also vor dem + ein anderes OpSym
				case '+':
					t[len_first-1]= '\0';
					return calcStr(t) + (calcStr(s+len_first));
				case '-':
					t[len_first-1]= '\0';
					return calcStr(t) - (calcStr(s+len_first));
				case '*':
					t[len_first-1]= '\0';
					return calcStr(t) * (calcStr(s+len_first));
				case '/':
					t[len_first-1]= '\0';
					return calcStr(t) / (calcStr(s+len_first));
				case '^':
					t[len_first-1]= '\0';
					return pow(calcStr(t), (calcStr(s+len_first)));
				default:
					return calcStr(t_first) + (calcStr(s+len_first+1)); //Rekursives Aufrufen der Strings links und rechts des Operationszeichens
			}
		}
	}

	strcpy(t,s); //da t bei Überprüfung auf + verändert wurde, Wiederherstellung der Arbeitskopie aus Ursprungsstring
	if(t[0] != '-') { //- als unärer Operator
		t_first= strtok(t,"-"); //analog oben
		len_first=strlen(t_first); 
		if (len_first != len_s) {
			switch ( t_first[len_first-1] ) {
				case '+':
					t[len_first-1]= '\0';
					return calcStr(t) + (calcStr(s+len_first));
				case '-':
					t[len_first-1]= '\0';
					return calcStr(t) - (calcStr(s+len_first));
				case '*':
					t[len_first-1]= '\0';
					return calcStr(t) * (calcStr(s+len_first));
				case '/':
					t[len_first-1]= '\0';
					return calcStr(t) / (calcStr(s+len_first));
				case '^':
					t[len_first-1]= '\0';
					return pow(calcStr(t), (calcStr(s+len_first)));
				default:
					return calcStr(t_first) + (calcStr(s+len_first+1)); //Rekursives Aufrufen der Strings links und rechts des Operationszeichens
			}
		}
	}

	strcpy(t,s);
	t_first= strtok(t,"*");
	len_first=strlen(t_first);
	if (len_first != len_s) {
		return calcStr(t_first) * (calcStr(s+len_first+1));
	}

	strcpy(t,s); //analog
	t_first= strtok(t,"/");
	len_first=strlen(t_first);
	if (len_first != len_s) {
		return calcStr(t_first) * (1 / calcStr(s+len_first+1));
	}
	
	strcpy(t,s);
	t_first= strtok(t,"^");
	len_first=strlen(t_first);
	if (len_first != len_s) {
		if (t_first[len_first-1]=='e') {
			if (t_first[0]=='-') {
				return (-1)*exp(calcStr(s+len_first+1));
			} else if (t_first[0]=='+') {
				return exp(calcStr(s+len_first+1));
			} else {
				return pow(calcStr(t_first), (calcStr(s+len_first+1)));
			}
		}
	}

	return atof(s); //String ist bei keinem Operationszeichen zerfallen => String ist Zahl ; atof castet string zu double (aus stdlib.h)

}

/****************************************************
 * Funktion zur Verschachtelten Berechnung des Terms
 ****************************************************/
double calc (char *s) {
	double result_d=0.;
	char *t, *result_str;
	int check=1, i=hasPar(s), j=0;
	if ( i == (-1) ) {
		return calcStr(s);
	} else {
		j= ++i; //j=++i ist Position des Chars nach der ersten öffnenden Klammer
		while (check > 0) {
			if (s[i] == '(') {
				check++;
			} else if (s[i] == ')') {
				check--;
			}
			i++;
		}	//Bestimmen der Länge der "obersten" Klammer, i ist Position des ersten Zeichens nach der Klammer

		t= calloc(strlen(s), sizeof(char)); //Hilfsstring, Länge: s[0 bis j-1] + Puffer für Nachkommastellen
		result_str= calloc(strlen(s), sizeof(char)); //Hilfsstring, Länge s[j bis i-2] + Puffer für Nachkommastellen
		result_d= calc( subStr(result_str, s, j, i-j-1)); //Trennung auf zwei Zeilen zur besseren Lesbarkeit der verschachtelten Funktionsaufrufe
		sprintf(result_str, "%-.16f", result_d); //casten des errechneten Double-Ergebnisses zurück in *char !!!ACHTUNG!!! Präzision nicht auf 16 stellen, da sonst undefiniertes Verhalten erzeugt wird
		return calcStr( strcat( subStr(t, s, 0, j-1), strcat( result_str, s+i))); //rekursiver Aufruf: calc( s[bis j-2] ## s[j bis i-2] ## s[i bis Ende]) //##: Konkatenation
	}
}
