473,320 Members | 2,020 Online
Bytes | Software Development & Data Engineering Community
Post Job

Home Posts Topics Members FAQ

Join Bytes to post your question to a community of 473,320 software developers and data experts.

Overloading 'casting' operator

Hi all,

I have a templated Vector3D class which holds (x,y,z) components as the
specified type. I quite often wish to cast a Vector3D holding ints into
a Vector3D holding floats and vice versa. Like so:

Vector3D<int> intVec(10,20,30);
Vector3D<float> floatVec = intVec;

Of course this doesn't work. I would be happy if instead the following
worked:
Vector3D<int> intVec(10,20,30);
Vector3D<float> floatVec = static_cast< Vector3D<float> >(intVec);

but of course that doesn't either. I have read online that it is not
possible to overload the static_cast operator (why, incidently?) but
what is the best approach to this problem? Ideally I would like the
first example to compile but generate a warning and the second example
to be fine (as works for built-in types).

Any thoughts appriciated,

David
Jan 17 '06 #1
8 9150
For the first one, I think you can overload = by
template <class T>
class Vector3D{
.....
template <class N> operator=(const Vector3D<N>& rvalue)
{....}
.....
};
my first reply may not be correct.

I don't know the answer to the second one though.
David Williams wrote:
Hi all,

I have a templated Vector3D class which holds (x,y,z) components as the
specified type. I quite often wish to cast a Vector3D holding ints into
a Vector3D holding floats and vice versa. Like so:

Vector3D<int> intVec(10,20,30);
Vector3D<float> floatVec = intVec;

Of course this doesn't work. I would be happy if instead the following
worked:
Vector3D<int> intVec(10,20,30);
Vector3D<float> floatVec = static_cast< Vector3D<float> >(intVec);

but of course that doesn't either. I have read online that it is not
possible to overload the static_cast operator (why, incidently?) but
what is the best approach to this problem? Ideally I would like the
first example to compile but generate a warning and the second example
to be fine (as works for built-in types).

Any thoughts appriciated,

David


Jan 17 '06 #2
David Williams wrote:
I have a templated Vector3D class which holds (x,y,z) components as the
specified type. I quite often wish to cast a Vector3D holding ints into
a Vector3D holding floats and vice versa. Like so:

Vector3D<int> intVec(10,20,30);
Vector3D<float> floatVec = intVec;

Of course this doesn't work. I would be happy if instead the following
worked:
Vector3D<int> intVec(10,20,30);
Vector3D<float> floatVec = static_cast< Vector3D<float> >(intVec);

but of course that doesn't either. I have read online that it is not
possible to overload the static_cast operator (why, incidently?) but
what is the best approach to this problem? Ideally I would like the
first example to compile but generate a warning and the second example
to be fine (as works for built-in types).


Without code showing the definition of template<typename T> Vector3D,
it's difficult to respond. See the FAQ:

http://www.parashift.com/c++-faq-lit...t.html#faq-5.8

As another poster said, a templated operator=() may give you what you
want for the first issue. For the second issue, although you can't
overload the static_cast operator directly, you could make the
static_cast operator work the way you want by providing a templated
copy constructor - something like (VERY untested code):

template<typename T> class Vector3D {
public:
template<typename U>
explicit Vector3D(const Vector3D<U>&);
};

Best regards,

Tom

Jan 17 '06 #3
David Williams wrote:
I have a templated Vector3D class which holds (x,y,z) components as the
specified type. I quite often wish to cast a Vector3D holding ints into
a Vector3D holding floats and vice versa. Like so:

Vector3D<int> intVec(10,20,30);
Vector3D<float> floatVec = intVec;
What you need here is not, as other have suggested, an overloaded
operator=. This is not assignment, but initialization -- i.e.,
construction. For construction, we use constructors. If you define a
single-argument constructor which takes another Vector3D of a different
type, you can do what you want here. It's important to recognize the
difference, though, between the above, assignment, and casting.

Initialization (construction) creates a new instance, based in some way
on the parameter(s) provided to the constructor.

Assignment modifies an existing instance, based in some way on the
parameter provided to operator=.

Casting is more complicated -- it depends on whether you're casting by
value, pointer, or reference. What's more, it is possible to provide
an overloaded "conversion operator" (not casting operator) which allows
implicit (or explicit, if that keyword is used) conversion to another
type. A conversion operator is sort of the inverse of a constructor.
Of course this doesn't work. I would be happy if instead the following
worked:

Vector3D<int> intVec(10,20,30);
Vector3D<float> floatVec = static_cast< Vector3D<float> >(intVec);


You should not be satisfied with this. Implement an appropriate
single-argument constructor and you'll have a better result.

Luke

Jan 17 '06 #4
Luke Meyers wrote:
David Williams wrote:
I have a templated Vector3D class which holds (x,y,z) components as the
specified type. I quite often wish to cast a Vector3D holding ints into
a Vector3D holding floats and vice versa. Like so:

Vector3D<int> intVec(10,20,30);
Vector3D<float> floatVec = intVec;

What you need here is not, as other have suggested, an overloaded
operator=. This is not assignment, but initialization -- i.e.,
construction. For construction, we use constructors. If you define a
single-argument constructor which takes another Vector3D of a different
type, you can do what you want here. It's important to recognize the
difference, though, between the above, assignment, and casting.


Ah yes, I can see that now. However, could it be argued (according to
the 'Big Three' rule) that if I have a custom constructor I should have
a custom assignment operator as well?

Initialization (construction) creates a new instance, based in some way
on the parameter(s) provided to the constructor.
Great, this now seems to work:

template <typename Type>
template <typename CastType>
Vector3D<Type>::Vector3D(const Vector3D<CastType>& vectorToSet)
{
m_tX = vectorToSet.x();
m_tY = vectorToSet.y();
m_tZ = vectorToSet.z();
}

Assignment modifies an existing instance, based in some way on the
parameter provided to operator=.

Casting is more complicated -- it depends on whether you're casting by
value, pointer, or reference. What's more, it is possible to provide
an overloaded "conversion operator" (not casting operator) which allows
implicit (or explicit, if that keyword is used) conversion to another
type. A conversion operator is sort of the inverse of a constructor.
I found the following code and modified it for my class. Is this the
'conversion operator' you are refering to?

template <typename Type>
template <typename CastType>
Vector3D<Type>::operator Vector3D<CastType> () throw()
{
return Vector3D<CastType>(m_tX,m_tY,m_tZ);
}

I was confused by the fact that the return type (Vector3D<CastType>)
appears to come between 'operator' and the empty brackets. But it's ok?

Of course this doesn't work. I would be happy if instead the following
worked:

Vector3D<int> intVec(10,20,30);
Vector3D<float> floatVec = static_cast< Vector3D<float> >(intVec);

Come to think of it, I could write a vector_cast function to do the
same. However, given what I now know I don't think it's necessary.

You should not be satisfied with this. Implement an appropriate
single-argument constructor and you'll have a better result.

Luke


Thanks for your help!
Jan 17 '06 #5
David Williams wrote:
Ah yes, I can see that now. However, could it be argued (according to
the 'Big Three' rule) that if I have a custom constructor I should have
a custom assignment operator as well?


No. The 'Big Three' in the rule are specifically the copy constructor,
assignment operator and destructor. The functions that the compiler
will generate for you if your don't define them yourself. If you need
to write your own version of any one of these three, you probably need
to write them all yourself because the compiler generated ones probably
will not do what you want. There is nothing in the 'rule' about any
other constructors you may write.

Gavin Deane

Jan 17 '06 #6
Luke Meyers wrote:
What you need here is not, as other have suggested, an overloaded
operator=. This is not assignment, but initialization -- i.e.,
construction. For construction, we use constructors. If you define a
single-argument constructor which takes another Vector3D of a different
type, you can do what you want here. It's important to recognize the
difference, though, between the above, assignment, and casting.


I agree... but didn't I suggest exactly that?

Best regards,

Tom

Jan 17 '06 #7
Gavin Deane wrote:
David Williams wrote:
Ah yes, I can see that now. However, could it be argued (according to
the 'Big Three' rule) that if I have a custom constructor I should have
a custom assignment operator as well?

No. The 'Big Three' in the rule are specifically the copy constructor,
assignment operator and destructor. The functions that the compiler
will generate for you if your don't define them yourself. If you need
to write your own version of any one of these three, you probably need
to write them all yourself because the compiler generated ones probably
will not do what you want. There is nothing in the 'rule' about any
other constructors you may write.

Gavin Deane


Ok, thanks.
Jan 17 '06 #8
Yeah, you're right. It's copy constructor instead of assignment
operator, thanks for pointing out.

Jan 18 '06 #9

This thread has been closed and replies have been disabled. Please start a new discussion.

Similar topics

3
by: David.H | last post by:
Good evening everyone, Im using this code to get an idea on overloading the + operator: class Graph { public: Graph(void); Graph(int valX, int valY); Graph operator+(const Graph&); int...
2
by: ryan.fairchild | last post by:
I have a problem I am trying to create a MyInt class to hanlde very large ints. Its for a class, therefore I can only do what the teach tells me. I want to be able to overload the insertion...
4
by: hall | last post by:
Hi all. I have run into a problem of overloading a templatized operator>> by a specialized version of it. In short (complete code below), I have written a stream class, STR, which defines a...
2
by: Peter | last post by:
Hello, Thanks for reviewing my question. I am trying to override the = operator for my UserControl class; However, I am getting a syntax error. public class MyClass { private int x = 0; ...
3
by: Suresh Tri | last post by:
Hi all, I was trying to overload '<' operator for (varchar,varchar). But in the function which handles the comparision I want to use the previous '<' operator.. but it is going into a recursion....
12
by: Achim Domma | last post by:
Hi, I want to use Python to script some formulas in my application. The user should be able to write something like A = B * C where A,B,C are instances of some wrapper classes. Overloading...
16
by: EM.Bateman | last post by:
Working on Visual Studio .Net I've implemented a class: #ifndef CONTRIBUTOR_H #define CONTRIBUTOR_H enum Gender {male=1, female, unk}; #include <iostream> #include <iomanip> #include...
4
by: UofFprogrammer | last post by:
Hello, I was taking a look at overloading the = operator in a C Plus Plus book. Part of the code of the overloaded =operator included checking for the case when the same object was on both sides of...
9
by: Faisal | last post by:
Hi, Why C++ doesn't allow overloading of size of operator. I think it would be much handy to check the type userdefined types. For eg. In my project, I've some structures which contains...
0
by: DolphinDB | last post by:
The formulas of 101 quantitative trading alphas used by WorldQuant were presented in the paper 101 Formulaic Alphas. However, some formulas are complex, leading to challenges in calculation. Take...
0
by: ryjfgjl | last post by:
ExcelToDatabase: batch import excel into database automatically...
0
isladogs
by: isladogs | last post by:
The next Access Europe meeting will be on Wednesday 6 Mar 2024 starting at 18:00 UK time (6PM UTC) and finishing at about 19:15 (7.15PM). In this month's session, we are pleased to welcome back...
1
isladogs
by: isladogs | last post by:
The next Access Europe meeting will be on Wednesday 6 Mar 2024 starting at 18:00 UK time (6PM UTC) and finishing at about 19:15 (7.15PM). In this month's session, we are pleased to welcome back...
1
by: PapaRatzi | last post by:
Hello, I am teaching myself MS Access forms design and Visual Basic. I've created a table to capture a list of Top 30 singles and forms to capture new entries. The final step is a form (unbound)...
0
by: CloudSolutions | last post by:
Introduction: For many beginners and individual users, requiring a credit card and email registration may pose a barrier when starting to use cloud servers. However, some cloud server providers now...
0
by: Defcon1945 | last post by:
I'm trying to learn Python using Pycharm but import shutil doesn't work
0
by: Shællîpôpï 09 | last post by:
If u are using a keypad phone, how do u turn on JavaScript, to access features like WhatsApp, Facebook, Instagram....
0
by: af34tf | last post by:
Hi Guys, I have a domain whose name is BytesLimited.com, and I want to sell it. Does anyone know about platforms that allow me to list my domain in auction for free. Thank you

By using Bytes.com and it's services, you agree to our Privacy Policy and Terms of Use.

To disable or enable advertisements and analytics tracking please visit the manage ads & tracking page.