2

我已经浪费时间试图解决这个问题,但到目前为止还没有运气......我已经尝试通过 StackOverflow 查看相同的问题,但主要与人们使用的 IDE 有关,例如 VS 或 Eclipse。

我正在为斯坦福读者的 C++ 课程做一些示例,但代码无法正常工作。我试图弄清楚如何使用外部库,但总是出错。我可能没有使用正确的命令,但是我不知道要使用哪个命令。

我正在使用 Cygwin,它是做 C++ 练习的终端。我将所有文件都放在同一个文件夹中。我正在使用 Windows 7,但这不应该是最大的问题。

作为一个例子,这个错误很好地展示了我在编写 g++ Craps.cpp 时得到的结果:

$ g++ 废话.cpp

/tmp/ccdTdi0t.o:Craps.cpp:(.text+0x1cf): 对“randomInteger(int, int)”的未定义引用

/tmp/ccdTdi0t.o:Craps.cpp:(.text+0x1e6): undefined reference to `randomInteger(int, int)'

/usr/lib/gcc/i686-pc-cygwin/4.5.3/../../../../i686-pc-cygwin/bin/ld: /tmp/ccdTdi0t.o: 坏

在“.ctors”部分中重新定位地址 0x0

collect2: ld 返回 1 个退出状态

我这次运行的示例只是带有外部库的示例之一,如果我错了另一个,它会给我同样的错误。它不会找到图书馆。

这是我的主要也称为 Craps.cpp:

#include <iostream>
#include "random.h"
using namespace std;

bool tryToMakePoint(int point);
int rollTwoDice();
int main() {
   cout << "This program plays a game of craps." << endl;
   int point = rollTwoDice();
   switch (point) {
    case 7: case 11:
      cout << "That's a natural. You win." << endl;
      break;
    case 2: case 3: case 12:
      cout << "That's craps. You lose" << endl;
      break;
    default:
      cout << "Your point is " << point << "." << endl;
      if (tryToMakePoint(point)) {
         cout << "You made your point. You win." << endl;
         } else {
            cout << "You rolled a seven. You lose." << endl;
         }
   }
   return 0;
}


bool tryToMakePoint(int point) {
   while (true) {
      int total = rollTwoDice();
      if (total == point) return true;
      if (total == 7) return false;
   }
}


int rollTwoDice() {
   cout << "Rolling the dice . . . " << endl;
   int d1 = randomInteger(1, 6);
   int d2 = randomInteger(1, 6);
   int total = d1 + d2;
   cout << "You rolled " << d1 << " and " << d2 
        << " - that's " << total << endl;
   return total;
}

我的随机数.cpp:

#include <cstdlib>
#include <cmath>
#include <ctime>
#include "random.h"
using namespace std;

void initRandomSeed();

int randomInteger(int low, int high) {
   initRandomSeed();
   double d = rand() / (double(RAND_MAX) + 1);
   double s = d * (double(high) - low + 1);
   return int(floor(low + s ));
}

double randomReal(double low, double high) {
   initRandomSeed();
   double d = rand() / (double(RAND_MAX) + 1);
   double s = d * (high - low);
   return low + s;
}

bool randomChance(double p) {
   initRandomSeed();
   return randomReal(0, 1) < p;
   
}

void setRandomSeed(int seed) {
   initRandomSeed();
   srand(seed);
}

void initRandomSeed() {
   static bool initialized = false;
   if (!initialized) {
      srand(int(time(NULL)));
      initialized = true;
   }
}

最后是我的 random.h:

#ifndef _random_h
#define _random_h

int randomInteger(int low, int high);

double randomReal(double low, double high);

bool randomChance(double p);

void setRandomSeed(int seed);

#endif

我希望有一个人可以帮助我。如果这是错误的 Cygwin 命令,那么如果我能看到我应该写什么就太好了。

编辑:刚刚发现我什至无法正确写下书中的示例。现在已修复,应该没有错误......我非常希望如此。对于那个很抱歉。

4

2 回答 2

3

很快,您应该random.cpp在命令行中添加(或者,如果您已经编译它,则添加一个目标文件或编译代码所在的库):

g++ Craps.cpp random.cpp

你面临的问题是命令行说 Craps.cpp 中的代码应该被编译然后链接到一个可执行文件中。虽然有外部函数的前向声明来编译文件就足够了,但您应该将这些函数的代码提供给链接器,以便它能够创建可执行文件。

至于库,您通常会指定需要与-lGCC 选项一起使用的库。并且您可能需要(使用-L)指定从中获取库的路径,即使它们都在当前目录中。例如,假设您有一个名为librandom.aorlibrandom.so在当前目录中的库:

g++ Craps.cpp -L. -l random

对于外部库,可能需要指定其他目录,以便链接器知道在哪里可以找到它需要的库。

于 2012-11-14T20:03:36.767 回答
1

其他答案有简单的命令来编译它,但这里有一个非常粗略和简单的 Makefile 示例,让你开始走这条路。创建一个以Makefile这些内容命名的文件,其中<tab>必须是一个实际的制表符,而不是一些空格:

Craps.exe : Craps.o random.o
<tab>g++ -o $@ $^

Craps.o random.o : random.h

然后运行

make Craps.exe

(可能gmakemingw32-make代替 plain make)。

现在 Make 内置了一些关于文件类型的魔法,所以当它看到Craps.exe需要这两个.o文件时,它会开始寻找可以制作它们的文件,它会找到 .cpp 文件,并知道如何将它们编译成 .o 文件. 现在它并没有隐含地知道如何制作.exe文件.o,所以这就是我们有第二行的原因。There$@表示规则的目标(Craps.exe此处),而$^表示 . 之后列出的所有文件:

规则的Craps.o random.o:random.h意思是,如果random.h改变了,Craps.o和random.o需要重新重新创建(这些.h文件依赖可以自动生成,但只要你只有几个文件,写它们用手可以)。

如果需要添加更多的源文件,只需在第一行添加.o后缀即可,make会将它们包含在编译中。另请注意 make 将如何只编译更改的文件...然后当您获得更多 .h 文件时,您将需要添加 .h 文件依赖项。

于 2012-11-14T20:42:02.587 回答