c语言if else语句(d[d]==i)错误 请问怎么修改才正确

C语言之数组
第九章 数组(二)
1、有以下程序
&&&& main()
&&&& { int&&a[10]={1,2,3,4,5,6,7,8,9,10}, *p=&a[3], *q=p+2;
&&&&&& printf(&%d\n&, *p + *q);
&&&&程序运行后的输出结果是
A)16&&&&&&&&&& B)10&&&&&&&&&&C)8&&&&&&&&&& D)6
2、有以下程序
&&&& void sort(int&&a[], int&&n)
&&&& { int&&i, j ,t;
&&&&&& for (i=0; i& n-1; i++)
&&&&&&&& for (j=i+1; j& j++)
&&&&&&&&&& if (a[i]& a[j]){&&t=a[i];&&a[i]=a[j];&&a[j]=t;&&}
&&&& main()
&&&&{ int&&aa[10]={1,2,3,4,5,6,7,8,9,10},
&&&&&& sort(aa+2,&&5);
&&&&&& for (i=0; i& 10; i++) printf(&%d,&,aa[i]);
&&&&&& printf(&\n&);
&&&&程序运行后的输出结果是
A)1,2,3,4,5,6,7,8,9,10,&&&&&& B)1,2,7,6,3,4,5,8,9,10,
C)1,2,7,6,5,4,3,8,9,10,&&&&&& D)1,2,9,8,7,6,5,4,3,10,
3、有以下程序
&&&&void&&f(int&&*x,int&&*y)
&&&&{&& int&&t;
&&&&&&&&t=*x;*x=*y;*y=t;
&& {&& int&&a[8]={1,2,3,4,5,6,7,8},i,*p,*q;
&&&&&&&&p=a;q=&a[7];
&&&&&& while(p & q)
&&&&&&&& f(p,q);
&&&&&&&& p++;
&&&&&&&& q--;
&&&&&&for(i=0;i& 8;i++)
&&&&&&&&&&printf(&%d,&,a[i]);
程序运行后的输出结果是
A)8,2,3,4,5,6,7,1,&&&&&&&& B)5,6,7,8,1,2,3,4,
C)1,2,3,4,5,6,7,8,&&&&&&&& D)8,7,6,5,4,3,2,1,
4、有以下程序
&&&&#define&&N&&20
&&&&fun(int&&a[],int&&n,int m)
&&&&{&& int i,j;
&&&&&&&&for(i=m;i&=n;i--)
&&&&&&&&a[i+1]=a[i];
&&&&&& int i,a[N]={1,2,3,4,5,6,7,8,9,10};
&&&&&& fun(a,2,9);
&&&&&& for(i=0;i& 5;i++)
&&&&&& printf(&%d&,a[i]);
&&&&程序运行后的输出结果是
A)10234&&&&&&&& B)12344&&&&&&&& C)12334&&&&&&&&D)12234
5、有以下程序
&&&&prt(int&&*m,int&&n)
&&&&&&&&for(i=0;i&i++)
&&&&&&&& m[i]++;
&&&&main()
&&&&&& int a[]={1,2,3,4,5},i;
&&&&&& prt(a,5);
&&&&&& for(i=0;i&5;i++)
&&&&&& printf(&%d,&,a[i]);
&&&&程序运行后的输出结果是
A)1,2,3,4,5,&&&&&&&&B)2,3,4,5,6,
C)3,4,5,6,7,&&&&&&&&D)2,3,4,5,1,
6、有以下程序
&&&&int f(int a)
&&&&{ return a%2; }
&&&&main()
&&&&{ int s[8]={1,3,5,2,4,6},i,d=0;
&&&&&&for (i=0;f(s[i]);i++) d+=s[i];
&&&&&&printf(&%d\n&,d);
程序运行后的输出结果是
A)9&&&&&&&&&&B)11&&&&&&&&&&C)19&&&&&&&&&& D)21
7、有以下程序
&&&&void sort(int a[],int n)
&&&&{&&int i,j,t;
&&&&&& for(i=0;i& n-1;i+=2)
&&&&&&&&for(j=i+2;j&j+=2)
&&&&&&&&&&if(a[i]& a[j])&&{&&t=a[i];a[i]=a[j];a[j]=t;}
&&&&main()
&&&&{ int aa[10]={1,2,3,4,5,6,7,8,9,10},i;
&&&&&&sort(aa,10);
&&&&&&for(i=0;i& 10;i++)&&&&printf(&%d, &,aa[i]);
&&&&&&printf(&\n&);
&&&&其输出结果是
A)1,2,3,4,5,6,7,8,9,10,&&&&&&&& B)10,9,8,7,6,5,4,3,2,1,
C)9,2,7,4,5,6,3,8,1,10,&&&&&&&& D)1,10,3,8,5,6,7,4,9,2,
8、以下程序的输出结果是
&&&&f(int&&b[],int&&m,int&&n)
&&&&{ int&&i,s=0;
&&&&&&for(i=m;i&i=i+2)&&s=s+b[i];
&&&&&&return&&s;
&&&&main()
&&&&{int&&x,a[]={1,2,3,4,5,6,7,8,9};
&&&& x=f(a,3,7);
&&&& printf(&%d\n&,x);&& }
A)10&&&&&&&&&& B)18&&&&&&&&&&&&C)8&&&&&&&&&&&&D)15
9、假定int类型变量占用四个字节,其有定义:int&&x[10]={0,2,4};,则数组x在内存中所
占字节数是
A)12&&&&&&&&&& B)24&&&&&&&&&&&&C)20&&&&&&&&&& D)40
10、当调用函数时,实参是一个数组名,则向函数传送的是
A)数组的长度&&&&&&&&&&&&&&&&&&&&&&&& B)数组的首地址
C)数组每一个元素的地址&&&&&&&&&&&&&& D)数组每个元素中的值
11、执行下面的程序段后,变量k中的值为
&& int&&k=3, s[2];
&& s[0]=k;&&k=s[1]*10;
A)不定值&&&&&&&& B)33&&&&&&&&&&&&&& C)30&&&&&&&&&&&&D)10
12、下列程序执行后的输出结果是
&& void func(int *a,int b[])
&& { b[0]=*a+6; }
&& { int a,b[5];
&&&& a=0;&&&&&& b[0]=3;
&&&& func(&a,b);&&printf(&%d\n&,b[0]);
A)6&&&&&&&&&& B)7&&&&&&&&&&&&C)8&&&&&&&&&& D)9
13、下列程序的输出结果是
&& { int a[5]={2,4,6,8,10},*P,* *k;
&&&& p=a;&&&&&& k=&p;
&&&& printf(&%d&,*(p++));&&&&&& printf(&%d\n&,* *k);&&&& }
A)4&&4&&&&&&&&B)2&&2&&&&&&&& C)2&&4&&&&&&&& D)4&&6
二、填空题
1、以下程序的输出结果是&&&&&&
&&&&main()
&&&&{ int&&arr[]={30,25,20,15,10,5},&&*p=
&&&&&&p++;
&&&&&&printf(&%d\n&,*(p+3));
2、若有以下定义,则不移动指针p,且通过指针p引用值为98的数组元素的表达式
是&&&&&&&&
&&&&int&& w[10]={23,54,10,33,47,98,72,80,61}, *p=w;
3、若有以下定义和语句,则使指针p指向值为36的数组元素的表达式是&&&&&&&&
&& int a[10]={19,23,44,17,37,28,49,36},*p;
4、下面的findmax函数返回数组s中最大元素的下标,数组中元素的个数由t传入,请填空。
&&&&findmax( int s[], int t)
&& { int k,
&&&& for( p=0, k=p; p& p++)
&&&& if( s[p]&s[k])&&&&&&&&
第九章 数组(三)
一、选择题
1、有以下程序
&&&& void swap1(int&&c0[], int&&c1[])
&&&& { int&&
&&&&&& t=c0[0];&&&&c0[0]=c1[0];&&&&c1[0]=t;
&&&& void swap2(int&&*c0, int&&*c1)
&&&& { int&&t;
&&&&&& t=*c0;&&&&*c0=*c1;&&&&*c1=t;
&&&& main()
&&&& { int&&a[2]={3,5},&&b[2]={3,5};
&&&&&& swap1(a, a+1);&&&&swap2(&b[0], &b[1]);
&&&&&& printf(&%d %d %d %d\n&,a[0],a[1],b[0],b[1]);
&&&&程序运行后的输出结果是
A)3 5 5 3&&&&&&B)5 3 3 5&&&&&&C)3 5 3 5&&&&&&D)5 3 5 3
2、以下叙述中错误的是
A)对于double类型数组,不可以直接用数组名对数组进行整体输入或输出
B)数组名代表的是数组所占存储区的首地址,其值不可改变
C)当程序执行中,数组元素的下标超出所定义的下标范围时,将给出&下标越界&的出错信息
D)可以通过赋初值的方式确定数组元素的个数
3、以下程序调用findmax函数求数组中值最大的元素在数组中的下标,请选择填空。
&&&&# #include
&&&&findmax(*s,t,*k)
&& for(p=0,*k=p;p &p++)
&&&&&&&&&&&& if(*k & s[p])
&&&&&&&&&&&&&& _________;}
&&&& main()
&&&&{ int&&a[10] , i ,
&&&&&&for ( i=0 ; i&10 ; i + + )&&scanf(&%d&,&a[i]);
&&&&&&findmax ( a,10,&k );
&&&&&&printf&&( &%d,%d\n& , k , a[k] );&&}
A)k=p&&&&&&B)*k=p-s&&&& C)k=p-s&&&& D)*k=p
4、有以下程序
&&&&void sum(int *a)
&&&&{ a[0]=a[1];}
&&&&main()
&&&&{&&int aa[10]={1,2,3,4,5,6,7,8,9,10},i;
&&&&&& for(i=2;i&=0;i--)
&&&&&& sum(&aa[i]);
&&&&&& printf(&%d\n&,aa[0]);
&&&&执行后的输出结果是
A)4&&&&&&&&&&&&B)3&&&&&&&&&&&&C)2&&&&&&&&&&&& D)1
5、请选出以下程序的输出结果
&&&&#include
&&&&fun(int *s,int n1,int n2)
&&&&{ int i, j,
&&&&&&i=n1; j=n2;
&&&&&&while(i& j)
&&&& { t= *(s+i); *(s+i)= *(s+j); *(s+j)=t;
&&&&&&i++; j--;
&&&&&&}&& }&&
&&&&main()
&& { int a[10]={1,2,3,4,5,6,7,8,9,0},i, *p=a;
&&&&fun(p,0,3); fun(p,4,9); fun(p,0,9);
&&&&for(i=0; i&10; i++) printf(&%d&, *(a+i));
&&&&printf(&\n&);
A)0 9 8 7 6 5 4 3 2 1&&&&&&&& B)4 3 2 1 0 9 8 7 6 5
C)5 6 7 8 9 0 1 2 3 4&&&&&&&& D)0 9 8 7 6 5 1 2 3 4
6、以下程序中函数reverse的功能是将a所指数组中的内容进行逆置。
&&&&void reverse(int&&a[ ],int n)
&&&&{ int&&i,t;
&&&& for(i=0;i& n/2;i++)
&&&&{ t=a[i]; a[i]=a[n-1-i];a[n-1-i]=t;}
&&&&main()
&&&&{ int&&b[10]={1,2,3,4,5,6,7,8,9,10}; int i,s=0;
&&&&&&reverse(b,8);
&&&&&&for(i=6;i&10;i++) s+=b[i];
&&&&&&printf(&%d\n&,s);
&&&&程序运行后的输出结果是
A)22&&&&&&&& B)10&&&&&&&& C)34&&&&&&&&&&D)30
7、下列程序的输出结果是
{&&char&&a[10]={9,8,7,6,5,4,3,2,1,0},*p=a+5;
&&&&printf(&%d&,*--p);
A)非法&&&&&& B)a[4]的地址&&&&&&&&C)5&&&&&&&&&&D)3
8、请读程序:
&&&&#include
&&&&f(in b[], int n)
&& { int i,
&&&& for(i=0; i&=n; i++) r=r*b[i];
&& { int x, a[]={ 2,3,4,5,6,7,8,9};
&&&& x=f(a, 3);
&&&& printf(&%d\n&,x);
上面程序的输出结果是
A)720&&&&&&&& B)120&&&&&&&&&&C)24&&&&&&&&&& D)6
9、以下函数返回a所指数组中最小的值所在的下标值
&&&&fun(int&& *a,&&int&&n)
&&&&{ int&& i,j=0,p;
&&&&&&p=j;
&&&&&&for(i=j;i&i++)
&&&&&&if(a[i]& a[p])__________;
&&&&&&return(p);
&&&&在下划线处应填入的是
A)i=p&&&&&&&& B)a[p]=a[i]&&&&&& C)p=j&&&&&&&& D)p=i
10、若有以下调用语句,则不正确的fun函数的首部是
&&&& int a[50],n;&&&&&&&&&&&&&&&&&
&&&& fun(n, &a[9]);
A)void fun(int m, int x[])&&&&&&B)void fun(int s, int h[41])
C)void fun(int p, int *s)&&&&&& D)void fun(int n, int a)
11、以下程序的运行结果是
&&&&#include&& &stdio.h&
&&&&main()
&&&& { int a[]={1,2,3,4,5,6,7,8,9,10,11,12,};
&&&&&& int&&*p=a+5, *q=NULL;
&&&&&& *q=*(p+5);
&&&&&& printf(&%d&&%d\n&,*p,*q);
A)运行后报错&& B)6&&6&&&& C)6&&12&&&&&& D)5&&5
12、 下面程序把数组元素中的最大值放入a[0]中。则在if 语句中的条件表达式应该是
&& { int&&a[10]={6,7,2,9,1,10,5,8,4,3},*p=a,i;
&&&& for(i=0;i&10;i++,p++)
&&&& if(________) *a=*p;
&&&& printf(&%d&,*a);
A)p&a&&&& B)*p&a[0]&&&&C)*p&*a[0]&&&& D)*p[0]& *a[0]
13、下面程序输出数组中的最大值,由s指针指向该元素.
&&&&main()
&&&&{ int a[10]={6,7,2,9,1,10,5,8,4,3,},*p,*s;
&&&&&&for(p=a,s=a; p-a&10; p++)
&&&&&&if(&&&&&&&&)s=p;
&&&&&&printf(&The max:%d&,*s):
则在if语句中的判断表达式应该是
A)p&s&&&&&&&&&&&&B)*p&*s&&&&&&&& C)a[p]&a[s]&&&&&& D)p-a&p-s
14、要求函数的功能是在一维数组中查找值;若找到则返回所在的下标值,否则返回0;数列
放在到中。不能正确执行的函数是
A) funa(int *a,int n,int x )&&&&&&&&&&B)funb(int *a,int n,int x)
&&&&{ *a=x;&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& {
&&&&&&while(a[n]!=x)n--;&&&&&&&&&&&&&&&&&&&&for(k=1;k&=n;k++)
&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& if(a[k]==x)
&&&& }&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&return 0; }
C) func(int a[],int n,int x )&&&&&&&&&&D)fund(int a[],int n,int x )
&&&&{ int *k;&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&{ int k=0;
&&&& a[0]=x;k=a+n;&&&&&&&&&&&&&&&&&&&&&&&&&& do
&&&& while(*k!=x) k--;&&&&&&&&&&&&&&&&&&&&&& k++;
&&&& return k-n;&&&&&&&&&&&&&&&&&&&&&&&&&&&& while((k& n+1)&&(a[k]!=x));
&&&& }&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&if(a[k]==x)
else return 0; }
二、填空题:
1、以下程序从终端读入数据到数组中,统计其中正数的个数,并计算它们之和。请填空。
&&&&main()
&& { int i,a[20],sum,
&&&& sum=count=0;
&&&& for(i=0;i&20;i++)&&scanf(&%d&, );&&
&&&& for (i=0;i&20;i++)
&&&&{ if(a[i]&0)
&&&&&&{ count++;
&&&&&&&&sum+= ;&&
&& printf(&sum=%d,count=%d\n&,sum,count);}
2、若已定义:int&& a[10],,以下fun函数的功能是:在第一个循环中给前10个数组元素
依次赋1、2、3、4、5、6、7、8、9、10;在第二个循环中使a数组前10个元素中的值对称折叠,
变成1、2、3、4、5、5、4、3、2、1。请填空。
&&&&fun(int a[])
&& { int&&
&&&& for(i=1; i&=10; i++)&&&&=i;&&&&&&
&&&& for(i=0; i&5; i++)&& =a[i];&&&&&&
3、以下fun函数的功能是:累加数组元数中的值.n为数组中元素的个数, 累加的和值放入x所指
的存储单元中,请填空.
&&&&fun(int b[ ],int n, int&&*x)
&& { int&& k, r=0;
&&&& for(k=0;k&k++) r=
&&&&&&=r;&&&&&&&&
4、下面fun函数的功能是将形参x的值转换成二进制数,所得二进制数的每一位数放在一维数组
中返回,二进制数的最低位放在下标为0的元素中,其它依此类推。请填空。
&&&&fun(int x,int b[])
&& { int k=0,r;
&&&&{ r=x%&&;&&
&&&&b[k++]=r;
&&&&x/=&&; } while(x); }&&
5、下面程序通过函数average计算数组中各元素的平均值,请填空.
&&&&float average(int * pa,int n)
&&&& float avg=0.0;
&&&& for(i=0;i &i++) avg=avg+;
&&&& avg= ;&&
&&&&main()
&&{ int i,a[5]={2,4,6,8,10};
&&&&mean=average(a,5); printf(&mean=%f\n&,mean);}
6、下面程序的输出是&&&&&&
&&&&main()
&&{ int a[]={ 2,4,6}, *prt=&a[0], x=8,y,z;
&&&&for(y=0; y&3; y++)
&&&&z=(*(prt+y)& x)? *(prt+y):x;&&
&&&&printf(&%d\n&, z);}
第九章数组(二)
一.选择题 1&&5 BCDCB&&&& 6&&10 ACADB&&&&11&&13 AAC&& 二.填空 【1】 10&& 【2】 p[5] 【3】 p=p+7 【4】&&k=p;
第九章数组(三)
一.选择题 1&&5 DCDAC&&&&&&6&&10&&ACBDD&&&& 11&&14&&ABBC 二.填空 【1】、&&&a[i]&&a[i] 【2】、a[i-1]&& a[9-i] 【3】、r+b[k]或b[k]或*(b+k)+r&& *x 【4】、2&& 2&& 【5】、*(pa+i)或pa[i]&& avg/n 或avg/(float)n 或 avg/(double)n&& 【6】、6&&
您对本文章有什么意见或着疑问吗?请到您的关注和建议是我们前行的参考和动力&&
您的浏览器不支持嵌入式框架,或者当前配置为不显示嵌入式框架。C语言中这些错误提示是什么意思?该怎么改呢?_百度知道
C语言中这些错误提示是什么意思?该怎么改呢?
undeclared identifierD;writeDat&#39.cpp(81) ;11:\11: error C; : error C2373.11.cpp(45) ;Microsoft Visual Studio\MyProjects\11\11;Microsoft Visual Studio\Microsoft Visual Studio&#92: function s return type assumed执行 cl.cpp(69) : &#39: '11&#92.cpp(95) ;readDat'readDat' :&#92: undeclared identifierD: .exe 时出错:\MyProjects\Microsoft Visual Studio\MyProjects&#92: function should return a value: error C2373; : warning C4508: error C2065; return type assumedD;Microsoft Visual Studio\clrscr'11;软件\11;11\void' ;Microsoft Visual Studio\11;MyProjects&#92: error C2065;软件\11\void'11&#92:\Microsoft Visual Studio\ ; &#39: undeclared identifierD; ;11:\MyProjects\11;MyProjects&#92.cpp(53) ; different type modifiersD: &#39: warning C4508:\Microsoft Visual Studio\软件\writeDat'writeDat'软件\system&#39: redefinition: &#39.cpp(75) : undeclared identifierD:\软件\MyProjects&#92.cpp(39) ;11\软件&#92: &#39: &#39: &#39.obj - 1 error(s): error C2065; ' different type modifiersD;readDat&#39:\MyProjects\软件&#92: &#39.cpp(43) ; ;11\软件&#92
fclose(in);i++)
bb[i]=aa[i],&),& writeDat();10;clrscr();int aa[200].dat&199;#include &lt.h&j++)
if(aa[i]%1000&aa[j]%1000||aa[i]%1000==aa[j]%1000&&aa[i]&;r&quot,&200; for(i=0;#include &
fprintf(;j&lt,& i&%d;200;in=fopen(&aa[j])
{data=aa[i].h&gt.h&} for(i=0; for(i=0,bb[10]; jsSort(); i++){
printf(&void jsSort(){n&
int i这是程序; %d&}readDat() {
FILE *}void main(){ readDat(); i++) fscanf();aa[j]=i&lt,bb[i]); i&i++)
for(j=i+1; out=fopen(& int i:#include &
for(i=0.dat&quot,bb[i]);%d&#92,&aa[i]);
}writeDat(){ FILE *out,j;;w&quot,i&10;;,& } fclose(out);aa[i]=aa[j]
提问者采纳
h&不接受参数;;);out=fopen(&aa[j]=in=fopen(&quot,&fclose(in); (需要包含string.0中没有clrscr()函数.h&不接受参数,如果没有返回值; /for(i=0; i++){printf(&,请将类型设为void函数在调用之前要声明;aa[j]%1000||aa[i]%1000==aa[j]%1000&&aa[i]&lt。要包含i++)bb[i]=aa[i].h头文件请写出你的程序;;out。*********************************************程序修改如下;200,data,无返回值,& %d&quot。{FILE *in.dat&w&i++)for(j=i+1,bb[i]),无返回值;);/}void writeDat(void) &#47,你应该是使用VC++6;system(&函数在使用前要声明void writeDat(void);200,在VC++6;writeDat(),bb[i]);}void readDat(void) /i&lt.0编译;n&quot,我使用了效果相同的system(&quot,bb[10];j&void jsSort(void){/for(i=0;cls&cls&);),&quot.h)#include &jsSort(); i++) fscanf(%d;,&}void readDat(void);199函数的返回值类型不可省略;;aa[j]){data=aa[i];/%d&#92,&aa[i]);}for(i=0;10;r& /#include &fprintf(out.dat&for(i=0;10;/}fclose(out);i&lt,&quot。{FILE * i&lt.h&#include &int aa[200];aa[i]=aa[j];同上void main(){readDat(),j;j++)if(aa[i]%1000& i&lt
提问者评价
谢谢,这个程序编译时是没有错误了,但是执行时还是有错误,这是怎么回事呢?
其他类似问题
为您推荐:
其他1条回答
像这些'readDat' 等都没有定义或者没有把正确的头文件包含进来
您可能关注的推广回答者:
错误提示的相关知识
等待您来回答
下载知道APP
随时随地咨询
出门在外也不愁C语言..编写完后改了很多错误,可是下面这个还有错误,却不知道怎么改,求教..._百度知道
C语言..编写完后改了很多错误,可是下面这个还有错误,却不知道怎么改,求教...
\/A[c]=A[y];/);),c+1);;n&quot:%d\ if(choose==6)
/每次比较C次
if(A[c]&/输入数字3;&#47,d; if(choose==3)
&#47,c;);0-&gt,y;
int main(){ n&查询信息\显示全部学生信息\修改信息&#92,&b);
for(c=0:&);/n&n&n&
/b&n&c&;5.h&);n&n&%d学号的同学的成绩为,c),显示全部学生信息
/);A[y])
t=A[c];请先添加信息\
for(c=0;输入的数字不是0的情况下
if(choose==1)
/);退出成功\c++)
/添加信息\y++)
/n&);/c++)
printf(&n&n请输入学号; printf(&quot,&a[b][c]);
printf(&输入数字6,b是学号 ;c++)
printf(&quot,a;/30;);;请输入第%d科成绩;30;A[y]=t,添加信息
for(c=0;输入数字1;/ printf(&请先添加信息\d=1说明已经输入了数据
for(b=0; printf(&quot,t:&
else printf(&
scanf(&quot:\1-&
for(y=0;6-&第%d科的成绩为;; if(choose=;),d是能否读入数据的关键
printf(&);
printf(&;退出\ printf(&quot,A[c]),b;);y&lt,排序数据
printf(&请先添加信息\/n&
d+1;%d& printf(&quot,c+1;30;),i; printf(&choose是输入数字; if(choose==2)
int A[30];
printf(&;删除信息\n&quot#include&5;5-&b++)
printf(&quot:\n&
printf(&quot!0)
&#47,c+1;输入数字2,&a[b][c]);c&30次比较
for(c=1;4-&gt,b);%d&5;;stdio,&i[b][c]):\第%d科的成绩为;c++)
printf(&3-&n&;c&n&n&请输入要查询的科目;
scanf(&排序后;),查询信息
printf(&按成绩排序\\c&2-&%d &quot:&#92,c是科目;n&quot
我有更好的答案
你这里面问题不是一点大,加我qq,给你解释,qq私信发你
#include&stdio.h&
int caozuo(int a[30][5],int choose,int d)
int b,c,x,y,t;
if(choose==1)
//输入数字1,显示全部学生信息
//d=1说明已经输入了数据
for(b=0;b&5;b++)
{printf(&%d学号的同学的成绩为:\n&,b+1);
for(c=0;c&5;c++)
printf(&第%d科的成绩为:%d\n&,c+1,a[b][c]);}}
printf(&请先添加信息\n&);}
if(choose==2)
//输入数字2,查询信息
printf(&\n请输入学号:&);
scanf(&%d&,&b);
QQ多少..T.T
其他类似问题
为您推荐:
等待您来回答
下载知道APP
随时随地咨询
出门在外也不愁Bash编程易犯的错误 - 博客 - 伯乐在线
& Bash编程易犯的错误
& 4.4K 阅读
前一段时间发现一个很好的,上面有很多优秀的Bash文章。最近挑了一篇介绍Bash编程容易犯的各种错误的文章看,收获很多,不敢独享,把这篇文章以半翻译半笔记的形式分享给大家。
1. for i in $(ls *.mp3)
Bash写循环代码的时候,确实比较容易犯下面的错误:
for i in $(ls *.mp3); do
some command $i
for i in $(ls)
for i in `ls`
for i in $(find . -type f)
for i in `find . -type f`
files=($(find . -type f))
for i in ${files[@]}
这里主要两个问题:
使用命令展开时不带引号,其执行结果会使用IFS作为分隔符,拆分成参数传递给for循环处理;
我们不能避免某些文件名中包含空格,Shell会对$(ls *.mp3)展开的结果会被做单词拆分()的处理。假设有一个文件,名字为01 – Don’t Eat the Yellow Snow.mp3,for循环处理的时候,会今次遍历文件名中的每个单词:01, -, Don’t, Eat等等:
$ for i in $(ls *.mp3); do echo $i; done
Don't
比这更差的情况是,上面命令展开的结果可能被Shell进一步处理,比如。比如,ls执行的结果中包含*号,按照通配符的规则, *号会被展开成当前目录下的所有文件:
$ touch &1*.mp3& &1.mp3& &11.mp3& &12.mp3&
$ for i in $(ls *.mp3); do echo $i; done
1*.mp3 1.mp3 11.mp3 12.mp3
不过,在这种场景下,你即使加上引号,也是无济于事的:
$ for i in &$(ls *.mp3)&; do echo --$i--; done
--1*.mp3 1.mp3 11.mp3 12.mp3--
加上引号后,ls执行的结果会被当成一个整体,所以for循环只会执行一次,达不到预期的效果。
事实上,这种情况下,根本不需要使用ls命令。ls命令的结果本身就设计成给人读的,而不是给脚本解析的。正确的处理方法是,直接使用文件名展开(通配符)的功能:
$ for i in *.mp3; do
文件名展开是位于各种展开(花括号展开、变量替换、命令展开等)功能中的最后一个环节,所以不会有之前不带引号的命令展开的副作用。如果你需要递归地处理文件,可以考虑。
到这一步,之间的问题看样子已经修复了。但是,如果你进一步思考,假设当前目录上没有文件时会怎么样?没有文件的时候,*.mp3不会被展开直接传递给for循环处理,所以这个时候循环还是会执行一次。这种情况不是我们预期的行为。保险起见,可以在循环处理的时候,检查下文件是否存在:
for i in *.mp3; do
[ -e &$i& ] || continue
some command &$i&
如果你有和避免的习惯,你完全可以避免很多错误。
注意下循环体内部的”$i”,这里会导致下面我们要说的另外一个比较容易犯的错误。
2. cp $file $target
上面的命令有什么问题呢?如果你提前知道,$file和$target文件名中不会包含空格或者*号。否则,这行命令执行前在经过单词拆分和文件名展开的时候会出现问题。所以,两次强调,在使用展开的地方切勿忘记使用引号:
$ cp -- &$file& &$target&
如果不带引号,当你执行如下命令时就会出错:
$ file=&01 - Don't Eat the Yellow Snow.mp3&
$ target=&/tmp&
$ cp $file $target
cp: cannot stat ‘01’: No such file or directory
如果带上引号,就不会有上面的问题,除非文件名以’-'开头,在这种情况下,cp会认为你提供的是一个命令行选项,这个错误下面会介绍。
3. 文件名中包含短横’-’
文件名以’-'开头会导致许多问题,*.mp3这种通配符会根据当前的展开成一个列表,但在绝大多数环境下,’-'排序的时候会排在大多数字母前。这个展开的列表传递给有些命令的时候,会错误的将-filename解析成命令行选项。这里有两种方法来解决这个问题。
第一种方法是在命令和参数之间加上–,这种语法告诉命令不要继续对–之后的内容进行命令行参数/选项解析:
$ cp -- &$file& &$target&
这种方法可以解这个问题,但是你需要在每个命令后面都要加上–,而且依赖具体的命令解析的方式,如果一些命令不兼容这种约定俗成的规范,这种做法是无效的。
另外一种方法是,确保文件名都使用相对或者绝对的路径,以目录开头:
for i in ./*.mp3; do
cp &$i& /target
这种情况下,即使某个文件以-开头,展开后文件名依然是./-foo.mp3这种形式,完全不会有问题。
4. [ $foo = "bar" ]
这是一个与第2个问题类似的问题,虽然用到了引号,但是放错了位置,对于字符串字面值,除非有特殊符号,否则不大需要用引号括起来。但是,你应该把变量的值用括号括起来,从而避免它们包含空格或能通配符,这一点我们在前面的问题中都解释过。
这个例子在以下情况下会出错:
如果[中的变量不存在,或者为空,这个时候上面的例子最终解析结果是:
[ = &bar& ] # 错误!
并且执行会出错:unary operator expected,因为=是二元操作符,它需要左右各一个操作数。
如果变量值包含空格,它首先在执行之前进行单词拆分,因此[命令看到的样子可能是这样的:
[ multiple words here = &bar& ];
正确的做法应该是:
[ &$foo& = bar ]
这种写法,在POSIX兼容的实现中都不会有问题,即使$foo以短横"-"开头,因为POSIX实现的test命令通过传递的参数来确定执行的行为。
只有一些非常古老的shell可能会遇到问题,这个时候你可以使用下面的写法来解决(相信你肯定看到过这种写法):
# POSIX / Bourne
[ x&$foo& = xbar ]
在Bash中,还有另外一种选择是使用:
# Bash / Ksh
[[ $foo == bar ]]
这里你不需要使用引号,因为在[[里面参数不会进行展开,当然带上引号也不会有错。
不过有一点要注意的是,[[里的==不仅仅是文本比较,它会检查左边的值是否匹配右侧的表达式,==右侧的值加上引号,会让它成为一个普通的字面量,*?等通配符会失去特殊含义。
5. cd $(dirname "$f")
这又是一个引号的问题,命令展开的结果会进一步地进行单词拆分或者文件名展开。因此下面的写法才是正确的:
cd &$(dirname &$f&)&
但是,上面引号的写法可能比较怪异,你可能会认为第一、二个引号,第三、四个引号是一组的。
但是事实上,Bash将命令替换里面的引号当成一组,外面的当成另外一组。如果你是用反引号的写法,引号的行为就不是这样的了,所以。
6. [ "$foo" = bar && "$bar" = foo ]
不要在内部使用&&,Bash解析器会把你的命令分隔成两个命令,在&&之前和之后。你应该使用下面的写法:
[ bar = &$foo& ] && [ foo = &$bar& ] # POSIX
[[ $foo = bar && $bar = foo ]]
# Bash / Ksh
尽量避免使用下面的写法,虽然它是正确的,但是这种写法可移植性不好,并且已经在POSIX-2008中被废弃:
[ bar = &$foo& -a foo = &$bar& ]
7. [[ $foo & 7 ]]
原文作者认为算术比较不应该用[[,而是用((,我没弄明白是为什么。
如果有理解的同学,欢迎以评论回复,谢谢。
8. grep foo bar | while read -r; do ((count++)); done
这种写法初看没有问题,但是你会发现当执行完后,count变量并没有变化。原因是管道后面的命令是在一个中执行的。
POSIX规范并没有说明管道的最后一个命令是不是在子Shell中执行的。一些shell,例如ksh93或者Bash&=4.2可以通过shopt -s lastpipe命令,指明管道中的最后一个命令在当前shell中执行。由于篇幅限制,在此就不展开,有兴趣的可以看。
9. if [grep foo myfile]
初学者会错误地认为,[是if语法的一部分,正如C语言中的if ()。但是事实并非如此,if后面跟着的是一个命令,[是一个命令,它是内置命令test的简写形式,只不过它要求最后一个参数必须是]。下面两种写法是一样的:
if [ false ]; then echo &HELP&; fi
then echo &HELP&; fi
两个都是检查参数"false"是不是非空的,所以上面两个语句都会输出HELP。
if语句的语法是:
if COMMANDS
then &COMMANDS&
elif &COMMANDS& # optional
then &COMMANDS&
else &COMMANDS& # optional
fi # required
再次强调,[是一个命令,它同其它常规的命令一样接受参数。if是一个复合命令,它包含其它命令,[并不是if语法中的一部分。
如果你想根据grep命令的结果来做事情,你不需要把grep放到[里面,只需要在if后面紧跟grep即可:
if grep - then
如果grep在myfile中找到匹配的行,它的执行结果为0(true),then后面的部分就会执行。
10. if [bar="$foo"]; then ...
正如上一个问题中提到的,[是一个命令,它的参数之间必须用空格分隔。
11. if [ [ a = b ] && [ c = d ] ]; then ...
不要用把[命令看成C语言中if语句的条件一样,它是一个命令。
如果你想表达一个复合的条件表达式,可以这样写:
if [ a = b ] && [ c = d ]; then ...
注意,if后面有两个命令,它们用&&分开。等价于下面的写法:
if test a = b && test c = then ...
如果第一个test(或者[)命令返回false,then后面的语句不会执行;如果第一个返回true,第二个test命令会执行;只有第二个命令同样返回true的情况下,then后面的语句才会执行。
除此之外,还可以使用[[关键字,因为它支持&&的用法:
if [[ a = b && c = d ]]; then ...
12. read $foo
read命令中你不需要在变量名之前使用$。如果你想把读入的数据存放到名为foo的变量中,下面的写法就够了:
或者,更加安全地方法:
IFS= read -r foo
read $foo会把一行的内容读入到变量中,该变量的名称存储在$foo中。所以两者的含义是完全不一样的。
13. cat file | sed s/foo/bar/ & file
你不应该在一个管道中,从一个文件读的同时,再往相同的文件里面写,这样的后果是未知的。
你可以为此创建一个临时文件,这种做法比较安全可靠:
# sed 's/foo/bar/g' file & tmpfile && mv tmpfile file
或者,如果你用得是 GNU Sed 4.x 以上的版本,可以使用-i 选项即时修改文件的内容:
# sed -i 's/foo/bar/g' file
14. echo $foo
这种看似无害的命令往往会给初学者千万极大的困扰,他们会怀疑是不是因为 $foo 变量的值是错误的。事实却是因为,$foo 变量在这里没有使用双引号,所以在解析的时候会进行和,最终导致执行结果与预期大相径庭:
msg=&Please enter a file name of the form *.zip&
这里整句话会被拆分成单词,然后其中的通配符会被展开,例如*.zip。当你的用户看到如下的结果时,他们会怎样想:
Please enter a file name of the form freenfss.zip lw35nfss.zip
再举一个例子(假设当前目录下有以 .zip 结尾的文件):
var=&*.zip&
# var 包括一个星号,一个点号和 zip
echo &$var&
# 输出 *.zip
# 输出所有以 .zip 结尾的文件
实际上,这里使用 echo 命令并不是绝对的安全。例如,当变量的值包含-n时,echo 会认为它是一个合法的选项而不是要输出的内容(当然如果你能够保证不会有-n 这种值,可以放心地使用 echo 命令)。
完全可靠的打印变量值的方法是使用 printf:
printf &%s\n& &$foo&
15. $foo=bar
16. foo = bar
当赋值时,等号两边是不允许出现空格的,这同 C 语言不一样。当你写下 foo = bar 时,shell 会将该命令解析成三个单词,然后第一个单词 foo 会被认为是一个命令,后面的内容会被当作命令参数。
同样地,下面的写法也是错误的:
$foo = # COMPLETELY WRONG!
正确的写法应该是这样的:
&pre class=&prettyprint lang-sh&&
# More Right.
17. echo &&EOF
当脚本需要嵌入大段的文本内容时,往往是一个非常有用的工具,它将其中的文本作为命令的标准输入。不过,echo 命令并不支持从标准输入读取内容,所以下面的写法是错误的:
# This is wrong:
echo &&EOF
Hello world
How's it going?
正确的方法是,使用 cat 命令来完成:
# This is what you were trying to do:
Hello world
How's it going?
或者可以使用双引号,它也可以跨越多行,而且因为 echo 命令是内置命令,相同情况下它会更加高效:
echo &Hello world
How's it going?&
18. su -c 'some command'
这种写法“几乎”是正确的。问题是,在许多平台上,su 支持 -c 参数,但是它不一定是你认为的。比如,在 OpenBSD 平台上你这样执行会出错:
$ su -c 'echo hello'
su: only the superuser may specify a login class
在这里,。如果你想要传递 -c 'some command' 给 shell,最好在之前显示地指定 username:
$ su root -c 'some command' # Now it's right.
19. cd / bar
如果你不检查 cd 命令执行是否成功,你可以会在错误的目录下执行 bar 命令,这有可能会带来灾难,比如 bar 命令是 rm -rf *。
你必须经常检查 cd 命令执行是否有错误,简单的做法是:
cd /foo && bar
如果在 cd 命令后有多个命令,你可以选择这样写:
cd /foo || exit 1
bat ... # Lots of commands.
出错时,cd 命令会报告无法改变当前目录,同时将错误消息输出到标准错误,例如"bash: cd: /foo: No such file or directory"。如果你想要在标准输出同时输出自定义的错误提示,可以使用复合命令():
cd /net || { echo &Can't read /net. Make sure you've logged in to the Samba network, and try again.&; exit 1; }
more_stuff
注意,在{号和 echo 之间需要有一个空格,同时}之前要加上分号。
顺便提一下,如果你要在脚本里频繁改变当前目录,可以看看 pushd/popd/dirs 等命令,可能你在代码里面写的 cd/pwd 命令都是没有必要的。
说到这,比较下下面两种写法:
find ... -type d -print0 | while IFS= read -r -d '' do
cd &$subdir& && whatever && ...
cd &$here&
find ... -type d -print0 | while IFS= read -r -d '' do
(cd &$subdir& || ...)
下面的写法,在循环中 fork 了一个子 shell 进程,子 shell 进程中的 cd 命令仅会影响当前 shell的环境变量,所以父进程中的环境命令不会被改变;当执行到下一次循环时,无论之前的 cd 命令有没有执行成功,我们会回到相同的当前目录。这种写法相较前面的用法,代码更加干净。
20. [ bar == "$foo" ]
正确的用法:
[ bar = &$foo& ] && echo yes
[[ bar == $foo ]] && echo yes
21. for i in {1..10}; do ./something &; done
你不应该在&后面添加分号,删除它:
for i in {1..10}; do ./something & done
或者改成多行的形式:
for i in {1..10}; do
./something &
&和分号一样也可以用作命令终止符,所以你不要将两个混用到一起。一般情况下,分号可以被换行符替换,但是不是所有的换行符都可以用分号替换。
22. cmd1 && cmd2 || cmd3
有些人喜欢把&&和||作为if...then...else...fi 的简写语法,在多数情况下,这种写法没有问题。例如:
[[ -s $errorlog ]] && echo &Uh oh, there were some errors.& || echo &Successful.&
但是,这种结构并不是在所有情况下都完全等价于 if...fi 语法。这是因为在&&后面的命令执行结束时也会生成一个返回码,如果该返回码不是真值(0代表 true),||后面的命令也会执行,例如:
true && ((i++)) || ((i--))
echo $i # 输出 0
看起来上面的结果应该是返回1,但是结果却是输出0,为什么呢?原因是这里 i++ 和 i-- 都执行了一遍。
其中,((i++))命令执行算术运算,表达式计算的结果为0。这里和 C 语言一样,表达式的结果为0被认为是 false。所以当 i=0 的时候,((i++))命令执行的返回码为1(false),从而会执行接下来的((i--))命令。
如果我们在这里使用前缀自增运算符的话,返回的结果恰恰为1,因为((++i))执行的返回码是0(true):
true && (( ++i )) || (( --i ))
echo $i # Prints 1
不过在你无法保证 y 的执行结果是,绝对不要依靠 x && y || z这种写法。上面这种巧合,在 i 初始化为-1时也会有问题。
如果你喜欢代码更加安全健壮,建议使用 if...fi 语法:
echo $i # 输出 1
23. echo "Hello World!"
在交互式的 Shell 环境下,你执行以上命令会遇到下面的错误:
bash: !&: event not found
这是因为,在默认的交互式 Shell 环境下,Bash 发现感叹号时会执行历史命令展开。在 Shell 脚本中,这种行为是被禁止的,所以不会发生错误。
不幸地是,你认为明显正确地修复方法,也不能工作,你会发现:
# echo &hi\!&
最简单地方法是禁用 histexpand 选项,你可以通过 set +H 或者 set +o histexpand 命令来完成。
下面四种写法都可以解决:
# 1. 使用单引号
echo 'Hello World!'
# 2. 禁用 histexpand 选项
echo &Hello World!&
# 3. 重置 histchars
histchars=
# 4. 控制 shell 展开的顺序,命令行历史展开是在单词拆分之前执行的
# 参见:&a href=&http://linux.die.net/man/1/bash& target=&_blank&&Bash man 手册的History Expansion一节&/a&
exmark='!'
echo &Hello, world$exmark&
24. for arg in $*
和大多数 Shell 一样,Bash 支持依次读取单个命令行参数的语法。不过这并是$*或者$@,这两种写法都不正确,它们只能得到完整的参数列表,并非单独的一个个参数。
正确的语法是(没错要加上引号):
for arg in &$@&
# 或者更简单的写法
在脚本中遍历所有参数是一个再普遍不过的需求,所以 for arg 默认等价于 for arg in "$@"。$@使用双引号后就有特殊的魔力,每个参数展开后成为一个独立的单词。("$@"等价于"$1" "$2" "$3" ...)
下面是一个错误的例子:
for x in $*; do
echo &parameter: '$x'&
执行的结果为:
$ ./myscript 'arg 1' arg2 arg3
parameter: 'arg'
parameter: '1'
parameter: 'arg2'
parameter: 'arg3'
正确的写法:
for x in &$@&; do
echo &parameter: '$x'&
执行的结果为:
$ ./myscript 'arg 1' arg2 arg3
parameter: 'arg 1'
parameter: 'arg2'
parameter: 'arg3'
上面正确的例子中,第一个参数'arg 1'在展开后依然是一个独立的单词,而不会被拆分成两个。
25. function foo()
这种写法不一定能够兼容所有 shell,兼容的写法是:
26. echo "~"
仅当~没有引号的时候发生,在上面的例子中,只会向标准输出打印~符号,而不是当前用户的家目录路径。
当用引号将路径参数引起来时,
如果要用引号将相对于家目录的路径引起来时,推荐使用 $HOME 而不是 ~, 假如 $HOME 目录是"/home/my photos",路径中包含空格。
下面是几组例子:
&~/dir with spaces& # expands to &~/dir with spaces&
~&/dir with spaces& # expands to &~/dir with spaces&
~/&dir with spaces& # expands to &/home/my photos/dir with spaces&
&$HOME/dir with spaces& # expands to &/home/my photos/dir with spaces&
27. local varname=$(command)
当在函数中声明局部变量时,作为一个独立的命令,这种奇特的行为有时候可能会导致困扰。比如,当你想要捕获的返回码时,你就不能这样做。local命令的返回码会覆盖它。
这种情况下,你只能分成两行写:
local varname
varname=$(command)
28. export foo=~/bar
export 与 local 命令一样,并不是赋值语句的一部分。因此,在有些 Shell 下(比如Bash),export foo=~/bar会展开,但是有些(比如 Dash)却不行。
下面是两种比较健壮的写法:
foo=~/ export foo
export foo=&$HOME/bar&
29. sed 's/$foo/good bye/'
单引号内部不会展开 $foo变量,在这里可以换成双引号:
foo=&hello&; sed &s/$foo/good bye/&
但是要注意,如果你使用了双引号,就需要考虑更多转义的事情,具体可以看这一页。.
30. tr [A-Z] [a-z]
这里至少有三个问题。第一个问题是, [A-Z] 和 [a-z] 会被 shell 认为是通配符。如果在当前目录下没用文件名为单个字母的文件,这个命令似乎能正确执行,否则会错误地执行,也许你会在周末耗费许多小时来修复这个问题。
第二个问题是,这不是 tr 命令正确的写法,实际上,上面的命令会把[转换成[,将任意大写字符转换成对应的小写字符,将]转换成],所以你根本不需要加上括号,这样第一个问题就可以解决了。
第三个问题是,上面的命令执行结果依赖于当前的 ,A-Z 或者 a-z 不一定会代表26个 ASCII 字母。实际上,在一些语言环境下,z 位于字母表的中间位置。这个问题的解法,取决于你希望发生的行为是哪一种。
如果你仅希望改变26个英文字母的大小写(强制 locale为 C):
LC_COLLATE=C tr A-Z a-z
如果你希望根据实际的语言环境来转换:
tr '[:upper:]' '[:lower:]'
31. ps ax | grep gedit
这里的根本问题是正在运行的进程名称,本质上是不可靠的。可能会有多个合法的gedit进程,也有可能是别的东西伪装成gedit进程(改变执行命令名称是一件简单的事情 ),更多细节可以看这一篇文章。
执行以上命令,往往会在结果中包含 grep 进程:
# ps ax | grep gedit
6:23 gedit
32118 pts/0
0:00 grep gedit
这个时候,需要过滤多余的结果:
# ps ax | grep -v grep | grep gedit
上面的写法比较丑陋,另外一种方法是:
# ps ax | grep [g]edit
32. printf "$foo"
如果$foo 变量的值中包括\或者%符号,上面命令的执行结果可能会出乎你的意料之外。
下面是正确的写法:
printf %s &$foo&
printf '%s\n' &$foo&
33. for i in {1..$n}
Bash的会优先,所以这时大括号{}表达式里面看到的是文字上的$n(没有展开)。$n 不是一个数值,所以这里的大括号{}并不会展开成数字列表。可见,这导致很难使用大括号来展开大小只能在运行时才知道的列表。
可以用下面的方法:
for ((i=1; i&=n; i++)); do
注:之前我也有写过一篇文章来介绍这个问题:。
34. if [[ $foo = $bar ]]
在[[内部,当=号右边的值没有用引号引起来,bash 会将它当作模式来匹配,而不是一个简单的字符串。所以,在上面的例子中 ,如果 bar 的值是一个*号,执行的结果永远是 true。
所以,如果你想检查两侧的字符串是否相同,等号右侧的值一定要用引号引起来。
if [[ $foo = &$bar& ]]
如果你确实要执行模式匹配,聪明的做法是取一个更加有意义的变量名(例如$patt),或者加上注释说明。
35. if [[ $foo =~ 'some RE' ]]
同上,如果=~号右侧的值加上引号,它会散失特殊的正则表达式含义,而变成一个普通的字符串。
如果你想使用一个长的或者复杂的正则表达式,避免大量的反斜杠转义,建议把它放在一个变量中:
re='some RE'
if [[ $foo =~ $re ]]
36. [ -n $foo ] or [ -z $foo ]
这个例子中,$foo 没有用引号引起来,当$foo包含空格或者$foo为空时都会出问题:
$ foo=&some word& && [ -n $foo ] && echo yes
-bash: [: some: binary operator expected
$ foo=&& && [ -n $foo ] && echo yes
正确的写法是:
[ -n &$foo& ]
[ -z &$foo& ]
[ -n &$(some command with a &$file& in it)& ]
[[ -n $foo ]]
[[ -z $foo ]]
37. [[ -e "$broken_symlink" ]] returns 1 even though $broken_symlink exists
这里-e 选项是看文件是否存在,当紧跟的文件是一个软链接时,它不看软链接是否存在,而是看实际指向的文件是否存在。所以当软链接损坏时,即实际指向的文件被删除后,-e 的结果返回1。
所以如果你确实要判断后面的文件是否存在,正确的写法是:
[[ -e &$broken_symlink& || -L &$broken_symlink& ]]
38. ed file &&&"g/d\{0,3\}/s//e/g" fails
ed 命令使用的正则语法,不支持0次出现次数,下面的就可以正常工作:
ed file &&&&g/d\{1,3\}/s//e/g&
略过,现在很少会有人用 ed 命令吧。
39. expr sub-string fails for "match"
下面的例子多数情况下运行不会有问题:
word=abcde
expr &$word& : &.\(.*\)&
但是当 $work 不巧刚好是 match 时,就有可能出错了(MAC OSX 下的 expr 命令不支持 match,所以依然能正常工作):
word=match
expr &$word& : &.\(.*\)&
原因是 match 是 expr 命令里面的一个特殊关键字,针对 GNU系统,解决方法是在前面加一个'+':
word=match
expr + &$word& : &.\(.*\)&
'+'号可以让 expr 命令忽略后续 token 的特殊含义。
另外一个建议是,不要再使用 expr 命令了,expr 能做的事情都可以用 Bash 原生支持的参数展开()或者字符串展开(Substring Expansion)来完成。并且相同情况下,内置的功能肯定比外部命令的效率要高。
上面的例子,目的是为了删除单词中的首字符,可以这样做:
$ word=match
$ echo &${word#?}&
$ echo &${word:1}&
40. On UTF-8 and Byte-Order Marks (BOM)
多数情况下,UNIX 下 UTF-8 类型的文本不需要使用 BOM,文本的编码是根据当前语言环境,MIME类型或者其它文件元数据信息确定的。人为阅读时,不会因为在文件开始处加 BOM 标记而腚影响,但是当文件要被脚本解释执行时,BOM 标记会像 MS-DOS 下的换行符(^M)一样奇怪。
41. content=$(&file)
这里没有什么错误,不过你要知道命令替换会删除结尾多余的换行符。
略过,原文给的优化方法需要 Bash 4.2+ 以上的版本,手头没有这样的环境。
42. somecmd 2&&1 &&logfile
这是一个很常见的错误,显然你本来是想将标准输出与标准错误输出都重定向到文件logfile 中,但是你会惊讶地发现,标准错误依然输出到屏幕中。
这种行为的原因是,在命令执行之前解析,并且是从左往右解析。上面的命令可以翻译成,将标准错误输出重定向到标准输出(此刻是终端),然后将标准输出重定向到文件 logfile 中。所以,到最后,标准错误并没有重定向到文件中,而是依然输出到终端:
somecmd &&logfile 2&&1
更加详细的说明见。
43. (( ! $? )) || die
只有需要捕获上一个命令的执行结果进,才需要记录$?的值,否则如果你只需要检查上一个命令是否执行成功,直接检测命令:
或者使用 case 语句来检测多个或能的返回码:
case $status in
echo success &&2
echo 'Must supply a parameter, exiting.' &&2
echo 'Unknown error, exiting.' &&2
exit $status
31. ps ax | grep gedit
请问为什么ps aux | grep [g]edit 就可以呢?这个有区别么?
为作者带来更多读者;为读者筛选优质内容;专注IT互联网。
最新评论(期待您也参与评论)
汇集优质的Python技术文章和资源。人生苦短,我用Python!
JavaScript, CSS, HTML5 这里有前端的技术干货!
关注安卓移动开发业界动态,分享技术文章和优秀工具资源。
关注iOS移动开发业界动态,分享技术文章和优秀工具资源。
为作者带来更多读者;为读者筛选优质内容;专注IT互联网。
由数百名译者组成,立志翻译传播优秀的外文技术干货。
一个专门为IT单身男女服务的征婚传播平台。
收录优秀的工具资源,覆盖开发、设计、产品和管理等。
关于伯乐在线博客
在这个信息爆炸的时代,人们已然被大量、快速并且简短的信息所包围。然而,我们相信:过多“快餐”式的阅读只会令人“虚胖”,缺乏实质的内涵。伯乐在线博客团队正试图以我们微薄的力量,把优秀的原创/译文分享给读者,做一个小而精的精选博客,为“快餐”添加一些“营养”元素。
欢迎关注更多频道
– 分享和发现有价值的内容与观点
– 为IT单身男女服务的征婚传播平台
– 优秀的工具资源导航
– 翻译传播优秀的外文文章
– 国内外的精选博客文章
– JavaScript, HTML5, CSS
– 专注Android技术分享
– 专注iOS技术分享
– 专注Java技术分享
– 专注Python技术分享
(加好友请注明来意)
网站使用问题
请在询问或者反馈
& 2015 伯乐在线
赞助云主机, 赞助云存储}

我要回帖

更多关于 c语言if函数 的文章

更多推荐

版权声明:文章内容来源于网络,版权归原作者所有,如有侵权请点击这里与我们联系,我们将及时删除。

点击添加站长微信