creo que tiene una variable central intermedia es la mejor práctica de tal manera:
update z set c1 = @c := c1, c1 = c2, c2 = @c
En primer lugar, funciona siempre; segundo, funciona independientemente del tipo de datos.
A pesar de dos
update z set c1 = c1^c2, c2 = c1^c2, c1 = c1^c2
y
update z set c1 = c1 + c2, c2 = c1 - c2, c1 = c1 - c2
están trabajando normalmente, sólo por el número de tipo de datos, por cierto, y es su responsabilidad para evitar desbordamiento, no se puede utilizar XOR entre firmado y sin firmar, tampoco puede usar la suma por posibilidad de desbordamiento.
Y
update z set c1 = c2, c2 = @c where @c := c1
no está funcionando si c1 es 0 o NULL o cero longitud de la cadena o sólo espacios.
Necesitamos cambiarlo a
update z set c1 = c2, c2 = @c where if((@c := c1), true, true)
Aquí es las secuencias de comandos:
mysql> create table z (c1 int, c2 int)
-> ;
Query OK, 0 rows affected (0.02 sec)
mysql> insert into z values(0, 1), (-1, 1), (pow(2, 31) - 1, pow(2, 31) - 2)
-> ;
Query OK, 3 rows affected (0.00 sec)
Records: 3 Duplicates: 0 Warnings: 0
mysql> select * from z;
+------------+------------+
| c1 | c2 |
+------------+------------+
| 0 | 1 |
| -1 | 1 |
| 2147483647 | 2147483646 |
+------------+------------+
3 rows in set (0.02 sec)
mysql> update z set c1 = c1^c2, c2 = c1^c2, c1 = c1^c2;
ERROR 1264 (22003): Out of range value for column 'c1' at row 2
mysql> update z set c1 = c1 + c2, c2 = c1 - c2, c1 = c1 - c2;
ERROR 1264 (22003): Out of range value for column 'c1' at row 3
mysql> select * from z;
+------------+------------+
| c1 | c2 |
+------------+------------+
| 0 | 1 |
| 1 | -1 |
| 2147483646 | 2147483647 |
+------------+------------+
3 rows in set (0.02 sec)
mysql> update z set c1 = c2, c2 = @c where @c := c1;
Query OK, 2 rows affected (0.00 sec)
Rows matched: 2 Changed: 2 Warnings: 0
mysql> select * from z;
+------------+------------+
| c1 | c2 |
+------------+------------+
| 0 | 1 |
| -1 | 1 |
| 2147483647 | 2147483646 |
+------------+------------+
3 rows in set (0.00 sec)
mysql> select * from z;
+------------+------------+
| c1 | c2 |
+------------+------------+
| 1 | 0 |
| 1 | -1 |
| 2147483646 | 2147483647 |
+------------+------------+
3 rows in set (0.00 sec)
mysql> update z set c1 = @c := c1, c1 = c2, c2 = @c;
Query OK, 3 rows affected (0.02 sec)
Rows matched: 3 Changed: 3 Warnings: 0
mysql> select * from z;
+------------+------------+
| c1 | c2 |
+------------+------------+
| 0 | 1 |
| -1 | 1 |
| 2147483647 | 2147483646 |
+------------+------------+
3 rows in set (0.00 sec)
mysql>update z set c1 = c2, c2 = @c where if((@c := c1), true, true);
Query OK, 3 rows affected (0.02 sec)
Rows matched: 3 Changed: 3 Warnings: 0
mysql> select * from z;
+------------+------------+
| c1 | c2 |
+------------+------------+
| 1 | 0 |
| 1 | -1 |
| 2147483646 | 2147483647 |
+------------+------------+
3 rows in set (0.00 sec)
como una nota, 'tabla de ACTUALIZACIÓN SET X = Y, Y = X' es la forma estándar de hacerlo en SQL, solo MySQL se comporta mal. –