this , self y Me son palabras clave que se usan en algunos lenguajes de programación informática para referirse al objeto, clase u otra entidad de la que forma parte el código que se está ejecutando actualmente. La entidad a la que se hace referencia depende, por lo tanto, del contexto de ejecución (por ejemplo, qué objeto tiene su método llamado). Los distintos lenguajes de programación usan estas palabras clave de formas ligeramente diferentes. En los lenguajes en los que una palabra clave como "this" es obligatoria, la palabra clave es la única forma de acceder a los datos y métodos almacenados en el objeto actual. Cuando son opcionales, estas palabras clave pueden desambiguar variables y funciones con el mismo nombre.
En muchos lenguajes de programación orientados a objetos , (también llamado o ) es una variable que se utiliza en métodos de instancia para hacer referencia al objeto en el que están trabajando. El primer lenguaje orientado a objetos, SIMULA 67 , solía hacer referencia explícita al objeto local. [1] : 4.3.2.3 C++ y los lenguajes que derivan en estilo de él (como Java , C# , D y PHP ) también suelen utilizar . Smalltalk y otros, como Object Pascal , Perl , Python , Ruby , Rust , Objective-C , DataFlex y Swift , utilizan . Visual Basic de Microsoft utiliza .this
self
Me
this
this
self
Me
El concepto es similar en todos los lenguajes: es usualmente una referencia o punterothis
inmutable que se refiere al objeto actual; el objeto actual es a menudo el código que actúa como 'padre' o 'invocador' de la propiedad , método , subrutina o función que contiene la palabra clave. Después de que un objeto se construye o instancia correctamente, es siempre una referencia válida. Algunos lenguajes lo requieren explícitamente; otros usan el alcance léxico para usarlo implícitamente para hacer visibles los símbolos dentro de su clase. O alternativamente, el objeto actual al que se hace referencia puede ser un objeto de código independiente que ha llamado a la función o método que contiene la palabra clave . Tal cosa sucede, por ejemplo, cuando un controlador de eventos de JavaScript adjunto a una etiqueta HTML en una página web llama a una función que contiene la palabra clave almacenada en el espacio global fuera del objeto de documento; en ese contexto, se referirá al elemento de página dentro del objeto de documento, no al objeto de ventana que lo encierra. [2]this
this
this
this
this
this
En algunos lenguajes, por ejemplo C++, Java y Raku this
, o self
es una palabra clave y la variable existe automáticamente en los métodos de instancia. En otros, por ejemplo, Python, Rust y Perl 5, el primer parámetro de un método de instancia es una referencia de este tipo. Debe especificarse explícitamente. En Python y Perl, el parámetro no necesita necesariamente ser nombrado this
o self
; el programador puede nombrarlo libremente como cualquier otro parámetro. Sin embargo, por convención informal, el primer parámetro de un método de instancia en Perl o Python es nombrado self
. Rust requiere que se llame al objeto self &self
o self
, dependiendo de si la función invocada toma prestado el invocante o lo mueve, respectivamente.
Los métodos estáticos en C++ o Java no están asociados a instancias sino a clases, y por lo tanto no pueden usar this
, porque no hay ningún objeto. En otros lenguajes, como Ruby, Smalltalk, Objective-C o Swift, el método está asociado a un objeto de clase que se pasa como this
, y se denominan métodos de clase . Para los métodos de clase, Python usa cls
para acceder al objeto de clase .
Cuando se utiliza el alcance léxico para inferir this
, el uso de this
en el código, si bien no es ilegal, puede generar alarmas para un programador de mantenimiento, aunque todavía hay usos legítimos de this
en este caso, como hacer referencia a variables de instancia ocultas por variables locales del mismo nombre, o si el método desea devolver una referencia al objeto actual, es decir this
, a sí mismo.
En algunos compiladores (por ejemplo GCC ), los punteros a métodos de instancia de C++ se pueden convertir directamente a un puntero de otro tipo, con un this
parámetro de puntero explícito. [3]
La semántica de envío de this
, es decir, que las llamadas a métodos de this
se envían dinámicamente, se conoce como recursión abierta y significa que estos métodos pueden ser reemplazados por clases u objetos derivados. Por el contrario, la recursión directa nombrada o la recursión anónima de una función utiliza la recursión cerrada , con envío estático. Por ejemplo, en el siguiente código de Perl para el factorial, el token __SUB__
es una referencia a la función actual:
utilizar la función ":5.16" ; sub { mi $x = shift ; $x == 0 ? 1 : $x * __SUB__ -> ( $x - 1 ); }
Por el contrario, en C++ (usando un explícito this
para mayor claridad, aunque no es necesario) se this
vincula al objeto mismo, pero si el método de clase se declaró "virtual", es decir, polimórfico en la base, se resuelve mediante envío dinámico para que las clases derivadas puedan anularlo.
unsigned int factorial ( unsigned int n ) { si ( n == 0 ) devuelve 1 ; de lo contrario devuelve n * esto -> factorial ( n - 1 ); }
Este ejemplo es artificial ya que se trata de una recursión directa, por lo que anular el factorial
método anularía esta función; ejemplos más naturales son cuando un método en una clase derivada llama al mismo método en una clase base, o en casos de recursión mutua. [4] [5]
El problema de la clase base frágil se ha achacado a la recursión abierta, con la sugerencia de que la invocación de métodos por this
defecto se realiza mediante recursión cerrada (despacho estático) en lugar de recursión abierta (despacho dinámico), utilizando solo la recursión abierta cuando se solicita específicamente; las llamadas externas (sin utilizar this
) se enviarían dinámicamente como de costumbre. [6] [7] La forma en que esto se resuelve en la práctica en el JDK es a través de una cierta disciplina de programación; esta disciplina ha sido formalizada por C. Ruby y GT Leavens; consta de las siguientes reglas: [8]
public
métodos en this
.protected
o private
; si también necesita exponerse directamente a los usuarios, entonces un public
método contenedor llama al método interno.Las primeras versiones de C++ permitían this
cambiar el puntero; al hacerlo, un programador podía cambiar el objeto en el que estaba trabajando un método. Esta característica se eliminó con el tiempo y ahora this
en C++ es un valor r . [9]
Las primeras versiones de C++ no incluían referencias y se ha sugerido que si hubieran estado en C++ desde el principio, this
habrían sido una referencia, no un puntero. [10]
C++ permite que los objetos se destruyan a sí mismos con la declaración del código fuente: delete this
.
La palabra clave this
en C# funciona de la misma manera que en Java para los tipos de referencia. Sin embargo, dentro de C# los tipos de valor tienen una semántica bastante diferente , this
ya que es similar a una referencia de variable mutable común y puede incluso aparecer en el lado izquierdo de una asignación.
Un uso de this
en C# es permitir la referencia a una variable de campo externo dentro de un método que contiene una variable local que tiene el mismo nombre. En tal situación, por ejemplo, la declaración var n = localAndFieldname;
dentro del método asignará el tipo y valor de la variable local localAndFieldname
a n
, mientras que la declaración var n = this.localAndFieldname;
asignará el tipo y valor de la variable de campo externo a n
. [11]
En D, this
en una clase, una estructura o un método de unión hace referencia a una referencia inmutable de la instancia del agregado que la contiene. Las clases son tipos de referencia y las estructuras y uniones son tipos de valor. En la primera versión de D, la palabra clave this
se utiliza como un puntero a la instancia del objeto al que está vinculado el método, mientras que en D2 tiene el carácter de un refargumento de función implícito.
En el lenguaje de programación Dylan , que es un lenguaje orientado a objetos que admite métodos múltiples y no tiene un concepto de this
, el envío de un mensaje a un objeto se mantiene en la sintaxis. Las dos formas siguientes funcionan de la misma manera; las diferencias son solo azúcar sintáctica .
objeto.metodo(param1, param2)
y
método (objeto, parámetro1, parámetro2)
Dentro de un texto de clase, el tipo actual es el tipo obtenido de la clase actual . Dentro de las características (rutinas, comandos y consultas) de una clase, se puede usar la palabra clave Current
para hacer referencia a la clase actual y sus características. El uso de la palabra clave Current
es opcional, ya que la palabra clave Current
está implícita simplemente al referirse abiertamente al nombre de la característica de la clase actual. Por ejemplo: Uno podría tener una característica 'foo' en una clase MY_CLASS y hacer referencia a ella mediante:
clase MI_CLASE característica -- Acceso foo : ENTERO mi_funcion : ENTERO hacer Resultado := foo fin fin
[12]
La línea n.° 10 (arriba) tiene la referencia implícita Current
mediante la llamada al simple 'foo'.
La línea n.° 10 (abajo) tiene la referencia explícita a Current
la llamada a 'Current.foo'.
clase MI_CLASE característica -- Acceso foo : ENTERO mi_funcion : ENTERO hacer Resultado := Actual . foo fin fin
Cualquiera de los dos enfoques es aceptable para el compilador, pero se prefiere la versión implícita (por ejemplo, x := foo
) ya que es menos detallada.
Al igual que en otros idiomas, hay ocasiones en las que Current
es obligatorio el uso de la palabra clave, como por ejemplo:
clase MI_CLASE característica -- Acceso mi_comando -- Crear MY_OTHER_CLASS con 'Actual' local x : MI_OTRA_CLASE hacer crear x . make_with_something ( Actual ) fin fin
En el caso del código anterior, la llamada en la línea n.° 11 a make_with_something pasa la clase actual pasando explícitamente la palabra clave Current
.
La palabra clave this
es una palabra clave del lenguaje Java que representa la instancia actual de la clase en la que aparece. Se utiliza para acceder a las variables y métodos de la clase.
Dado que todos los métodos de instancia son virtuales en Java, this
nunca pueden ser nulos. [13]
En JavaScript, que es un lenguaje de programación o scripting ampliamente utilizado en navegadores web, this
es una palabra clave importante, aunque su evaluación depende de dónde se utilice.
this
se refiere al objeto circundante, que en este caso es la ventana del navegador circundante, el window
objeto.this
se refiere la palabra clave depende de cómo se llama a la función. Cuando se llama a una función de este tipo directamente (por ejemplo, f(x)
), this
hará referencia al espacio global en el que se define la función y en el que también pueden existir otras funciones y variables globales (o en modo estricto, es undefined
). Sin embargo, si se llama a una función global que contiene this
como parte del controlador de eventos de un elemento en el objeto de documento, this
hará referencia al elemento HTML que realiza la llamada.new
palabra clave (por ejemplo, var c = new Thing()
) entonces dentro de Thing this
se hace referencia al objeto Thing en sí.obj.f(x)
), this
se referirá al objeto dentro del cual está contenida la función. [14] [15] Incluso es posible especificar manualmente this
cuándo se llama a una función, utilizando los métodos .call()
o .apply()
del objeto de función. [16] Por ejemplo, la llamada al método obj.f(x)
también podría escribirse como obj.f.call(obj, x)
.Para solucionar los diferentes significados de this
las funciones anidadas, como los controladores de eventos DOM, en JavaScript es común guardar la this
referencia del objeto que llama en una variable (comúnmente llamada that
o self
) y luego usar la variable para hacer referencia al objeto que llama en funciones anidadas.
Por ejemplo:
// En este ejemplo, $ es una referencia a la biblioteca jQuery $ ( ".element" ). hover ( function () { // Aquí, tanto this como that apuntan al elemento bajo el cursor del mouse. var that = this ; $ ( this ). find ( '.elements' ). each ( function () { // Aquí, this apunta al elemento DOM que se está iterando. // Sin embargo, that todavía apunta al elemento bajo el cursor del mouse. $ ( this ). addClass ( "highlight" ); }); });
Cabe destacar que JavaScript utiliza ambos this
y la palabra clave relacionada self
[17] (a diferencia de la mayoría de los otros lenguajes que tienden a emplear uno u otro), y self
está restringido específicamente a los trabajadores web. [18]
Finalmente, como una forma confiable de hacer referencia específica al objeto global (ventana o equivalente), JavaScript presenta la globalThis
palabra clave. [19]
En Lua, self
se crea como azúcar sintáctico cuando se definen funciones utilizando el :
operador. [20] Al invocar un método utilizando :
, el objeto que se está indexando se dará implícitamente como el primer argumento de la función que se está invocando.
Por ejemplo, las siguientes dos funciones son equivalentes:
objeto local = {}función obj . foo ( arg1 , arg2 ) print ( arg1 , arg2 ) -- no se puede usar "self" aquí finfunción obj : bar ( arg ) print ( self , arg ) -- "self" es un primer argumento implícito antes del fin de arg-- Todas las funciones se pueden invocar de ambas formas, con "." o con ":"obj : foo ( "Foo" ) -- equivalente a obj.foo(obj, "Foo") obj .bar ( obj , " Bar " ) -- equivalente a obj :bar("Bar")
Lua en sí no está orientado a objetos, pero cuando se combina con otra característica llamada metatablas, su uso self
permite a los programadores definir funciones de una manera similar a la programación orientada a objetos.
En PowerShell, la variable automática $_
especial contiene el objeto actual en el objeto de canalización. Puede utilizar esta variable en comandos que realizan una acción en cada objeto o en objetos seleccionados en una canalización. [21]
"uno" , "dos" , "tres" | % { write $_ }
También a partir de PowerShell 5.0, que agrega una sintaxis formal para definir clases y otros tipos definidos por el usuario, [22] $this
la variable describe la instancia actual del objeto.
En Python, no existe una palabra clave para this
. Cuando se llama a una función miembro en un objeto, se invoca la función miembro con el mismo nombre en la clase object del objeto, y el objeto se vincula automáticamente al primer argumento de la función. Por lo tanto, el primer parámetro obligatorio de los métodos de instancia es this
; este parámetro se denomina convencionalmente self
, pero puede tener cualquier nombre.
En los métodos de clase (creados con el classmethod
decorador), el primer argumento se refiere al objeto de clase en sí, y se denomina convencionalmente cls
; estos se utilizan principalmente para constructores heredables, [23] donde el uso de la clase como parámetro permite subclasificar el constructor. En los métodos estáticos (creados con el staticmethod
decorador), no existe ningún primer argumento especial.
En Rust, los tipos se declaran por separado de las funciones asociadas a ellos. Las funciones diseñadas para ser análogas a los métodos de instancia en lenguajes orientados a objetos más tradicionales deben tomar explícitamente self
como su primer parámetro. Estas funciones pueden luego ser invocadas usando instance.method()
sintaxis simple. Por ejemplo:
estructura Foo { bar : i32 , } impl Foo { función new () -> Foo { Foo { bar : 0 , } } función referir ( & self ) { println! ( " { } " , self.bar ) ; } función mutar ( & mut self , baz : i32 ) { self.bar = baz ; } función consumir ( self ) { self.refer ( ) ; } }
Esto define un tipo, Foo
, que tiene cuatro funciones asociadas. La primera, Foo::new()
, no es una función de instancia y debe especificarse con el prefijo de tipo. Las tres restantes toman un self
parámetro de diversas maneras y se pueden llamar en una Foo
instancia utilizando la sintaxis de notación de puntos, que es equivalente a llamar al nombre de la función calificada por tipo con un primer parámetro explícito self
.
let mut foo = Foo :: new (); // debe llamarse como una función de tipo especificado foo . refer (); // imprime "0". Foo::refer() tiene acceso de solo lectura a la instancia foo foo . mutate ( 5 ); // muta foo en su lugar, permitido por la especificación &mut, necesita que foo sea declarado mut foo . consume (); // imprime "5" y destruye foo, ya que Foo::consume() toma plena propiedad de sí mismo // equivalente a foo.refer() Foo :: refer ( foo ); // error de compilación: foo está fuera del alcance
El lenguaje del yo recibe su nombre de este uso de "yo".
Self
se utiliza estrictamente dentro de los métodos de una clase. Otra forma de referirse a Self
es usar ::
.