Having spent the last couple of days experimenting with C++ operator[] methods, I’ve come across an anomaly that I can’t easily explain. The code below implements a simple test string class, which allows the user to access single characters via its subscript “operator” method. As I would like to differentiate between lvalue and rvalue subscript contexts, the “operator” method returns an instance of the custom “CReference” class as opposed to a standard C++ character reference. Whilst the “CReference” “operator char()” and “operator=(char)” methods appear to handle each rvalue and lvalue context respectively, g++ refuses to compile without the presence of an additional “operator=(CReference&)” method as documented below. Whilst the addition of this method appears to placate some kind of compile-time dependency, it is never actually invoked at run-time during the execution of the program.
As someone who thought they had acquired a fundamental understanding of C++ intricacies, this project has certainly proved to be a humbling experience. If anyone could see their way to enlightening me as to what’s going on here, I would be eternally grateful. In the meantime, I’m going to have to pull out the C++ books in order to reconcile the void** between what I know and what I think know.
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
// Class for holding a reference to type char, retuned by the "operator[](int)"
// method of class "TestString". Independent methods permit differentiation
// between lvalue and rvalue contexts.
class CReference
{
private:
char& m_characterReference;
public:
// Construct CReference from char&
CReference(char& m_initialiser)
: m_characterReference(m_initialiser) {}
// Invoked when object is referenced in a rvalue char context.
operator char()
{
return m_characterReference;
}
// Invoked when object is referenced in a lvalue char= context.
char operator=(char c)
{
m_characterReference = c;
return c;
}
// NEVER INVOKED, but will not compile without! WHY???
void operator=(CReference &p_assignator){}
};
// Simple string class which permits the manipulation of single characters
// via its "operator[](int)" method.
class TestString
{
private:
char m_content[23];
public:
// Construct string with test content.
TestString()
{
strcpy(m_content, "This is a test object.");
}
// Return pointer to content.
operator const char*()
{
m_content[22] = 0;
return m_content;
}
// Return reference to indexed character.
CReference operator[](int index)
{
return m_content[index];
}
};
int main(int argc, char *argv[])
{
TestString s1;
// Test both lvalue and rvalue subscript access.
s1[0] = s1[1];
// Print test string.
printf("%s\n", (const char*)s1);
return 0;
}
The line
s1[0] = s1[1];causes the compiler to generate an implicit copy assignment operator for CReference if you didn’t declare one yourself. This causes an error because your class has a reference member, which can’t be copied.If you added an assignment operator that takes a parameter of type
const CReference&, it would get called by the assignment.In your code, you declared a copy assignment operator of type
void operator=(CReference &p_assignator). This can’t be called because the righthand side of the assignment is a temporary object, which can’t be bound to a non-const reference. However, the act of declaring this operator causes the compiler not to try to define an implicit copy assignment operator, and therefore avoids the previous compilation error. Since this operator can’t be called, the compiler goes for the other assignment operator that takes a parameter of typechar.