[comp.lang.c++] operator[][] ?

neath@solar-1.stars.flab.Fujitsu.JUNET (05/15/89)

I  am trying to  determine what support there  is in C++  for multi-dimensional
arrays within a class. For example, the following class  definition attempts to
implement an array of pointers  to vectors.   Now, I  know that  I can overload
operator[],  but  what I would really  like to be able  to  do  is overload the
ficticious  operator[][]. How can  I achieve  this  functionality? Am I missing
something very obvious?

        typedef int Type
        
        class C {
        private:
          int r, c;
          Type** data;
        public:
          X (int row, int col) {
            this->r = row; this->c = col;
            this->data = new Type* [row];
            for (int i = 0; i < col; i++)
              this->data[i] = new Type [col];
          }
        
          ~X () {
            for (int i = 0; i < r; i++)
              delete this->data[i];
            delete this->data;
          }
        
          inline Type* operator[] (int row) {
            return this->data[row];
          }
        
          inline Type& operator[][] (int row, int col) {
            return this->data[row][col];
          }
        };
        

Regards
Martin Neath <neath@dsg.ti.com>
------------------------------------------------------------------------
 DISCLAIMER: As always, the opinions expressed above are strictly my own
 and do not reflect those of my employer, Texas Instruments.
------------------------------------------------------------------------

daniel@saturn.ucsc.edu (Daniel Edelson) (05/17/89)

In article <NEATH.89May15154230@solar-1.stars.flab.Fujitsu.JUNET> neath@solar-1.stars.flab.Fujitsu.JUNET writes:

>I am trying to determine what support there is in C++ for multi-dimensional
>arrays within a class. For example, the following class definition attempts to
>implement an array of pointers to vectors. Now, I know that I can overload
>operator[], but what I would really like to be able to do is overload the
>ficticious operator[][]. How can I achieve this functionality? Am I missing
>something very obvious?

>Martin Neath <neath@dsg.ti.com>

You need the type of A to be different from the type of A[i]
so that the [] operator applied to expressions of each type 
works correctly. 

Example: 

	class Element { /* stuff */ };

	class Vector {
	private:
		Element * col;		// an array of elements
		...
	public:
		Element & operator[](int index2);
		...
	};

	class Array{
	private:
		Vector * rows;		// an array of vectors
		...
	public:
		Vector & operator[](int index1);
		...
	};

Let A be of type Array, A[i] is of type Vector and
A[i][j] is of type Element. The two instances of the 
subscripting operator in the expression A[i][j] are not 
the same implementation, but are both [] operators.

daniel edelson
daniel@saturn.ucsc.edu

heimir@rhi.hi.is (Heimir Thor Sverrisson) (06/20/91)

Hi fellow Netlanders,

is there anyone out there that can help me understand why the following
code works!   I'm trying to build a class that contains integer data
that should be organized as a two-dimensional array (of fixed dimensions).

I was trying to implement an operator that would allow me to use code like:
    Status st;
    st[TOTAL][COD]=123;
where Status is the name of my class and the [][] notation would allow me
to manipulate that element in my private array.  On top of the well known
problem of not being able to detect if I'm being used as an l- or r-value
I also ran into trouble defining the operator(s) involved to do this.

What I ended up with were two operator[]'s and the code did actually work
in g++ 1.39.  The implementation is :

    int* Status::operator[](int index)
    {
	return stat[index];		// private: int stat[i][j]
    }
    int& Status::operator[](int* iarr)
    {
	return *iarr;
    }

And I'm asking you why does this work?
If it is not a compiler bug/feature, how can I access the latter index (COD)
from within the second member function above (i.e. to test it's range)?

Following is the whole code that I used to test this:
---------------------------------------------------------------------------
// Status.h
// 	Declare the Status-class.
//
#ifndef Status_h
#define Status_h

#define S_LINES 13
#define S_COL 10

class Status {
public:
    Status(int num);
    int* operator[](int index);
    int& operator[](int *iarr);
    ~Status();
private:
    stat[S_LINES][S_COL];
    int recalc;
};
#endif
---------------------------------------------------------------------------
// Status.cc
//	Implementation of the Status-class.
//
#include "Status.h"

Status::Status(int num)
{
    for(int i=0; i < S_LINES; i++)
	for(int j=0; j < S_COL; j++)
	    stat[i][j] = i*S_COL+j;	// Set to zero!!
}

int* Status::operator[](int index)
{
    return stat[index];
}

int& Status::operator[](int* iarr)
{
    return *iarr;
}
---------------------------------------------------------------------------
// tStatus.cc
//
// Test the status class
//
#include <stream.h>
#include "Status.h"

main()
{
    Status st(1337);

    for(int i=0; i < S_LINES; i++){
	for(int j=0; j < S_COL; j++)
	    cout << st[i][j] << " ";
	cout << "\n";
    }
}
---------------------------------------------------------------------------

I hope somebody out there can clear up this matter, 'cause in the meantime
I'll have to make the 'stat' array public, and cannot test range, nor 
detect potential recalc operation :-(
--
Heimir Thor Sverrisson		heimir@plusplus.is
Laugavegi 13
101 Reykjavik
Iceland

juul@diku.dk (Anders Juul Munch) (06/21/91)

heimir@rhi.hi.is (Heimir Thor Sverrisson) writes:


HTS>Hi fellow Netlanders,

HTS>is there anyone out there that can help me understand why the following
HTS>code works!   I'm trying to build a class that contains integer data
HTS>that should be organized as a two-dimensional array (of fixed dimensions).

HTS>I was trying to implement an operator that would allow me to use code like:
HTS>    Status st;
HTS>    st[TOTAL][COD]=123;
HTS>where Status is the name of my class and the [][] notation would allow me
HTS>to manipulate that element in my private array.  On top of the well known
HTS>problem of not being able to detect if I'm being used as an l- or r-value
HTS>I also ran into trouble defining the operator(s) involved to do this.

HTS>What I ended up with were two operator[]'s and the code did actually work
HTS>in g++ 1.39.  The implementation is :

HTS>    int* Status::operator[](int index)
HTS>    {
HTS>	return stat[index];		// private: int stat[i][j]
HTS>    }
HTS>    int& Status::operator[](int* iarr)
HTS>    {
HTS>	return *iarr;
HTS>    }

HTS>And I'm asking you why does this work?
HTS>If it is not a compiler bug/feature, how can I access the latter index (COD)
HTS>from within the second member function above (i.e. to test it's range)?

HTS>[Remainder omitted.]

What is happening is this: First, 'stat[TOTAL]' is evaluated. This is done
by 'Status::operator[](int)', because 'stat' is of class 'Status', and the
argument matches.
	'Status::operator[](int)' returns an 'int*'. This return value is
then indexed by '[COD]'. However, as 'int*' is not a class type, no member
function is being called this time. Instead, usual C array referencing is
performed. Your second member function, 'Status::operator[](int*)' will
never get called! Yes, it works, but this way you have no way to range check
the second index.
	To get control of both indices is pretty hard, if you insist on
using operator[]. If you don't, there is no problem, you could define an
array access member function like this:

int& Status::Stat(int firstIndex, int secondIndex)
{
  // Check if indices are valid here.

  return stat[firstIndex][secondIndex];
}

// The brackets aren't square, but what the heck . . .
Status st;
st(TOTAL,COD) = 123;

If you insist on using operator[], it could be done using an intermediate
class: 

class ArrayRow
{
  int* row;
 public:
  ArrayRow(int* it) { row = it; }
  int& operator[](int secondIndex)
    {
      // Check range of secondIndex here.
      return row[secondIndex];
    }
};

ArrayRow Status::operator[](int firstIndex)
{
  // Check range of firstIndex here.
  
  return ArrayRow(stat[i]);
}


This should work: You get all the range checking you want, and you can still
use the usual []-s. The performance penanlty involved in creating the
intermediate 'ArrayRow' object is negliable. - A good compiler might
even optimize it away entirely. The cost is that the code gets distributed
over two classes in a not very intuitive manner. Indeed, the intermediate
class is a trick which would make your code a lot more confusing. 

-- Anders Munch
juul@diku.dk
University of Copenhagen
Denmark