From ec18ed8de3a791275eae59ec040f3c5c52467e31 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Dani=C3=ABl=20van=20Eeden?= Date: Wed, 31 Jul 2024 10:43:21 +0200 Subject: [PATCH] Functions: Add examples for bit functions and operators (#18253) --- .../bit-functions-and-operators.md | 292 +++++++++++++++++- tso.md | 2 +- 2 files changed, 286 insertions(+), 8 deletions(-) diff --git a/functions-and-operators/bit-functions-and-operators.md b/functions-and-operators/bit-functions-and-operators.md index 3760495c08e54..1627ff31d831d 100644 --- a/functions-and-operators/bit-functions-and-operators.md +++ b/functions-and-operators/bit-functions-and-operators.md @@ -12,13 +12,291 @@ TiDB supports all of the [bit functions and operators](https://dev.mysql.com/doc | Name | Description | | :------| :------------- | -| [`BIT_COUNT()`](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#function_bit-count) | Return the number of bits that are set as 1 | -| [&](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#operator_bitwise-and) | Bitwise AND | -| [~](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#operator_bitwise-invert) | Bitwise inversion | -| [\|](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#operator_bitwise-or) | Bitwise OR | -| [^](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#operator_bitwise-xor) | Bitwise XOR | -| [<<](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#operator_left-shift) | Left shift | -| [>>](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#operator_right-shift) | Right shift | +| [`BIT_COUNT()`](#bit_count) | Return the number of bits that are set as 1 | +| [`&`](#-bitwise-and) | Bitwise AND | +| [`~`](#-bitwise-inversion) | Bitwise inversion | +| [`\|`](#-bitwise-or) | Bitwise OR | +| [`^`](#-bitwise-xor) | Bitwise XOR | +| [`<<`](#-left-shift) | Left shift | +| [`>>`](#-right-shift) | Right shift | + +## [`BIT_COUNT()`](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#function_bit-count) + +The `BIT_COUNT(expr)` function returns the number of bits that are set as 1 in `expr`. + +```sql +SELECT BIT_COUNT(b'00101001'); +``` + +``` ++------------------------+ +| BIT_COUNT(b'00101001') | ++------------------------+ +| 3 | ++------------------------+ +1 row in set (0.00 sec) +``` + +> **Note:** +> +> If the argument `expr` is a binary number, you need to specify `b` explicitly before the number, such as `b'00101001'`. Otherwise, this function treats it as a string and returns a different result. For example, `BIT_COUNT('00101001')` returns `7` because it converts the string `'00101001'` as the decimal number `101001` and counts the number of `1` bits in its binary format `11000100001010001`. + +The following example is similar to the preceding one, but it uses a hex-literal instead of a bit-literal as the argument. The `CONV()` function converts `0x29` from hexadecimal (base 16) to binary (base 2), showing that it equals `00101001` in binary. + +```sql +SELECT BIT_COUNT(0x29), CONV(0x29,16,2); +``` + +``` ++-----------------+-----------------+ +| BIT_COUNT(0x29) | CONV(0x29,16,2) | ++-----------------+-----------------+ +| 3 | 101001 | ++-----------------+-----------------+ +1 row in set (0.01 sec) +``` + +A practical use of the `BIT_COUNT(expr)` function is to convert a netmask to a [CIDR](https://en.wikipedia.org/wiki/Classless_Inter-Domain_Routing) notation. In the following example, the netmask `255.255.255.0` is converted to its CIDR representation `24`. + +```sql +SELECT BIT_COUNT(INET_ATON('255.255.255.0')); +``` + +``` ++---------------------------------------+ +| BIT_COUNT(INET_ATON('255.255.255.0')) | ++---------------------------------------+ +| 24 | ++---------------------------------------+ +1 row in set (0.00 sec) +``` + +## [`&` (bitwise AND)](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#operator_bitwise-and) + +The `&` operator performs a bitwise AND operation. It compares the corresponding bits of two numbers: if both corresponding bits are 1, the corresponding bit of the result is 1; otherwise, it is 0. + +For example, a bitwise AND operation between `1010` and `1100` returns `1000`, because only the leftmost bit is set as 1 in both numbers. + +``` + 1010 +& 1100 + ---- + 1000 +``` + +In SQL, you can use the `&` operator as follows: + +```sql +SELECT CONV(b'1010' & b'1000',10,2); +``` + +``` ++------------------------------+ +| CONV(b'1010' & b'1000',10,2) | ++------------------------------+ +| 1000 | ++------------------------------+ +1 row in set (0.00 sec) +``` + +You can use the `&` operator with `INET_NTOA()` and `INET_ATON()` functions to perform a bitwise AND operation on an IP address and a network mask to get the network address. This is useful to determine whether multiple IP addresses belong to the same network or not. + +In the following two examples, the IP addresses `192.168.1.1` and `192.168.1.2` are in the same network `192.168.1.0/24` when masked with `255.255.255.0`. + +```sql +SELECT INET_NTOA(INET_ATON('192.168.1.1') & INET_ATON('255.255.255.0')); +``` + +``` ++------------------------------------------------------------------+ +| INET_NTOA(INET_ATON('192.168.1.1') & INET_ATON('255.255.255.0')) | ++------------------------------------------------------------------+ +| 192.168.1.0 | ++------------------------------------------------------------------+ +1 row in set (0.00 sec) +``` + +```sql +SELECT INET_NTOA(INET_ATON('192.168.1.2') & INET_ATON('255.255.255.0')); +``` + +``` ++------------------------------------------------------------------+ +| INET_NTOA(INET_ATON('192.168.1.2') & INET_ATON('255.255.255.0')) | ++------------------------------------------------------------------+ +| 192.168.1.0 | ++------------------------------------------------------------------+ +1 row in set (0.00 sec) +``` + +## [`~` (bitwise inversion)](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#operator_bitwise-invert) + +The `~` operator performs a bitwise inversion (or bitwise NOT) operation on a given value. It inverts each bit of the given value: bits that are 0 become 1, and bits that are 1 become 0. + +Before the operation, the value is expanded to 64 bits. + +Take the binary number `1111000011110000` as an example. When expanded to 64 bits and inverted, it looks like this: + +``` +Original (16 bits): 1111000011110000 +Expanded and inverted (64 bits): 1111111111111111111111111111111111111111111111110000111100001111 +``` + +In SQL, you can use the `~` operator as follows: + +```sql +SELECT CONV(~ b'1111000011110000',10,2); ++------------------------------------------------------------------+ +| CONV(~ b'1111000011110000',10,2) | ++------------------------------------------------------------------+ +| 1111111111111111111111111111111111111111111111110000111100001111 | ++------------------------------------------------------------------+ +1 row in set (0.00 sec) +``` + +You can reverse the inversion by applying the `~` operator again to the result: + +```sql +SELECT CONV(~ b'1111111111111111111111111111111111111111111111110000111100001111',10,2); +``` + +``` ++----------------------------------------------------------------------------------+ +| CONV(~ b'1111111111111111111111111111111111111111111111110000111100001111',10,2) | ++----------------------------------------------------------------------------------+ +| 1111000011110000 | ++----------------------------------------------------------------------------------+ +1 row in set (0.00 sec) +``` + +## [`|` (bitwise OR)](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#operator_bitwise-or) + +The `|` operator performs a bitwise OR operation. It compares the corresponding bits of two numbers: if at least one of the corresponding bits is 1, the corresponding bit in the result is 1. + +For example, a bitwise OR operation between `1010` and `1100` returns `1110`, because among the first three bits of the two numbers, at least one of the corresponding bits is set as 1. + +``` + 1010 +| 1100 + ---- + 1110 +``` + +In SQL, you can use the `|` operator as follows: + +```sql +SELECT CONV(b'1010' | b'1100',10,2); +``` + +``` ++------------------------------+ +| CONV(b'1010' | b'1100',10,2) | ++------------------------------+ +| 1110 | ++------------------------------+ +1 row in set (0.00 sec) +``` + +## [`^` (bitwise XOR)](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#operator_bitwise-xor) + +The `^` operator performs a bitwise XOR (exclusive OR) operation. It compares the corresponding bits of two numbers: if the corresponding bits are different, the corresponding bit in the result is 1. + +For example, a bitwise XOR operation between `1010` and `1100` returns `0110`, because the second and the third bits of the two numbers are different. + +``` + 1010 +^ 1100 + ---- + 0110 +``` + +In SQL, you can use the `^` operator as follows: + +```sql +SELECT CONV(b'1010' ^ b'1100',10,2); +``` + +``` ++------------------------------+ +| CONV(b'1010' ^ b'1100',10,2) | ++------------------------------+ +| 110 | ++------------------------------+ +1 row in set (0.00 sec) +``` + +Note that the result is shown as `110` instead of `0110` because the leading zero is removed. + +## [`<<` (left shift)](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#operator_left-shift) + +The `<<` operator performs a left shift operation, which shifts the bits of a number to the left by a specified number of positions, filling the vacated bits with zeros on the right. + +For example, the following statement uses `1<>` (right shift)](https://dev.mysql.com/doc/refman/8.0/en/bit-functions.html#operator_right-shift) + +The `>>` operator performs a right shift operation, which shifts the bits of a number to the right by a specified number of positions, filling the vacated bits with zeros on the left. + +For example, the following statement uses `1024>>n` to shift a value of `1024` (`10000000000` in binary) to the right by `n` positions: + +```sql +WITH RECURSIVE cte(n) AS ( + SELECT 0 AS n + UNION ALL + SELECT n+1 FROM cte WHERE n<11 +) +SELECT n,1024>>n,LPAD(CONV(1024>>n,10,2),11,0) FROM cte; +``` + +``` ++------+---------+-------------------------------+ +| n | 1024>>n | LPAD(CONV(1024>>n,10,2),11,0) | ++------+---------+-------------------------------+ +| 0 | 1024 | 10000000000 | +| 1 | 512 | 01000000000 | +| 2 | 256 | 00100000000 | +| 3 | 128 | 00010000000 | +| 4 | 64 | 00001000000 | +| 5 | 32 | 00000100000 | +| 6 | 16 | 00000010000 | +| 7 | 8 | 00000001000 | +| 8 | 4 | 00000000100 | +| 9 | 2 | 00000000010 | +| 10 | 1 | 00000000001 | +| 11 | 0 | 00000000000 | ++------+---------+-------------------------------+ +12 rows in set (0.00 sec) +``` + +The `>>` operator can also be useful for extracting specific parts of a larger number, such as extracting a UNIX timestamp from a TiDB [TSO](/tso.md) timestamp. ## MySQL compatibility diff --git a/tso.md b/tso.md index 897006c0120e1..add4ffdd956bd 100644 --- a/tso.md +++ b/tso.md @@ -78,7 +78,7 @@ FROM_UNIXTIME((@ts >> 18)/1000): 2023-08-27 20:33:41.6870 1 row in set (0.00 sec) ``` -The `>> 18` operation signifies a bitwise [right shift](/functions-and-operators/bit-functions-and-operators.md) by 18 bits, which is used to extract the physical timestamp. Because the physical timestamp is expressed in milliseconds, deviating from the more common UNIX timestamp format measured in seconds, you need to divide it by 1000 to convert it into a format compatible with [`FROM_UNIXTIME()`](/functions-and-operators/date-and-time-functions.md). This process aligns with the functionality of `TIDB_PARSE_TSO()`. +The `>> 18` operation signifies a bitwise [right shift](/functions-and-operators/bit-functions-and-operators.md#-right-shift) by 18 bits, which is used to extract the physical timestamp. Because the physical timestamp is expressed in milliseconds, deviating from the more common UNIX timestamp format measured in seconds, you need to divide it by 1000 to convert it into a format compatible with [`FROM_UNIXTIME()`](/functions-and-operators/date-and-time-functions.md). This process aligns with the functionality of `TIDB_PARSE_TSO()`. You can also extract the logical timestamp `000000000000000100` in binary, which is equivalent to `4` in decimal.