<< 使用继承和模板定义
<< Definition using inheritance and templates
我正在尝试使用模板在 C++ 中实现链表。不幸的是,当我尝试使用 ostream 时遇到未解决的外部错误。本站类似问题的解答说是头文件和.cpp文件不匹配,我觉得这里不是。任何人都可以帮我解决这个问题,我真的需要掌握这个。
LinkedList.h:
#ifndef LINKEDLIST_H
#define LINKEDLIST_H
#include <string>
#include <iostream>
template<class T> class LinkedList;
// Class representing a templated linkedlist element
template<class T> class LLElement {
friend class LinkedList<T>;
public:
// Constructor
LLElement(T* p_Data) : data(p_Data), next(NULL) {}
T* getData() const { return data; }
LLElement<T>* getNext() { return next; }
private:
LLElement<T>* next;
T* data;
};
// Class representing a templated linked list
template<class T> class LinkedList {
public:
LinkedList() : size(0), start(NULL) {}
LinkedList(const LinkedList&);
~LinkedList();
unsigned int getSize() const { return size; }
LLElement<T>* getStart() const { return start; }
//adds shallow copy of T to the LinkedList
void sortedAdd(T*);
//removes the first occurence of T from the LinkedList
void remove(T*);
//merges existing LinkedList with given LinkedList
void merge(const LinkedList<T>&);
//formatted print of LinkedList to given outputstream
void showLinkedList(std::ostream& p_Out) const;
friend std::ostream& operator<<(std::ostream& outputStream, const LinkedList<T>& sec);
private:
LLElement<T>* start;
unsigned int size;
};
//LinkedList copy constructor
//TODO
template<class T> LinkedList<T>::LinkedList(const LinkedList<T>& a){
this->size=a.getSize();
this->start=a.getStart();
}
// LinkedList destructor
// TODO
template<class T> LinkedList<T>::~LinkedList(){
LLElement<T>* a=this->getStart();
LLElement<T>* b=a->next;
while(a!=NULL){
delete a;
a=b;
b=a->next;
}
}
// sortedAdd
// adds shallow copy of T to the LinkedList
// TODO
template<class T> void LinkedList<T>::sortedAdd(T* add){
LLElement<T>* a=new LLElement<T>(add);
LLElement<T>* b=this->getStart();
while(!(*(b->data)<=*(a->getData()))){
b=b->next;
}
a->next=b->next;
b->next=a;
this->size++;
}
// remove
// removes the first occurence of T from the LinkedList
// TODO
template<class T> void LinkedList<T>::remove(T* ab){
LLElement<T>* a=this->getStart();
LLElement<T>* b=this->getStart();
while(a!=NULL){
if(*(a->data)==*(ab)){
if(b==a){
this->start=a->next;
delete a;
this->size--;
return;
}
b->next=a->next;
delete a;
this->size--;
return;
}
b=a;
a=b->getNext();
}
}
// TODO output operator
template<class T> std::ostream& operator<<(std::ostream& outputStream, const LinkedList<T>& sec){
sec.showLinkedList(outputStream);
return outputStream;
}
//merges existing LinkedList with given LinkedList
template<class T> void LinkedList<T>::merge(const LinkedList<T>& p_LL){
LLElement<T> *cursor = p_LL.start;
while (cursor != NULL){
sortedAdd(cursor->data);
cursor = cursor->next;
}
}
//formatted print of LinkedList to given outputstream
template<class T> void LinkedList<T>::showLinkedList(std::ostream& p_Out) const {
LLElement<T>* current = start;
int counter = 1;
p_Out << "LINKEDLIST OF SIZE " << size << std::endl;
while (current != NULL){
p_Out << "LL ELEMENT " << counter++ << " : ";
p_Out << *(current->getData()) << std::endl;
current = current->getNext();
}
}
#endif
错误:
Error 2 error LNK2019: unresolved external symbol "class
std::basic_ostream<char,struct std::char_traits<char> > & __cdecl
operator<<(class std::basic_ostream<char,struct std::char_traits<char> > &,class
LinkedList<class CMSContent> const &)" (??6@YAAAV?$basic_ostream@DU
$char_traits@D@std@@@std@@AAV01@ABV?$LinkedList@VCMSContent@@@@@Z) referenced in function "public: void __thiscall CMS::print(void)" (?print@CMS@@QAEXXZ) C:\Users\knudde\Documents\Visual Studio 2012\Projects\Ex1314\Ex1314\CMS.obj Ex1314
可能是 CMSContent 有问题?
Content.h:
#ifndef H_CONTENT
#define H_CONTENT
#include <map>
#include <set>
#include <iostream>
#include <string>
enum ACCESSRIGHT { owner, modify, read, none };
class CMSContent {
public:
CMSContent(std::string p_Id, std::string p_Username);
void addPermissionUser(ACCESSRIGHT p_Right, std::string p_Username);
ACCESSRIGHT getAccessright(std::string p_Username);
std::string getIdentifier(){ return identifier; };
virtual std::string serialise() const=0;
bool CMSContent::operator< (const CMSContent& param) const;
bool CMSContent::operator<= (const CMSContent& param) const;
bool CMSContent::operator== (const CMSContent& param) const;
friend std::ostream& operator<<(std::ostream& outputStream, CMSContent& secondOperand);
protected:
std::string identifier;
std::map<ACCESSRIGHT, std::set<std::string> > accessrightsByRank;
//map with key accessright and value set of usernames with given access right
private:
void setIdentifier(std::string p_Id){ identifier = p_Id; };
bool userHasHigherAccessRight(ACCESSRIGHT p_Right, std::string p_Username);
void removeLowerAccessRights(ACCESSRIGHT p_Right, std::string p_Username);
static int uniqueID;
};
class Project : public CMSContent {
public:
Project(std::string p_Name, std::string p_Description, std::string p_Username);
virtual std::string serialise() const;
std::string getProjectdescription(){ return projectdescription; };
void setProjectdescription(std::string p_Description){ projectdescription = p_Description; };
protected:
std::string projectdescription;
};
class Document : public CMSContent {
public:
Document(std::string p_Name, std::string p_Username, std::string p_URL);
virtual std::string serialise() const;
std::string getURL(){ return URL; };
void setURL(std::string p_Url){ URL = p_Url; };
protected:
std::string URL;
};
class HowTo : public CMSContent {
public:
HowTo(std::string p_Subject, std::string p_Howto_content, std::string p_Username);
virtual std::string serialise() const;
std::string getHowto_content() { return howto_content; }
void setHowto_content(std::string p_Howto_content) { howto_content = p_Howto_content; }
protected:
std::string howto_content;
};
#endif
Content.cpp:
#include "Content.h"
#include <sstream>
CMSContent::CMSContent(std::string p_Id, std::string p_Username) : identifier(p_Id){
accessrightsByRank[owner].insert(p_Username);
}
void CMSContent::addPermissionUser(ACCESSRIGHT right, std::string p_Username) {
if (!userHasHigherAccessRight(right, p_Username)){
removeLowerAccessRights(right, p_Username);
accessrightsByRank[right].insert(p_Username);
}
}
bool CMSContent::operator< (const CMSContent& param) const{
return (this->identifier.compare(param.identifier)<0);
}
bool CMSContent::operator<= (const CMSContent& param) const{
return (this->identifier.compare(param.identifier)<=0);
}
bool CMSContent::operator== (const CMSContent& param) const{
return (this->identifier.compare(param.identifier)==0);
}
std::ostream& operator<<(std::ostream& outputStream, CMSContent& cont){
outputStream<<cont.serialise();
outputStream<<"owner = ";
std::set<std::string>::iterator a,b;
a=cont.accessrightsByRank[owner].begin();
b=cont.accessrightsByRank[owner].end();
while(a!=b){
outputStream << (*a) <<std::endl;
}
outputStream<<"modify = ";
a=cont.accessrightsByRank[modify].begin();
b=cont.accessrightsByRank[modify].end();
while(a!=b){
outputStream << (*a) <<std::endl;
}
outputStream<<"read = ";
a=cont.accessrightsByRank[read].begin();
b=cont.accessrightsByRank[read].end();
while(a!=b){
outputStream << (*a) <<std::endl;
}
return outputStream;
}
bool CMSContent::userHasHigherAccessRight(ACCESSRIGHT right, std::string p_Username){
//TODO
if(right==owner){return false;}
std::map<ACCESSRIGHT, std::set<std::string> >::iterator beg, end;
std::set<std::string> ::iterator beg1, end1;
beg=accessrightsByRank.begin();
end=accessrightsByRank.find(right);
while(beg != end){
beg1=(*beg).second.begin();
end1=(*beg).second.end();
while(beg1!=end1){
if((*beg1).compare(p_Username)==0){
return true;
}
beg1++;
}
beg++;
}
return false;
}
void CMSContent::removeLowerAccessRights(ACCESSRIGHT right, std::string p_Username){
//TODO
std::map<ACCESSRIGHT, std::set<std::string> >::iterator beg, end;
std::set<std::string> ::iterator beg1, end1;
end=accessrightsByRank.end();
beg=accessrightsByRank.find(right)++;
while(beg != end){
beg1=(*beg).second.begin();
end1=(*beg).second.end();
while(beg1!=end1){
if((*beg1).compare(p_Username)==0){
(*beg).second.erase(beg1);
return;
}
beg1++;
}
beg++;
}
}
Project::Project(std::string p_Identifier, std::string p_Description, std::string p_Username) : CMSContent("P;"+p_Identifier, p_Username){
projectdescription = p_Description;
}
std::string Project::serialise() const {
std::stringstream output;
output << "Project name = " << identifier << std::endl;
output << "Project description = " << projectdescription << std::endl;
return output.str();
}
Document::Document(std::string p_Identifier, std::string p_Username, std::string p_URL) : CMSContent("D;"+p_Identifier, p_Username){
URL = p_URL;
}
std::string Document::serialise() const {
std::stringstream output;
output << "Document name = " << identifier << std::endl;
output << "Document URL = " << URL << std::endl;
return output.str();
}
HowTo::HowTo(std::string p_Identifier, std::string p_Howto_content, std::string p_Username) : CMSContent("H;"+p_Identifier, p_Username){
howto_content = p_Howto_content;
}
std::string HowTo::serialise() const{
std::stringstream output;
output << "Howto subject = " << identifier << std::endl;
output << "Howto content = " << howto_content << std::endl;
return output.str();
}
从 LinkedList class 定义中删除 friend std::ostream& operator<<(std::ostream& outputStream, const LinkedList<T>& sec);
。您在此 header 中已经有了此类运算符的模板版本,链接器告诉 non-templated 一个缺少实现。
LinkedList.h:
#ifndef LINKEDLIST_H
#define LINKEDLIST_H
#include <string>
#include <iostream>
template<class T> class LinkedList;
// Class representing a templated linkedlist element
template<class T> class LLElement {
friend class LinkedList<T>;
public:
// Constructor
LLElement(T* p_Data) : data(p_Data), next(NULL) {}
T* getData() const { return data; }
LLElement<T>* getNext() { return next; }
private:
LLElement<T>* next;
T* data;
};
// Class representing a templated linked list
template<class T> class LinkedList {
public:
LinkedList() : size(0), start(NULL) {}
LinkedList(const LinkedList&);
~LinkedList();
unsigned int getSize() const { return size; }
LLElement<T>* getStart() const { return start; }
//adds shallow copy of T to the LinkedList
void sortedAdd(T*);
//removes the first occurence of T from the LinkedList
void remove(T*);
//merges existing LinkedList with given LinkedList
void merge(const LinkedList<T>&);
//formatted print of LinkedList to given outputstream
void showLinkedList(std::ostream& p_Out) const;
//friend std::ostream& operator<<(std::ostream& outputStream, const LinkedList<T>& sec);
private:
LLElement<T>* start;
unsigned int size;
};
//LinkedList copy constructor
//TODO
template<class T> LinkedList<T>::LinkedList(const LinkedList<T>& a){
this->size=a.getSize();
this->start=a.getStart();
}
// LinkedList destructor
// TODO
template<class T> LinkedList<T>::~LinkedList(){
LLElement<T>* a=this->getStart();
LLElement<T>* b=a->next;
while(a!=NULL){
delete a;
a=b;
b=a->next;
}
}
// sortedAdd
// adds shallow copy of T to the LinkedList
// TODO
template<class T> void LinkedList<T>::sortedAdd(T* add){
LLElement<T>* a=new LLElement<T>(add);
LLElement<T>* b=this->getStart();
while(!(*(b->data)<=*(a->getData()))){
b=b->next;
}
a->next=b->next;
b->next=a;
this->size++;
}
// remove
// removes the first occurence of T from the LinkedList
// TODO
template<class T> void LinkedList<T>::remove(T* ab){
LLElement<T>* a=this->getStart();
LLElement<T>* b=this->getStart();
while(a!=NULL){
if(*(a->data)==*(ab)){
if(b==a){
this->start=a->next;
delete a;
this->size--;
return;
}
b->next=a->next;
delete a;
this->size--;
return;
}
b=a;
a=b->getNext();
}
}
// TODO output operator
template<class T> std::ostream& operator<<(std::ostream& outputStream, const LinkedList<T>& sec){
sec.showLinkedList(outputStream);
return outputStream;
}
//merges existing LinkedList with given LinkedList
template<class T> void LinkedList<T>::merge(const LinkedList<T>& p_LL){
LLElement<T> *cursor = p_LL.start;
while (cursor != NULL){
sortedAdd(cursor->data);
cursor = cursor->next;
}
}
//formatted print of LinkedList to given outputstream
template<class T> void LinkedList<T>::showLinkedList(std::ostream& p_Out) const {
LLElement<T>* current = start;
int counter = 1;
p_Out << "LINKEDLIST OF SIZE " << size << std::endl;
while (current != NULL){
p_Out << "LL ELEMENT " << counter++ << " : ";
p_Out << *(current->getData()) << std::endl;
current = current->getNext();
}
}
#endif
此外,关于 Content.h:
bool CMSContent::operator< (const CMSContent& param) const;
bool CMSContent::operator<= (const CMSContent& param) const;
bool CMSContent::operator== (const CMSContent& param) const;
应该是
bool operator< (const CMSContent& param) const;
bool operator<= (const CMSContent& param) const;
bool operator== (const CMSContent& param) const;
我正在尝试使用模板在 C++ 中实现链表。不幸的是,当我尝试使用 ostream 时遇到未解决的外部错误。本站类似问题的解答说是头文件和.cpp文件不匹配,我觉得这里不是。任何人都可以帮我解决这个问题,我真的需要掌握这个。
LinkedList.h:
#ifndef LINKEDLIST_H
#define LINKEDLIST_H
#include <string>
#include <iostream>
template<class T> class LinkedList;
// Class representing a templated linkedlist element
template<class T> class LLElement {
friend class LinkedList<T>;
public:
// Constructor
LLElement(T* p_Data) : data(p_Data), next(NULL) {}
T* getData() const { return data; }
LLElement<T>* getNext() { return next; }
private:
LLElement<T>* next;
T* data;
};
// Class representing a templated linked list
template<class T> class LinkedList {
public:
LinkedList() : size(0), start(NULL) {}
LinkedList(const LinkedList&);
~LinkedList();
unsigned int getSize() const { return size; }
LLElement<T>* getStart() const { return start; }
//adds shallow copy of T to the LinkedList
void sortedAdd(T*);
//removes the first occurence of T from the LinkedList
void remove(T*);
//merges existing LinkedList with given LinkedList
void merge(const LinkedList<T>&);
//formatted print of LinkedList to given outputstream
void showLinkedList(std::ostream& p_Out) const;
friend std::ostream& operator<<(std::ostream& outputStream, const LinkedList<T>& sec);
private:
LLElement<T>* start;
unsigned int size;
};
//LinkedList copy constructor
//TODO
template<class T> LinkedList<T>::LinkedList(const LinkedList<T>& a){
this->size=a.getSize();
this->start=a.getStart();
}
// LinkedList destructor
// TODO
template<class T> LinkedList<T>::~LinkedList(){
LLElement<T>* a=this->getStart();
LLElement<T>* b=a->next;
while(a!=NULL){
delete a;
a=b;
b=a->next;
}
}
// sortedAdd
// adds shallow copy of T to the LinkedList
// TODO
template<class T> void LinkedList<T>::sortedAdd(T* add){
LLElement<T>* a=new LLElement<T>(add);
LLElement<T>* b=this->getStart();
while(!(*(b->data)<=*(a->getData()))){
b=b->next;
}
a->next=b->next;
b->next=a;
this->size++;
}
// remove
// removes the first occurence of T from the LinkedList
// TODO
template<class T> void LinkedList<T>::remove(T* ab){
LLElement<T>* a=this->getStart();
LLElement<T>* b=this->getStart();
while(a!=NULL){
if(*(a->data)==*(ab)){
if(b==a){
this->start=a->next;
delete a;
this->size--;
return;
}
b->next=a->next;
delete a;
this->size--;
return;
}
b=a;
a=b->getNext();
}
}
// TODO output operator
template<class T> std::ostream& operator<<(std::ostream& outputStream, const LinkedList<T>& sec){
sec.showLinkedList(outputStream);
return outputStream;
}
//merges existing LinkedList with given LinkedList
template<class T> void LinkedList<T>::merge(const LinkedList<T>& p_LL){
LLElement<T> *cursor = p_LL.start;
while (cursor != NULL){
sortedAdd(cursor->data);
cursor = cursor->next;
}
}
//formatted print of LinkedList to given outputstream
template<class T> void LinkedList<T>::showLinkedList(std::ostream& p_Out) const {
LLElement<T>* current = start;
int counter = 1;
p_Out << "LINKEDLIST OF SIZE " << size << std::endl;
while (current != NULL){
p_Out << "LL ELEMENT " << counter++ << " : ";
p_Out << *(current->getData()) << std::endl;
current = current->getNext();
}
}
#endif
错误:
Error 2 error LNK2019: unresolved external symbol "class
std::basic_ostream<char,struct std::char_traits<char> > & __cdecl
operator<<(class std::basic_ostream<char,struct std::char_traits<char> > &,class
LinkedList<class CMSContent> const &)" (??6@YAAAV?$basic_ostream@DU
$char_traits@D@std@@@std@@AAV01@ABV?$LinkedList@VCMSContent@@@@@Z) referenced in function "public: void __thiscall CMS::print(void)" (?print@CMS@@QAEXXZ) C:\Users\knudde\Documents\Visual Studio 2012\Projects\Ex1314\Ex1314\CMS.obj Ex1314
可能是 CMSContent 有问题?
Content.h:
#ifndef H_CONTENT
#define H_CONTENT
#include <map>
#include <set>
#include <iostream>
#include <string>
enum ACCESSRIGHT { owner, modify, read, none };
class CMSContent {
public:
CMSContent(std::string p_Id, std::string p_Username);
void addPermissionUser(ACCESSRIGHT p_Right, std::string p_Username);
ACCESSRIGHT getAccessright(std::string p_Username);
std::string getIdentifier(){ return identifier; };
virtual std::string serialise() const=0;
bool CMSContent::operator< (const CMSContent& param) const;
bool CMSContent::operator<= (const CMSContent& param) const;
bool CMSContent::operator== (const CMSContent& param) const;
friend std::ostream& operator<<(std::ostream& outputStream, CMSContent& secondOperand);
protected:
std::string identifier;
std::map<ACCESSRIGHT, std::set<std::string> > accessrightsByRank;
//map with key accessright and value set of usernames with given access right
private:
void setIdentifier(std::string p_Id){ identifier = p_Id; };
bool userHasHigherAccessRight(ACCESSRIGHT p_Right, std::string p_Username);
void removeLowerAccessRights(ACCESSRIGHT p_Right, std::string p_Username);
static int uniqueID;
};
class Project : public CMSContent {
public:
Project(std::string p_Name, std::string p_Description, std::string p_Username);
virtual std::string serialise() const;
std::string getProjectdescription(){ return projectdescription; };
void setProjectdescription(std::string p_Description){ projectdescription = p_Description; };
protected:
std::string projectdescription;
};
class Document : public CMSContent {
public:
Document(std::string p_Name, std::string p_Username, std::string p_URL);
virtual std::string serialise() const;
std::string getURL(){ return URL; };
void setURL(std::string p_Url){ URL = p_Url; };
protected:
std::string URL;
};
class HowTo : public CMSContent {
public:
HowTo(std::string p_Subject, std::string p_Howto_content, std::string p_Username);
virtual std::string serialise() const;
std::string getHowto_content() { return howto_content; }
void setHowto_content(std::string p_Howto_content) { howto_content = p_Howto_content; }
protected:
std::string howto_content;
};
#endif
Content.cpp:
#include "Content.h"
#include <sstream>
CMSContent::CMSContent(std::string p_Id, std::string p_Username) : identifier(p_Id){
accessrightsByRank[owner].insert(p_Username);
}
void CMSContent::addPermissionUser(ACCESSRIGHT right, std::string p_Username) {
if (!userHasHigherAccessRight(right, p_Username)){
removeLowerAccessRights(right, p_Username);
accessrightsByRank[right].insert(p_Username);
}
}
bool CMSContent::operator< (const CMSContent& param) const{
return (this->identifier.compare(param.identifier)<0);
}
bool CMSContent::operator<= (const CMSContent& param) const{
return (this->identifier.compare(param.identifier)<=0);
}
bool CMSContent::operator== (const CMSContent& param) const{
return (this->identifier.compare(param.identifier)==0);
}
std::ostream& operator<<(std::ostream& outputStream, CMSContent& cont){
outputStream<<cont.serialise();
outputStream<<"owner = ";
std::set<std::string>::iterator a,b;
a=cont.accessrightsByRank[owner].begin();
b=cont.accessrightsByRank[owner].end();
while(a!=b){
outputStream << (*a) <<std::endl;
}
outputStream<<"modify = ";
a=cont.accessrightsByRank[modify].begin();
b=cont.accessrightsByRank[modify].end();
while(a!=b){
outputStream << (*a) <<std::endl;
}
outputStream<<"read = ";
a=cont.accessrightsByRank[read].begin();
b=cont.accessrightsByRank[read].end();
while(a!=b){
outputStream << (*a) <<std::endl;
}
return outputStream;
}
bool CMSContent::userHasHigherAccessRight(ACCESSRIGHT right, std::string p_Username){
//TODO
if(right==owner){return false;}
std::map<ACCESSRIGHT, std::set<std::string> >::iterator beg, end;
std::set<std::string> ::iterator beg1, end1;
beg=accessrightsByRank.begin();
end=accessrightsByRank.find(right);
while(beg != end){
beg1=(*beg).second.begin();
end1=(*beg).second.end();
while(beg1!=end1){
if((*beg1).compare(p_Username)==0){
return true;
}
beg1++;
}
beg++;
}
return false;
}
void CMSContent::removeLowerAccessRights(ACCESSRIGHT right, std::string p_Username){
//TODO
std::map<ACCESSRIGHT, std::set<std::string> >::iterator beg, end;
std::set<std::string> ::iterator beg1, end1;
end=accessrightsByRank.end();
beg=accessrightsByRank.find(right)++;
while(beg != end){
beg1=(*beg).second.begin();
end1=(*beg).second.end();
while(beg1!=end1){
if((*beg1).compare(p_Username)==0){
(*beg).second.erase(beg1);
return;
}
beg1++;
}
beg++;
}
}
Project::Project(std::string p_Identifier, std::string p_Description, std::string p_Username) : CMSContent("P;"+p_Identifier, p_Username){
projectdescription = p_Description;
}
std::string Project::serialise() const {
std::stringstream output;
output << "Project name = " << identifier << std::endl;
output << "Project description = " << projectdescription << std::endl;
return output.str();
}
Document::Document(std::string p_Identifier, std::string p_Username, std::string p_URL) : CMSContent("D;"+p_Identifier, p_Username){
URL = p_URL;
}
std::string Document::serialise() const {
std::stringstream output;
output << "Document name = " << identifier << std::endl;
output << "Document URL = " << URL << std::endl;
return output.str();
}
HowTo::HowTo(std::string p_Identifier, std::string p_Howto_content, std::string p_Username) : CMSContent("H;"+p_Identifier, p_Username){
howto_content = p_Howto_content;
}
std::string HowTo::serialise() const{
std::stringstream output;
output << "Howto subject = " << identifier << std::endl;
output << "Howto content = " << howto_content << std::endl;
return output.str();
}
从 LinkedList class 定义中删除 friend std::ostream& operator<<(std::ostream& outputStream, const LinkedList<T>& sec);
。您在此 header 中已经有了此类运算符的模板版本,链接器告诉 non-templated 一个缺少实现。
LinkedList.h:
#ifndef LINKEDLIST_H
#define LINKEDLIST_H
#include <string>
#include <iostream>
template<class T> class LinkedList;
// Class representing a templated linkedlist element
template<class T> class LLElement {
friend class LinkedList<T>;
public:
// Constructor
LLElement(T* p_Data) : data(p_Data), next(NULL) {}
T* getData() const { return data; }
LLElement<T>* getNext() { return next; }
private:
LLElement<T>* next;
T* data;
};
// Class representing a templated linked list
template<class T> class LinkedList {
public:
LinkedList() : size(0), start(NULL) {}
LinkedList(const LinkedList&);
~LinkedList();
unsigned int getSize() const { return size; }
LLElement<T>* getStart() const { return start; }
//adds shallow copy of T to the LinkedList
void sortedAdd(T*);
//removes the first occurence of T from the LinkedList
void remove(T*);
//merges existing LinkedList with given LinkedList
void merge(const LinkedList<T>&);
//formatted print of LinkedList to given outputstream
void showLinkedList(std::ostream& p_Out) const;
//friend std::ostream& operator<<(std::ostream& outputStream, const LinkedList<T>& sec);
private:
LLElement<T>* start;
unsigned int size;
};
//LinkedList copy constructor
//TODO
template<class T> LinkedList<T>::LinkedList(const LinkedList<T>& a){
this->size=a.getSize();
this->start=a.getStart();
}
// LinkedList destructor
// TODO
template<class T> LinkedList<T>::~LinkedList(){
LLElement<T>* a=this->getStart();
LLElement<T>* b=a->next;
while(a!=NULL){
delete a;
a=b;
b=a->next;
}
}
// sortedAdd
// adds shallow copy of T to the LinkedList
// TODO
template<class T> void LinkedList<T>::sortedAdd(T* add){
LLElement<T>* a=new LLElement<T>(add);
LLElement<T>* b=this->getStart();
while(!(*(b->data)<=*(a->getData()))){
b=b->next;
}
a->next=b->next;
b->next=a;
this->size++;
}
// remove
// removes the first occurence of T from the LinkedList
// TODO
template<class T> void LinkedList<T>::remove(T* ab){
LLElement<T>* a=this->getStart();
LLElement<T>* b=this->getStart();
while(a!=NULL){
if(*(a->data)==*(ab)){
if(b==a){
this->start=a->next;
delete a;
this->size--;
return;
}
b->next=a->next;
delete a;
this->size--;
return;
}
b=a;
a=b->getNext();
}
}
// TODO output operator
template<class T> std::ostream& operator<<(std::ostream& outputStream, const LinkedList<T>& sec){
sec.showLinkedList(outputStream);
return outputStream;
}
//merges existing LinkedList with given LinkedList
template<class T> void LinkedList<T>::merge(const LinkedList<T>& p_LL){
LLElement<T> *cursor = p_LL.start;
while (cursor != NULL){
sortedAdd(cursor->data);
cursor = cursor->next;
}
}
//formatted print of LinkedList to given outputstream
template<class T> void LinkedList<T>::showLinkedList(std::ostream& p_Out) const {
LLElement<T>* current = start;
int counter = 1;
p_Out << "LINKEDLIST OF SIZE " << size << std::endl;
while (current != NULL){
p_Out << "LL ELEMENT " << counter++ << " : ";
p_Out << *(current->getData()) << std::endl;
current = current->getNext();
}
}
#endif
此外,关于 Content.h:
bool CMSContent::operator< (const CMSContent& param) const;
bool CMSContent::operator<= (const CMSContent& param) const;
bool CMSContent::operator== (const CMSContent& param) const;
应该是
bool operator< (const CMSContent& param) const;
bool operator<= (const CMSContent& param) const;
bool operator== (const CMSContent& param) const;