C程序设计(第二版 新版)第七章 习题

1.int printf(char *format,arg1,arg2,...);  //返回值为输出的参数个数
  例如:printf("%*.*s",min,max,s);
min 指输出的空间最小为max
        max 指字符串s截取max个字符

2.printf("\\%04o ",c); // 本题参考http://blog.chinaunix.net/uid-26295753-id-2981750.html
  %后面的数字前的0的用途!
printf格式输出:%[flags][width][.perc][F|N|h|l]type
用到了flags中的 0 (注意是零不是欧) ,其百科描述为:将输出的前面补上0,直到占满指定列宽为止(不可以搭配使用-)
width 即表示需要输出的位数。

int a = 4;
printf("%03d",a);
输出:004

也可以用 * 代替位数,在后面的参数列表中用变量控制输出位数;
int a = 4;
int n = 3;
printf("%0*d",n,a);
输出:004

3.程序:以合理的方式打印任何输入。该程序至少能够更具客户用户习惯以八进制或十六进制形式打印非图形字符,并截断长文本行。

#include
#include
#define MAXLINE 20
#define OCTLEN  6

int main()
{
	int c, pos;
	int inc(int pos, int n);
	
	pos = 0;
	while((c = getchar()) != EOF)
	{
		if(iscntrl(c) || c == ' ')
		{
			pos = inc(pos,OCTLEN);
			printf("\\%04o ",c);
			if(c == '\n')
			{
				pos = 0;
				putchar('\n');
			}	
		}
		else
		{
			pos = inc(pos, 1);
			putchar(c);
		}
	}
	return 0;
}

int inc(int pos, int n)
{
	if(pos + n < MAXLINE)
		return pos+n;
	else
	{
		putchar('\n');
		return n;
	}
}

4.实现minprintf函数,使它能完成printf函数的更多功能。(练习7-3 变长参数应用)

#include
#include
#include

#define LOCALFMT	100

void minprintf(char * fmt,...);

int main()
{
	int i = 1234;
	unsigned u = 6767;
	double d = 1.11;
	char s[]= "chenyiming";
	minprintf("%d%u%f%s",i,u,d,s);
}
void minprintf(char * fmt,...)
{
	va_list ap;
	char localfmt[LOCALFMT];
	int ival;
	double dval;
	unsigned int uval;
	char *sval;
	char *p;
	va_start(ap,fmt);
	for(p = fmt; *p; p++)
	{
		if(*p != '%')
		{
			putchar(*p);
			continue;
		}
		int i = 0;
		localfmt[i++] = '%';
		while(*(p+1) && !isalpha(*(p+1)))
			localfmt[i++] = *++p;
		localfmt[i++] = *++p;
		localfmt[i] = '\0';
		switch(*p)
		{
			case 'd':
			case 'i':
				ival = va_arg(ap,int);
				printf(localfmt,ival);
				printf("\n");
				break;
			case 'x':
			case 'X':
			case 'u':
			case 'o':
				uval = va_arg(ap,unsigned);
				printf(localfmt,uval);
				printf("\n");
				break;
			case 'f':
				dval = va_arg(ap,double);
				printf(localfmt, dval);
				printf("\n");
				break;
			case 's':
				sval = va_arg(ap, char *);
				printf(localfmt,sval);
				printf("\n");
				break;
			default:
				printf(localfmt);
				printf("\n");
				break;
		}
	}
	va_end(ap);
}


5 scanf用法  //来自:http://www.cnblogs.com/pcwl/articles/2032149.html

scanf函数称为格式输入函数,即按用户指定的格式从键盘上把数据输入到指定的变量之中。
scanf函数是一个标准库函数,它的函数原型在头文件“stdio.h”中,与printf函数相同,C语言也允许在使用scanf函数之前不必包含stdio.h文件。
scanf函数的一般形式为:
scanf(“格式控制字符串”,地址表列);
其中,格式控制字符串的作用与printf函数相同,但不能显示非格式字符串,也就是不能显示提示字符串。地址表列中给出各变量的地址。地址是由地址运算符“&”后跟变量名组成的。
例如:
&a, &b
分别表示变量a和变量b 的地址。
这个地址就是编译系统在内存中给a,b变量分配的地址。在C语言中,使用了地址这个概念,这是与其它语言不同的。 应该把变量的值和变量的地址这两个不同的概念区别开来。变量的地址是C编译系统分配的,用户不必关心具体的地址是多少。
变量的地址和变量值的关系如下:
在赋值表达式中给变量赋值,如:
    a=567
 则,a为变量名,567是变量的值,&a是变量a的地址。
但在赋值号左边是变量名,不能写地址,而scanf函数在本质上也是给变量赋值,但要求写变量的地址,如&a。这两者在形式上是不同的。&是一个取地址运算符,&a是一个表达式,其功能是求变量的地址。

【例5.1】
main(){
  int a,b,c;
  printf("input a,b,c\n");
  scanf("%d%d%d",&a,&b,&c);
  printf("a=%d,b=%d,c=%d",a,b,c);
}
在本例中,由于scanf函数本身不能显示提示串,故先用printf语句在屏幕上输出提示,请用户输入a、b、c的值。执行scanf语句,则退出TC屏幕进入用户屏幕等待用户输入。用户输入7  8  9后按下回车键,此时,系统又将返回TC屏幕。在scanf语句的格式串中由于没有非格式字符在“%d%d%d”之间作输入时的间隔,因此在输入时要用一个以上的空格或回车键作为每两个输入数之间的间隔。如:
   7 8 9

7
  8
  9
格式字符串
格式字符串的一般形式为:
%[*][输入数据宽度][长度]类型
其中有方括号[]的项为任选项。各项的意义如下:
1)       类型:表示输入数据的类型,其格式符和意义如下表所示。
格式
字符意义
d 输入十进制整数
o 输入八进制整数
x 输入十六进制整数
u 输入无符号十进制整数
f或e 输入实型数(用小数形式或指数形式)
c 输入单个字符
s 输入字符串

2)“*”符:用以表示该输入项,读入后不赋予相应的变量,即跳过该输入值。
 如:
 scanf("%d %*d %d",&a,&b);
当输入为:1   2   3时,把1赋予a,2被跳过,3赋予b。

3) 宽度:用十进制整数指定输入的宽度(即字符数)。
例如:
    scanf("%5d",&a);
输入:12345678
只把12345赋予变量a,其余部分被截去。
又如:
        scanf("%4d%4d",&a,&b);
输入:12345678
将把1234赋予a,而把5678赋予b。

4)       长度:长度格式符为l和h,l表示输入长整型数据(如%ld) 和双精度浮点数(如%lf)。h表示输入短整型数据。
使用scanf函数还必须注意以下几点:
1)       scanf函数中没有精度控制,如:scanf("%5.2f",&a);是非法的。不能企图用此语句输入小数为2位的实数。
2)       scanf中要求给出变量地址,如给出变量名则会出错。如 scanf("%d",a);是非法的,应改为scnaf("%d",&a);才是合法的。
3)       在输入多个数值数据时,若格式控制串中没有非格式字符作输入数据之间的间隔则可用空格,TAB或回车作间隔。C编译在碰到空格,TAB,回车或非法数据(如对“%d”输入“12A”时,A即为非法数据)时即认为该数据结束。
4)       在输入字符数据时,若格式控制串中无非格式字符,则认为所有输入的字符均为有效字符。
例如:
              scanf("%c%c%c",&a,&b,&c);
输入为:
                  d   e   f
则把'd'赋予a, ' ' 赋予b,'e'赋予c。
只有当输入为:
                  def
时,才能把'd'赋于a,'e'赋予b,'f'赋予c。
如果在格式控制中加入空格作为间隔,
如:
scanf ("%c %c %c",&a,&b,&c);
则输入时各数据之间可加空格。
【例5.2】
main(){
  char a,b;
  printf("input character a,b\n");
  scanf("%c%c",&a,&b);
  printf("%c%c\n",a,b);
}
由于scanf函数"%c%c"中没有空格,输入M  N,结果输出只有M。而输入改为MN时则可输出MN两字符。

【例5.3】
main(){
  char a,b;
  printf("input character a,b\n");
  scanf("%c %c",&a,&b);
  printf("\n%c%c\n",a,b);
 }
本例表示scanf格式控制串"%c %c"之间有空格时,输入的数据之间可以有空格间隔。

5)   如果格式控制串中有非格式字符则输入时也要输入该非格式字符。
       例如:
scanf("%d,%d,%d",&a,&b,&c);
 其中用非格式符“ , ”作间隔符,故输入时应为:
  5,6,7
又如:
    scanf("a=%d,b=%d,c=%d",&a,&b,&c);
则输入应为:
          a=5,b=6,c=7

6) 如输入的数据与输出的类型不一致时,虽然编译能够通过,但结果将不正确。
【例5.4】
main(){
  int a;
  printf("input a number\n");
  scanf("%d",&a);
  printf("%ld",a);
}
由于输入数据类型为整型,而输出语句的格式串中说明为长整型,因此输出结果和输入数据不符。如改动程序如下:

【例5.5】
main(){
    long a;
    printf("input a long integer\n");
    scanf("%ld",&a);
    printf("%ld",a);
}
运行结果为:
    input a long integer
    1234567890
1234567890
当输入数据改为长整型后,输入输出数据相等。

【例5.6】
main(){
  char a,b,c;
  printf("input character a,b,c\n");
  scanf("%c %c %c",&a,&b,&c);
  printf("%d,%d,%d\n%c,%c,%c\n",a,b,c,a-32,b-32,c-32);
 }
 输入三个小写字母,输出其ASCII码和对应的大写字母。

【例5.7】
main(){
 int a;
 long b;
 float f;
 double d;
 char c;
 printf("\nint:%d\nlong:%d\nfloat:%d\ndouble:%d\nchar:%d\n",sizeof(a),sizeof(b),sizeof(f),sizeof(d),sizeof(c));

输出各种数据类型的字节长度。

 

6.sprintf  和 sscanf 的用法

6.1 sscanf函数
int sscanf(const char *s,const char *format,..);
int scanf(const char *format,...);

sscanf与scanf等价,所不同的是,前者的输入字符来源于字符串s,而scanf以stdin作为输入源。

6.1.1 %*d,%*s:*号表示此数据不读入,忽略掉。
比如
ch="MemTotal: 2028248 kB"
sscanf(ch,"%*s%d",&total);
printf("%d/n",total);

输出结果:2028248

6.1.2 取指定长度的字符串
sscanf("123456 ", "%4s", buf);
printf("%s/n", buf);
  结果为:1234

6.1.3 取到指定字符为止的字符串。如在下例中,取遇到空格为止字符串。
  sscanf("123456 abcdedf", "%[^ ]", buf);
  printf("%s/n", buf);
  结果为:123456

6.1.4 sscanf(ch,"%*[^e]%[^2]",ch1);
寻找e到2之间的数,包括e但不包括2
如果中间有空格,包括空格。

6.1.5 给定一个字符串““hello, world”,仅保留world。(注意:“,”之后有一空格)
  sscanf(“hello, world”, "%*s%s", buf);
  printf("%s/n", buf);
  结果为:world
  %*s表示第一个匹配到的%s被过滤掉,即hello被过滤了
  如果没有空格则结果为NULL。

6.1.6 分割字符串
sscanf("2006:03:18", "%d:%d:%d", a, b, c);

sscanf("2006:03:18 - 2006:04:18", "%s - %s", sztime1, sztime2);

如果2006:03:18 - 2006:04:18间没有空格

sscanf("2006:03:18 - 2006:04:18", "%[0-9,:] - %[0-9,:]", sztime1, sztime2);

6.2 sprintf

- 负号时表示从后向前处理
0 表示空格填0
字符总宽度 小数点前面的数
精确度 小数点后面的数

6.2.1 把整数123 打印成一个字符串保存在s 中。
sprintf(s, "%d", 123); //产生"123"

6.2.2 sprintf(s, "%08X", 4567); //产生:"000011D7"

6.2.3 char a1[] = {'A', 'B', 'C', 'D', 'E', 'F', 'G'};
  char a2[] = {'H', 'I', 'J', 'K', 'L', 'M', 'N'};
sprintf(s, "%.7s%.7s", a1, a2);//产生:"ABCDEFGHIJKLMN"
  这可以类比打印浮点数的”%m.nf”,在”%m.ns”中,m 表示占用宽度(字符串长度不足时补空格,超出了则按照实际宽度打印),n 才表示从相应的字符串中最多取用的字符数。通常在打印字符串时m 没什么大用,还是点号后面的n 用的多。自然,也可以前后都只取部分字符:
  sprintf(s, "%.6s%.5s", a1, a2);//产生:"ABCDEFHIJKL"

sprintf(s, "%.*s%.*s", 7, a1, 7, a2);
  或者:
  sprintf(s, "%.*s%.*s", sizeof(a1), a1, sizeof(a2), a2);

6.2.4 实际上,前面介绍的打印字符、整数、浮点数等都可以动态指定那些常量值,比如:
  sprintf(s, "%-*d", 4, 'A'); //产生"65 "
  sprintf(s, "%#0*X", 8, 128); //产生"0X000080","#"产生0X
  sprintf(s, "%*.*f", 10, 2, 3.1415926); //产生" 3.14"

6.2.5 sprintf(s, "%p", &i);

6.2.6 strlen 便已经知道了结果字符串的长度。如:
  int len = sprintf(s, "%d", i);
  对于正整数来说,len 便等于整数i 的10 进制位数。

 

7, ungetch(int c, FILE *fp)

ungetc(char c,FILE *stream);
ungetch(char c);
当ungetc()中的stream==&_streams[0]即stdin时,就差不多相当于ungetch了,不过里面还是有点差别,
它们的配套函数不同,前者一般与getchar(),后者与getch();
举两个最简单的例子:
#include "stdio.h"
#include "conio.h"
int main()
{
    ungetc('a',stdin);
    printf("%c",getchar());
    getch();
}
int main()
{
    ungetch('a');
    printf("%c",getch());
    getch();
}

你可能感兴趣的:(C,Programing,Language,Note)