"ref const" parameters in functions

L-MAN vangelisforever at yandex.ru
Sun Apr 1 00:10:58 PDT 2012


On Saturday, 31 March 2012 at 21:42:05 UTC, Jonathan M Davis 
wrote:
> On Saturday, March 31, 2012 23:25:51 L-MAN wrote:
>> Hello everybody!
>> 
>> I'm trying to use some function FN like this:
>> 
>> struct X
>> {
>>   protected double _x;  // double type for example
>>   public @property double X() const { return _x; }
>> 
>>   // ctor
>>   public this(double x) { _x = x; } // double type for example
>> 
>>   void FN(ref const(double) in_x) // double type for example
>>   {
>>     // ... do some operations with _x
>>   }
>> 
>> }
>> 
>> main(..)
>> {
>>   ....
>>   X x = X(20);
>> 
>>   x.FN(30); // getting an error
>> }
>> 
>> 
>>   why x.FN(20) gets me an error?
>>   the construction "ref const(double)" or "ref 
>> immutable(double)"
>> must be an rvalue by default I think, but in FN parameter the
>> compiler expects an lvalue...
>>   this strategy is the way to some unnecessary copy operations,
>> when value 20 (or some big struct instead of it) will copy to 
>> the
>> stack..
>> 
>> How can I resolve this problem?
>
> Unlike C++, const ref _must_ be an lvalue just like ref. If you 
> use auto ref
> instead of const ref, the compiler is supposed to choose 
> between ref and non-
> ref based on which it thinks would be more efficient, but it 
> currently only
> works with templated types.
>
> You can duplicate the function and have a version whose 
> parameter is const ref
> and one which is not, but be careful if you try and make the 
> non-const ref
> version call the const ref version (to avoid duplicating the 
> function's body)
> and make sure that you don't get infinite recursion. It usually 
> works, but I've
> run into cases before where I ended up with infinite recursion, 
> so make sure
> that you have unit tests which check.
>
> Regardless, if you're dealing with a primitive type like 
> double, don't bother
> with const ref. It's not going to be more efficient. It's with 
> large structs
> that you can see a difference.
>
> - Jonathan M Davis

Thank you for reply, but how can I exclude some copy operations?
Look at this sample:

struct ABC // simple class
{
	private double _d=0;
	@property
	{
		double D() const { return _d; }
		double D(double val) { return _d=val; }
	}
	
	this(double d)
	{
		_d = d;
		writeln(" *** ctor ABC, d = ", _d);
	}
	
	ABC opAdd(ABC abc)
	{
		//ABC temp = ABC(this._d+abc._d);
		//return temp;
		return ABC(this._d+abc._d);
	}

	ABC opMul(double d)
	{
		//ABC temp = ABC(this._d*d);
		//return temp;
		return ABC(this._d*d);
	}
		
	ref ABC opAssign(ABC abc)
	{
		this._d = abc.D;
		return this;
	}
	
	~this()
	{
		writeln(" *** dtor ABC, d = ", _d);
	}
}


struct F {

	private ABC _abc;
	
	@property double D() const { return _abc.D; }
	
	public this(ABC abc)
	{
		_abc = abc;
		
		//abc.D=90;
		//pnt.X = 30;
	}
}


void main(string[] args)
{
   ABC abc1 = ABC(10);
   ABC abc2 = ABC(20);
   F f = F(abc1+abc2*20.0);
   writeln(f.D);
  .......
}

Operation abc1+abc2*20.0 consists of 2 ABC copy:
1. Copy result of abc2*20.0 temp value to opAdd function
2. Copy result of abc1+abc2*20.0 temp2 value to F() ctor
two temp variables and two copy operations.
But if I can use "ref const" rvalue operations 0 copies will be 
need:
   abc2*20.0 creates temp valiable $$$temp
   opAdd(ref const ABC abc)  use a ref rvalue to $$$temp, is not a 
copy
   opAdd return a $$$temp2 that use as ref rvalue for F() ctor
And try to imagine how many unnecessary copies of ABC will create 
a simple operation like this:
  ((abc1+abc2*20.0)/0.25) + (abc1/abc2)*(abc2-1)....
   it killing a time of application, especially if time is 
critical.

The constructions "auto ref" in return functions values are not 
working well, if I use it in a "auto ref opAdd(ref const ABC)" 
"auto ref opMul(double d)" " F.this(ref const ABC)" to exclude 
some copy operations it gives me a strange result :-((

Thank you!!!




More information about the Digitalmars-d-learn mailing list