Получение ошибок компиляции с помощью примера кода подсчитанного указателя Copliens 1994 - PullRequest
2 голосов
/ 06 июля 2011

Хорошо, я читаю книгу Copliens C ++ Idioms и пытаюсь запустить примеры из дескриптора / тела в этой книге. После ввода кода я получаю ошибки компиляции:

Вот код классов String и StringRep.

#ifndef _STRINGREP_H_
#define _STRINGREP_H_
#include <stdio.h>
#include <string.h>

class String;

class StringRep {

    friend class String;

public:
    StringRep() {*(rep = new char[1])='\0';}
    StringRep(const StringRep& s) {
        ::strcpy(rep=new char[::strlen(s.rep)+1], s.rep);
    }
    ~StringRep() { delete [] rep;}
    StringRep(const char* s) {
        ::strcpy(rep=new char[::strlen(s)+1], s);
    }
    String operator+(const String& s) const {
        char *buf = new char[s->length() + length() + 1];
        ::strcpy(buf, rep);
        ::strcat (buf, s->rep);
        String retval(&buf);
        return retval;
    }
    int length() const { return ::strlen(rep); }
    void print() const {::printf("%s\n", rep); }

private:
    StringRep(char ** const r) {
        rep = *r; 
        *r = 0;
        count = 1;
    };
    char *rep;
    int count;
};

#endif

#ifndef _STRING_H_
#define _STRING_H_

#include <stdio.h>
#include <string.h>
#include "StringRep.h"

class String {

    friend class StringRep;

public:
    String operator+(const String& s) const {return *p + s;}
    StringRep* operator->() const {return p;}
    String() {
        (p = new StringRep())->count = 1;
    }
    String (const String &s) { (p=s.p)->count++;}
    String(const char* s) {
        (p = new StringRep(s))->count = 1;
    }
    String operator=(const String& s) {
        if (--p->count <=0) delete p;
        (p = s.p)->count++;
        return *this;
    }
    ~String() { if (--p->count <= 0) delete p;; }

private:
    String(char **r) {
        p = new StringRep(r);
    }
    StringRep *p;
};

#endif

и main.cc

#include <stdio.h>
#include <string.h>
#include "StringRep.h"
#include "String.h"

int main() {

    String a("abcd"), b("efgh");
    printf("a is "); a->print();
    printf("b is "); b->print();
    printf("concat of a+b is "); (a+b)->print();
    return 0;
}

Ошибки компиляции;

GNU C++ version 4.1.2 20080704 (Red Hat 4.1.2-44) (x86_64-redhat-linux)
        compiled by GNU C version 4.1.2 20080704 (Red Hat 4.1.2-44).
GGC heuristics: --param ggc-min-expand=100 --param ggc-min-heapsize=131072
Compiler executable checksum: 2d02d8750f9b337bb19a7dd5b4e2167e
StringRep.h: In member function 'String StringRep::operator+(const String&) const':
StringRep.h:21: error: return type 'struct String' is incomplete
StringRep.h:22: error: base operand of '->' has non-pointer type 'const String'
StringRep.h:24: error: base operand of '->' has non-pointer type 'const String'
StringRep.h:25: error: variable 'String retval' has initializer but incomplete type
String.h: In member function 'String String::operator+(const String&) const':
String.h:13: error: conversion from 'void' to non-scalar type 'String' requested

Я полагаю, что не могу использовать класс String, пока он не будет полностью определен. Изменение подпись функции

String& operator+(const String& s) const {...

решает первую ошибку, но вызывает ту же ошибку, чтобы показать, где я создаю новый объект String

String retval(&buf);

Я понимаю, что моя книга - это перепечатка 1994 года, которую я взял. Так может кто-нибудь либо указать мне более новый код (если стиль кодирования C ++ изменился), либо указать, как это исправить?

Спасибо

Ответы [ 3 ]

2 голосов
/ 06 июля 2011

Вы получили циклическую ссылку, так как StringRep необходимо знать полное определение String, чтобы построить его в operator+.Я советую не помещать в заголовочные файлы все, а только объявления 1005 * функций-членов и помещать определения 1007 * в файл .cpp (или .cc).Это должно исправить это.Это также, как код должен быть разделен, если сам класс и / или функции не являются шаблоном.

0 голосов
/ 07 июля 2011

Вы не можете включить файл заголовка String в файл заголовка StringRep, если вы также включили файл заголовка StringRep в файл заголовка String.Это вызывает циклическое включение и предотвращает компиляцию вашего кода.

Думайте об этом как:

StringRep.h loads String.h
  which loads StringRep.h
    which loads String.h
      which loads StringRep.h
        which loads ...

Вы можете избежать этого, переместив объявления функций в файлы .cpp и просто переместив их вперед.объявив класс String внутри StringRep.h (что вы уже делаете!).На самом деле он не будет пытаться разрешить ссылку на класс String до тех пор, пока не скомпилирует код, и к тому времени проблема кругового включения будет устранена.

0 голосов
/ 07 июля 2011

Вы должны разделить определение и декларацию всего.Если вы действительно хотите поместить их в заголовочный файл, вы можете сделать это следующим образом:

<forward declarations, required includes>
<class itself, no functions defined>
<includes for forward declarations that are needed for function bodies>
<function bodies>

Это всегда будет работать.

...