您的位置:首页 > 其它

The C programing language 第四章,课后题

2016-04-10 11:25 429 查看

chap4.c

/*The C programing language 第四章,课后题 */
#include <stdio.h>
//#include "chap2.c"
#include <ctype.h>
//#include "chap1.c"
#include <string.h>
#include  "chap4_calculator.c" //for 4-3 to 4-10
#define abs(x)  ((x > 0) ? (x) : -(x))
extern void calculator();
/*
4-1 strindex function which return the position of the righmost
occurrence of t  in s , or -1 if there in none.
*/
int strindex(char s[], char t[])
{
int i, j, k;
int position = -1;
for(i = 0; s[i] != '\0'; i++)
{
for(j=i,k=0; s[j]!='\0' && t[k]!='\0' && t[k]==s[j];k++,j++);

if(k>0 && t[k]=='\0')
position = i;
}

return position;
}
/*
看了答案之后,发现从右边进行扫描对比,效率更高
*/
int _strindex(char s[], char t[])
{
int i, j, k;

for(i = strlen(s)-strlen(t); i>=0; i--)
{
for(j=i,k=0; s[j]!='\0' && t[k]!='\0' && t[k]==s[j];k++,j++);

if(k>0 && t[k]=='\0')
return i;
}

return -1;
}
void evaluated_strindex()
{
char s[] = "afscbajcbai";
char t[] = "cba";
printf("rightmost positon : %d \n",strindex(s, t));
printf("rightmost positon : %d \n",_strindex(s, t));
}

/*
Exercise 4-2
convert string s to double
test :
char s[] = "10.12e10";
printf("%g \n", _atof(s));
*/
double _atof(char s[])
{
double val, power;
int i, sign, exp;

for(i = 0; isspace(s[i]); i++)
;

sign = (s[i] == '-') ? -1: 1;
if(s[i] == '-' || s[i] == '+')
i++;
for(val = 0.0; isdigit(s[i]); i++)
val = 10.0 * val + (s[i]-'0');

if(s[i] == '.')
i++;

for(power = 1.0; isdigit(s[i]); i++)
{
val = 10.0 * val + (s[i] - '0'); //继续加上去
power *= 10.0;  //计算正数要除的数,还原小数点后面的值
}

val = sign * val / power;  //把正负,小数 还原

if(s[i] == 'e' || s[i] == 'E')
{
sign = (s[++i] == '-') ? -1 : 1;
if(s[i] == '-' || s[i] == '+')
i++;
for(exp = 0; isdigit(s[i]); i++) //计算e的值 跟之前atoi是一样的原理
exp = 10 * exp + (s[i] - '0');
if(sign == -1) // 负数
while(exp-- > 0)
val /=10;
else //正数
while(exp-- > 0)
val *= 10;
}
return  val;

}
/* from 4-3  to 4-10 */
void calculator_()
{
calculator();
}
/*
4-12
atoi : convert n to  characters in s  by calling a recursive routine
*/
void itoa(int n, char s[]) {
static int i = 0;

if (n/10)
itoa(n/10, s);
else {
if (n < 0)
s[i++] = '-';
}
// recursive
s[i++] = abs(n%10) + '0';
s[i] = '\0';
}
/*4-13
reverse : reverses string  s in place
*/
void reverse(char s[])
{
void reverse_r(char [], int index, int len);
reverse_r(s, 0, strlen(s));
}
/*
reverse : reverses string s in place ;recrusive
*/
void reverse_r(char s[], int index, int len)
{
int temp, j;
j = len - (index + 1);/*why +1 */

if (index < j) {
temp = s[index];
s[index] = s[j];
s[j] = temp;
reverse_r(s, ++index, len);
}
}//the why is :the len is begin 1 ,but the index is begin from 0
/*
4-14
interchages two arguments of type t
*/
#define swap(t, x, y) { t temp;\
temp = x; \
x = y; \
y = temp; \
}

main()
{
calculator_();
}


chap4_calculator.c //handle calculator

/*e
calculator control of main
while(getop(s))
if(number )
push it
else if(operator)
pop operands
do operation
push result
else if(newline)
pop and print top of stack
else
error
we seperator the funcion from sames source files
includes : stack.c getop.c

*/
#include <stdio.h>
#include "stack.c" // has push() and pop() funtions
#include "getop.c"  //has getop() function
#include <stdlib.h> // has atof()
//#include "chap4.c" // has _atof()  declare and defition by myself
#include <string.h>//for strcmp()
#include <math.h> // for sin() exp() pow()
#define NUMBER '0'
#define MATHFUN 'n'
extern void push(double); /* store the digit into the top of  stack*/
extern double pop(void); /*  return with  the top of stack into */
extern void clear(void);/* clear the stack */
extern int getop(char []);/*  return with   an  operator or operand */
extern double mathfun(char []); /* access the math function through the function's name*/
//extern void ungets(char []) /* push back an entire string onto the input*/
void calculator()
{
int type, i, var = 0;
double op2, v;
char strop[100];
double variables[26];
for (i = 0; i < 26; i++)
variables[i] = 0.0;
while ((type = getop_line(strop)) !=EOF) {
switch (type) {
case NUMBER://handle didit
//  printf("debug strop : %s \n",strop);
push(atof(strop));
break;
case MATHFUN: //handle the math function
mathfun(strop);
break;
case '+':
push(pop() + pop());
break;
case '*':
push(pop() * pop());
break;
case '-':
printf("debug - :  \n");
op2 = pop();
push(pop() - op2);
break;
case '/':
op2 = pop();
if (op2 > 0.0)
push(pop() / op2);
else
printf("errot : zero  divisor \n");
break;
case '%':
op2 = pop();
if (op2 > 0.0)
push((int)pop() % (int)op2);
else
printf("errot : zero  divisor \n");
break;
case 'd': /* operator stack : duplicate the top element*/
op2 = pop();
push(op2);
printf("ths number top of stack is : %.5g\n", op2);
break;
case 's': /* operator stack : swap the top two elements */
op2 = pop();
double temp = pop();
push(op2);
push(temp);
printf("swap successful \n");
break;
case 'c': /* operator stack : clear the  stack  */
clear();
printf("clear the stack successful \n");
break;
case '=':
if (var >= 'A' && var <= 'Z') {
pop(); //pop the value of A
variables[var-'A'] = pop(); //store the value of A into array variables
}
case '\n':
v = pop();
printf("the result is : %g \n", v);
break;
default :
if (type >= 'A' && type <= 'Z')
push(variables[type - 'A']);
else if (type == 'v')
push(v);
else
printf("error input ,unknown command %s\n",strop);
}
var = type;
}
}
double mathfun(char s[])
{
double temp;
if (strcmp(s, "sin") == 0)
push(sin(pop()));
else if (strcmp(s, "exp") == 0)
push(exp(pop()));
else if (strcmp(s, "pow") == 0) {
temp = pop();
push(pow(pop(), temp));
}
else {
printf("mathfun errot : input %s , I can't understading \n ", s);
}
}


getop.c

/*
getop.c for getop() :
store the characters of input into string opstr
and return the type of opstr
*/
#pragma once
#include <string.h> //for strlen()
#include <ctype.h> //for isdigit()
#include "getch.c"// for getch() ungetch() getline()
#define MAXSIZE 100;
#include <stdio.h>
extern int getch(void);
extern void ungetch(int);
extern int  _getline(char [], int limit);
int getop(char opstr[])
{
int cr, next;
int i = 0;

while ((cr = getch()) == ' ' || cr == '\t');

/* handle the mathfun */
if (islower(cr))
{
opstr[i++] = cr;
while (islower(opstr[i++] = cr = getch()));
opstr[i-1] = '\0';
printf("handle the mathfun : %s, lentgh=%d \n", opstr, (int)strlen(opstr));
if (cr != EOF)
ungetch(cr);
if (strlen(opstr) > 1)
return 'n';
else
return cr;
}

if (isdigit(cr)==0 && cr != '.' && cr != '-')//add provision the negative numners
{
opstr[i] = '\0';
return cr;
}

if (cr == '-')
{
if (isdigit(cr = getch()) || cr == '.')//add provision the negative numners
{
opstr[i++] = '-';
opstr[i++] = cr;
cr = getch();
}
else
{
if (cr != EOF)
ungetch(cr);
opstr[i] = '\0';
return '-';
}
}
while(isdigit(cr))
{
opstr[i++] = cr;
if (cr != EOF)
cr = getch();
}; // collect integer part

if (cr == '.') {
opstr[i] = '.';
while(isdigit(opstr[++i] = cr = getch()))
;
}
opstr[i] = '\0';
// printf("debug  return ungetch  : %c \n", cr);
if (cr != EOF)
ungetch(cr);

printf("collect the digit : %s, lentgh=%d \n", opstr, (int)strlen(opstr));
return '0';
}
/*
_getline() exchange getch()
*/
int li = 0; //li of line
char line[100];
int getop_line(char opstr[])
{
int cr, next;
int i = 0;

if (line[li] == '\0') {
if (!_getline(line, 100)) {
//printf("line : %s, %c, li=%d \n", line, line[li-1], li-1);
opstr[i] = '\0';
return EOF;
}
//printf("line : %s, %c, li=%d \n", line, line[li], li);
li = 0;
}
while ((cr = line[li++]) == ' ' || cr == '\t')
;

/* handle the mathfun */
if (islower(cr))
{
opstr[i++] = cr;
while (islower(opstr[i++] = cr = line[li++]));
opstr[i-1] = '\0';
printf("handle the mathfun : %s, lentgh=%d \n", opstr, (int)strlen(opstr));
if (cr != EOF)
li--;
if (strlen(opstr) > 1)
return 'n';
else
return cr;
}

if (isdigit(cr)==0 && cr != '.' && cr != '-')//add provision the negative numners
{
opstr[i] = '\0';
return cr;
}

if (cr == '-')
{
if (isdigit(cr = line[li++]) || cr == '.')//add provision the negative numners
{
opstr[i++] = '-';
opstr[i++] = cr;
cr = line[li++];
}
else
{
if (cr != EOF)
li--;
opstr[i] = '\0';
return '-';
}
}
while(isdigit(cr))
{
opstr[i++] = cr;
if (cr != EOF)
cr = line[li++];
}; // collect integer part

if (cr == '.') {
opstr[i] = '.';
while(isdigit(opstr[++i] = cr = line[li++]))
;
}
opstr[i] = '\0';
// printf("debug  return ungetch  : %c \n", cr);
if (cr != EOF)
li--;

printf("collect the digit : %s, lentgh=%d \n", opstr, (int)strlen(opstr));
return '0';
}


getch.c

#define BUFFSIZE 100
#pragma once
#include <stdio.h>
#include <string.h>
char buf[BUFFSIZE]; // buffer for ungetch
int bufp = 0;// next free position in buf

int getch(void)
{
return (bufp > 0)? buf[--bufp]: getchar();

}

void ungetch(int c)
{
if(bufp < BUFFSIZE)
buf[bufp++] = c;
else
printf("ungetch : too many characters \n");

}
void ungets(char s[])
{
int i = 0;
for (i = strlen(s) - 1; i >= 0; i--) {
buf[bufp++] = s[i];
}
}
/*将一行数据读入line中并返回其长度
line 是一个数组,用来存放输入数据
lim  是定义的行的最大长度
*/
int _getline(char line[],int lim)
{
int c,i;
for(i=0;i<lim-1 && (c=getchar())!=EOF && c!='\n';i++)
line[i] = c;
if(c == '\n')
{
line[i] = c;
i++;
}
line[i] = '\0';
return i;
}


stack.c

#define NUMBER '0'
#define MAXVAL 100
double val[MAXVAL]; //value stack
int sp=0; //next the free position
void push(double f)
{
if (sp < MAXVAL)
val[sp++] = f;
//printf("stack : %g, \n", val[sp-1]);
}

double pop(void)
{
if (sp > 0)
return val[--sp];
else
printf("pop : the stack is empty \n");
return  0.0;
}
void clear(void)
{
sp =0;
}


getch_4-8.c

/*
4-8
*/
#include <stdio.h>
char buf = '\0';
int getch() /*return a char */
{
int c;
if (buf != '0') {
c = buf;
buf = '\0';
}
else
c = getchar();
return c;
}

void ungetch(int cr)
{
if (buf != '\0')
printf("ungetch : too many characters \n  ");
else
buf = cr;

}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: