close Warning: Can't synchronize with repository "(default)" (/var/svn/tolp does not appear to be a Subversion repository.). Look in the Trac log for more information.

Opened 13 years ago

Closed 13 years ago

Last modified 13 years ago

#1431 closed enhancement (fixed)

Reference Assignment Operator

Reported by: Pedro Gea Owned by: Víctor de Buen Remiro
Priority: highest Milestone: Mantainance
Component: Kernel Version:
Severity: critical Keywords:
Cc:

Description

Se solicita la implementación del operador de asignación pore referencia.

La motivación a este operador puede verse en el tique #1426, se sugirió incluso un símbolo =& (véanse los comentarios: 1426#comment:2 y 1426#comment:5).

La mejor alternativa que encuentro hoy para trabajar con referencias es usar la asignación por referencia natural de los conjuntos y por estética apoyarse en una estructura:

Struct @R { Anything D };

y escribir código del tipo:

Set data. = @R(Get(...));
// siendo Get un método que devuelve un dato ya existente
// por ejemplo uno cacheado
... = ... data.->D ...;

cuando sea conveniente, en lugar de:

Matrix data = Get(...);
// pues aquí se vuelve a crear (repitiéndose) la matriz
... = ... data ...;

Change History (12)

comment:1 Changed 13 years ago by Víctor de Buen Remiro

A mí no me parece nada mal esa solución, aunque se puede mejorar haciendo una estructura de referencia específica para cada gramática, para no perder el control de tipos y no abusar de Anything, lo cual resta eficiencia, robustez y legibilidad al lenguaje. Su uso es algo habría que restringir a los casos en los que no es posible dilucidar el tipo de datos en tiempo de programación.

Struct @Real { Real V };
Struct @Text{ Text V };
Struct @Date{ Date V };
Struct @Matrix { Matrix V };
Struct @Serie { Serie V };
...
Struct @Anything{ Anything V };

Dado un objeto previamente definido como por ejemplo

Matrix a = Constant(1000,10,1);

sería posible definir una referencia mediante tres formas alternativas y equivalentes

@Matrix M1 = @Matrix(a);
Set M2 = @Matrix(a);
@Matrix M3 = [[a]];

y luego habría dos formas alternativas de acceder al valor referenciado

Matrix b = M1->V + 1;
Matrix c = M1[1] + 2;

Estas estructuras se pueden incorporar fácilmente a la StdLib y su uso es muy sencillo. Quizás se podría estudiar un operador monario del estilo de C que permitiera abreviar aún más la escritura

Matrix d = *M + 2;

aunque esto podría despistar al usuario típico de TOL que no suele estar familiarizado con esta sintaxis.

Incluso se podría intervenir estas estructuras en el interfaz de TOLBase para representarlas directamente como su contenido en lugar de como un conjunto y evitar así acciones de ratón al usuario a la hora de explorarlos.

comment:2 Changed 13 years ago by Víctor de Buen Remiro

Una opción de abreviatura más intuitiva sería crear un nuevo operador monario $ con el mismo efecto que el * en C pero sin causar tanta confusión pues no se usa para ninguna otra cosa, aunque sí como parte de operadores como $* y $/, y el $ sí que recuerda el concepto de valor de algo

@Matrix M = @Matrix(...);
Matrix d = $M + 2;

comment:3 Changed 13 years ago by Víctor de Buen Remiro

(In [4135]) Refs #1431

comment:4 Changed 13 years ago by Víctor de Buen Remiro

(In [4136]) Refs #1431
Adds monary operator $

comment:5 Changed 13 years ago by Víctor de Buen Remiro

(In [4137]) Refs #1431
Implements referential structs

comment:6 Changed 13 years ago by Víctor de Buen Remiro

(In [4138]) Refs #1431
Declare initializer and access methods to referential structs

comment:7 Changed 13 years ago by Víctor de Buen Remiro

(In [4139]) Refs #1431
Calls to initializer of referential structs

comment:8 Changed 13 years ago by Víctor de Buen Remiro

(In [4140]) Refs #1431

comment:9 Changed 13 years ago by Víctor de Buen Remiro

Resolution: fixed
Status: newclosed

(In [4141]) Fixes #1431
Implements the referential operator $

comment:10 Changed 13 years ago by Víctor de Buen Remiro

(In [4143]) Refs #1431
Enhanced description of referential operator $

comment:11 Changed 13 years ago by imendez

Hola, tengo una duda: ¿cómo debe usarse este nuevo operador?

Me intentaré explicar. Si quiero hacer una asignación por referencia, entiendo que lo que persigo es lo siguiente:

1.- Definir una variable 'a'.
2.- Definir una variable 'b' que se identifique con 'a' antes de reasginar el valor de 'a'.
3.- Que al cambiar el valor de 'a', cambie el de 'b'.

No soy capaz de conseguir esto tras leer este ticket y el #1426.
Lo intento con el siguiente código:

Matrix a = Constant(1000,10,1);
@Matrix M1 = @Matrix(a);
Matrix b1 = $M1;
Matrix a := Constant(500,10,1);
Matrix b2 = $M1;
Matrix b3 = a;

Yo esperaría que 'b1' tuviera 500 filas ('b2' y 'b3' por supuesto también). Sin embargo, sigue teniendo el valor original de 'a'. Si sólo puedo cambiar el valor 'b2' definiéndolo después de haber reasignado el valor de 'a', ¿para qué me sirve todo el artilugio de '$M1'? Para eso, no es más sencilla la definición de b3?

No sé, me he debido perder en algún punto, os agradecería que me dijerais dónde.

Un saludo.

comment:12 Changed 13 years ago by Víctor de Buen Remiro

La referencia es $M1, el objeto b1 es una copia de la referencia que es lo mismo que copiar el original pues así se comporta el tipo Matrix. Eso no ha cambiado ni se puede cambiar pues sería incompatible hacia atrás.

Supongo que el lío viene de que hay tipos en los que el operador de definición = funciona como copia y otros tipos en los que funciona como copia. Esta lamentable situación ya he explicado cómo ha surgido y no es soluble por el mismo problema de compatibilidad hacia atrás.

El objetivo del $ y las estructuras @<type> es precisamente permitir la creación explícita de referencias sea cual sea el tipo de datos y sin perder el control sintáctico sobre los tipos de datos.

La idea es usar expresiones tipo $M1 en las operaciones que queramos que tengan efectos secundarios en el objeto original y usar copias como b1 en las que no queramos modificar el original.

La referencia puede hacerse a un objeto, a un elemento de un conjunto, a un miembro de un NameBlock o, en general a cualquier función que devuelva una referencia a un objeto existente.

Note: See TracTickets for help on using tickets.