今天上C++的第一堂实验课,老师为了了解一下我们之前学的C语言怎么样,出了一到程序题,让我们编一下。题目很简单就是给定一元二次方程ax^2+bx+c=0的三个系数,然后给出结果,并能够循环执行,主函数只负责输入参数,要至少有一个函数,本人很快就理清思路开始编写,很快写完了,并且编译了,下面的就是我的第一遍源代码(VC编译环境):
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 #include <stdio.h> #include <math.h> void charge(float a,float b,float c); //判断过程 void runorstop(); //是否继续运行 int run = 1; //继续运行的标志,1为继续运行,0为停止运行 void main() { while(run == 1) { float a,b,c; printf("请输入a\n"); scanf("%f",&a); printf("请输入b\n"); scanf("%f",&b); printf("请输入c\n"); scanf("%f",&c); charge(a,b,c);//进行运算 } } void runorstop() { printf("是否还要继续运行程序,“是”请输入“1”,“否”请输入“0”\n"); scanf("%d",&run); } void charge(float a,float b,float c) { float x1,x2; float d; //daite if(a == 0) { if(b == 0) { if(c == 0) { printf("方程有无穷个解!\n\n"); runorstop(); } else { printf("a,b同时为0,方程错误!"); runorstop(); } } else { x1 = -c/b; printf("方程的解为 x = %f",x1); runorstop(); } } else { d = b*b - 4*a*c; if(d>0) { x1 = (-b+sqrt(d))/(2*a); x2 = (-b-sqrt(d))/(2*a); printf("方程的两个根分别是:\n"); printf("x1 = %f , x2 = %f\n",x1,x2); runorstop(); } if(d=0)// 《《---这里就是错误的出现的地方 { x1=-b/(2*a); printf("方程有两个相同的根:%f\n\n",x1); runorstop(); } if(d<0) { printf("方程没有实数解!\n\n"); runorstop(); } } }
但是在运行的时候,发现输入完a,b,c三个数据后,程序不停的出现 【是否还要继续运行程序,“是”请输入“1”,“否”请输入“0”】 这个提示信息,看来可能是while循环导致的不停的循环,先注释掉,然后再运行,输入完三个数后什么也没显示,程序就结束了。
我前前后后看了好几遍也没找到问题,第一次猜测是数据输入有些问题,于是在第16行之后加了一行printf输出一下a,b,c三个值,编译运行,成功输出,证明输入正确。于是又猜测可能是数据没有被带入到函数中去,于是又在28行后面插入输出语句进行测试(这里注意,一定要把printf放到后面,不能放到27或者28行前面,因为27,28行是声明的局部变量,局部变量的声明必须要放在这个函数的头部,这是老师说的)。测试结果正常,证明数据已经能进入函数。由于我输入测试的数据,应该进入68行的那个花括号,所以就在那附近检查,终于发现了问题,原来是一时打字疏忽,把62行的判断相等双等号打成了赋值单等号。这样一来,就成了把0赋值给d,而if(0)肯定不执行花括号里的内容,所以程序就跳过了所有的if,结果就是什么都不会显示。于是我想到了,如果当时使用了else的话,那么问题很快就能找到,因为一个判断不正确,他会跳转到else的花括号,这样就等于所以了检查范围。既然找到了问题,那么把之前注释掉的while循环改过来,然后把=加上,编译运行成功了。
附:由于机房使用的是VC,而我现在写这篇文章的时候是在我的本本的ubuntu系统下,于是又出现了点小插曲,就是sqar()函数在编译时,提示错误信息:
1 2 3 4 /tmp/ccszYPUG.o: In function `charge': 1.c:(.text+0x1a6): undefined reference to `sqrt' 1.c:(.text+0x1cc): undefined reference to `sqrt' collect2: ld returned 1 exit status
我在网上搜索了一下,找到了下面的这个资料,先收了,以备后用:
1 2 3 4 5 6 7 8 9 10 -l参数和-L参数 -l参数就是用来指定程序要链接的库,-l参数紧接着就是库名,那么库名跟真正的库文件名有什么关系呢?就拿数学库来说,他的库名是m,他的库文件名是libm.so,很容易看出,把库文件名的头lib和尾.so去掉就是库名了 好了现在我们知道怎么得到库名,当我们自已要用到一个第三方提供的库名字libtest.so,那么我们只要把libtest.so拷贝到/usr/lib里,编译时加上-ltest参数,我们就能用上libtest.so库了(当然要用libtest.so库里的函数,我们还需要与libtest.so配套的头文件) 放在/lib和/usr/lib和/usr/local/lib里的库直接用-l参数就能链接了,但如果库文件没放在这三个目录里,而是放在其他目录里,这时我们只用-l参数的话,链接还是会出错,出错信息大概是:“/usr/bin/ld: cannot find -lxxx”,也就是链接程序ld在那3个目录里找不到libxxx.so,这时另外一个参数-L就派上用场了,比如常用的X11的库,它在/usr/X11R6/lib目录下,我们编译时就要用-L/usr/X11R6/lib -lX11参数,-L参数跟着的是库文件所在的目录名。再比如我们把libtest.so放在/aaa/bbb/ccc目录下,那链接参数就是-L/aaa/bbb/ccc -ltest 另外,大部分libxxxx.so只是一个链接,以RH9为例,比如libm.so它链接到/lib/libm.so.x,/lib/libm.so.6又链接到/lib/libm-2.3.2.so, 如果没有这样的链接,还是会出错,因为ld只会找libxxxx.so,所以如果你要用到xxxx库,而只有libxxxx.so.x或者libxxxx-x.x.x.so,做一个链接就可以了ln -s libxxxx-x.x.x.so libxxxx.so 手工来写链接参数总是很麻烦的,还好很多库开发包提供了生成链接参数的程序,名字一般叫xxxx-config,一般放在/usr/bin目录下,比如 gtk1.2的链接参数生成程序是gtk-config,执行gtk-config --libs就能得到以下输出"-L/usr/lib -L/usr/X11R6/lib -lgtk -lgdk -rdynamic -lgmodule -lglib -ldl -lXi -lXext -lX11 -lm",这就是编译一个gtk1.2程序所需的gtk链接参数,xxx-config除了--libs参数外还有一个参数是--cflags用来生成头文件包含目录的,也就是-I参数,在下面我们将会讲到。你可以试试执行gtk-config --libs --cflags,看看输出结果 现在的问题就是怎样用这些输出结果了,最笨的方法就是复制粘贴或者照抄,聪明的办法是在编译命令行里加入这个`xxxx-config --libs --cflags`,比如编译一个gtk程序:gcc gtktest.c `gtk-config --libs --cflags`这样就差不多了。注意`不是单引号,而是1键左边那个键。
最后还是找到了,需要加-lm参数,编译指令如下:
这样就编译生成名字为a的程序了,但是运行的时候全是乱码,这个问题好解决,因为在VC下编写的源代码文件都是GBK格式的,只需要在ubuntu下面用leafpad打开源文件并且另存为UTF-8再编译运行就可以了。