|
@ -1,72 +1,88 @@ |
|
|
#pragma once |
|
|
#pragma once |
|
|
|
|
|
|
|
|
/// ShortCode is a high-compression encoding scheme based on CommonCode. Since
|
|
|
/// ShortCode is a high-compression encoding scheme based on CommonCode. Since
|
|
|
/// there are a total of 29334498 valid klotski layouts, arrange their
|
|
|
/// there are a total of 29334498 valid klotski cases, arrange then from small
|
|
|
/// CommonCodes from small to large (36-bit positive integers), and use the
|
|
|
/// to large by their CommonCodes (36-bit positive integers), and use the index
|
|
|
/// index as the ShortCode.
|
|
|
/// as the ShortCode.
|
|
|
|
|
|
|
|
|
/// Therefore, the valid value of ShortCode is [0, 29334498), stored in
|
|
|
/// Therefore, the valid value of ShortCode is [0, 29334498), which stored in a
|
|
|
/// `uint32_t`. The goal of high compression ratio is to facilitate verbal
|
|
|
/// `uint32_t` variable. The goal of high compression ratio is to facilitate
|
|
|
/// sharing, so it is necessary to represent it in a suitable string. Similar
|
|
|
/// verbal sharing, so it is necessary to represent it into a suitable string.
|
|
|
/// to Bitcoin's `base58`, in ShortCode, 4 confusing characters `0` `O` `I` `l`
|
|
|
/// Similar to Bitcoin's base58 encoding, in ShortCode, 4 confusing characters
|
|
|
/// are removed from 10 numbers and 26 characters, forming a private base32
|
|
|
/// `0` `O` `I` `l` are removed from 10 numbers and 26 characters, forming a
|
|
|
/// scheme.
|
|
|
/// private base32 scheme.
|
|
|
|
|
|
|
|
|
/// Coincidentally, log(32, 29334498) is approximately equal to `4.96`, so
|
|
|
/// Coincidentally, log(32, 29334498) is approximately equal to 4.96, so using
|
|
|
/// using 5-bit base32 can make good use of space, so any valid klotski layout
|
|
|
/// 5-bit base32 can make good use of space, so any valid klotski cases can be
|
|
|
/// can be represented by a 5-bit length code. As in CommonCode, the characters
|
|
|
/// represented by a 5-bit length code. As in CommonCode, the characters here
|
|
|
/// here are case insensitive, but uppercase is still recommended.
|
|
|
/// are case insensitive, but uppercase is still recommended.
|
|
|
|
|
|
|
|
|
/// ShortCode Convert Table
|
|
|
/// Compared with the CommonCode, although ShortCode saves space, it completely
|
|
|
/// -------------------------------------------------
|
|
|
/// loses readability. The former can directly get the case without the help of
|
|
|
/// | 00 | 01 | 02 | 03 | 04 | 05 | 06 | 07 |
|
|
|
/// a computer, while the latter is almost impossible to complete by the human
|
|
|
/// | `1` | `2` | `3` | `4` | `5` | `6` | `7` | `8` |
|
|
|
/// brain. But anyway, ShortCode makes it easy to manually record the klotski
|
|
|
/// |-----------------------------------------------|
|
|
|
/// cases, either verbally or handwritten.
|
|
|
/// | 08 | 09 | 10 | 11 | 12 | 13 | 14 | 15 |
|
|
|
|
|
|
/// | `9` | `A` | `B` | `C` | `D` | `E` | `F` | `G` |
|
|
|
/// ShortCode Convert Table ///
|
|
|
/// |-----------------------------------------------|
|
|
|
/// ------------------------------------------------- ///
|
|
|
/// | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 |
|
|
|
/// | 00 | 01 | 02 | 03 | 04 | 05 | 06 | 07 | ///
|
|
|
/// | `H` | `J` | `K` | `M` | `N` | `P` | `Q` | `R` |
|
|
|
/// | `1` | `2` | `3` | `4` | `5` | `6` | `7` | `8` | ///
|
|
|
/// |-----------------------------------------------|
|
|
|
/// |-----------------------------------------------| ///
|
|
|
/// | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 |
|
|
|
/// | 08 | 09 | 10 | 11 | 12 | 13 | 14 | 15 | ///
|
|
|
/// | `S` | `T` | `U` | `V` | `W` | `X` | `Y` | `Z` |
|
|
|
/// | `9` | `A` | `B` | `C` | `D` | `E` | `F` | `G` | ///
|
|
|
/// -------------------------------------------------
|
|
|
/// |-----------------------------------------------| ///
|
|
|
|
|
|
/// | 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | ///
|
|
|
/// Eg1:
|
|
|
/// | `H` | `J` | `K` | `M` | `N` | `P` | `Q` | `R` | ///
|
|
|
/// 0x1A9BF0C00 -> index 4091296
|
|
|
/// |-----------------------------------------------| ///
|
|
|
/// 4091296 = 3 * (32 ^ 4) + 28 * (32 ^ 3) + 27 * (32 ^ 2) + 13 * (32 ^ 1) + 0 * (32 ^ 0)
|
|
|
/// | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | ///
|
|
|
/// => (3), (28), (27), (13), (0)
|
|
|
/// | `S` | `T` | `U` | `V` | `W` | `X` | `Y` | `Z` | ///
|
|
|
/// => `4`, `W`, `V`, `E`, `1`
|
|
|
/// ------------------------------------------------- ///
|
|
|
/// => "4WVE1"
|
|
|
|
|
|
|
|
|
/// ------------------------------------------------------------------------------------------ ///
|
|
|
/// Eg2:
|
|
|
/// Eg1: ///
|
|
|
/// 0x4FEA13400 -> index 10399732
|
|
|
/// % # # % ///
|
|
|
/// 10399732 = 9 * (32 ^ 4) + 29 * (32 ^ 3) + 11 * (32 ^ 2) + 31 * (32 ^ 1) + 20 * (32 ^ 0)
|
|
|
/// % # # % ///
|
|
|
/// => (9), (29), (11), (31), (20)
|
|
|
/// @ $ $ @ CommonCode = 0x1A9BF0C00 (index 4091296) ///
|
|
|
/// => `A`, `X`, `C`, `Z`, `N`
|
|
|
/// @ & * @ ///
|
|
|
/// => "AXCZN"
|
|
|
/// * & ///
|
|
|
|
|
|
/// ///
|
|
|
/// Compared with CommonCode, although ShortCode saves space, it completely
|
|
|
/// 4091296 => 3 * (32 ^ 4) + 28 * (32 ^ 3) + 27 * (32 ^ 2) + 13 * (32 ^ 1) + 0 * (32 ^ 0) ///
|
|
|
/// loses readability. The former can directly get the layout without the help
|
|
|
/// => (3), (28), (27), (13), (0) ///
|
|
|
/// of a computer, while the latter is almost impossible to complete by the
|
|
|
/// => `4`, `W`, `V`, `E`, `1` ///
|
|
|
/// human brain.
|
|
|
/// => "4WVE1" ///
|
|
|
|
|
|
/// ------------------------------------------------------------------------------------------ ///
|
|
|
|
|
|
|
|
|
|
|
|
/// -------------------------------------------------------------------------------------------- ///
|
|
|
|
|
|
/// Eg2: ///
|
|
|
|
|
|
/// * @ & % ///
|
|
|
|
|
|
/// # # $ % ///
|
|
|
|
|
|
/// # # $ ^ CommonCode = 0x4FEA13400 (index 10399732) ///
|
|
|
|
|
|
/// ~ ~ ^ ///
|
|
|
|
|
|
/// @ % % ///
|
|
|
|
|
|
/// ///
|
|
|
|
|
|
/// 10399732 => 9 * (32 ^ 4) + 29 * (32 ^ 3) + 11 * (32 ^ 2) + 31 * (32 ^ 1) + 20 * (32 ^ 0) ///
|
|
|
|
|
|
/// => (9), (29), (11), (31), (20) ///
|
|
|
|
|
|
/// => `A`, `X`, `C`, `Z`, `N` ///
|
|
|
|
|
|
/// => "AXCZN" ///
|
|
|
|
|
|
/// -------------------------------------------------------------------------------------------- ///
|
|
|
|
|
|
|
|
|
#include <string> |
|
|
#include <string> |
|
|
#include <cstdint> |
|
|
#include <cstdint> |
|
|
#include <ostream> |
|
|
#include <ostream> |
|
|
#include <optional> |
|
|
#include <optional> |
|
|
|
|
|
#include "all_cases.h" |
|
|
|
|
|
|
|
|
namespace klotski { |
|
|
namespace klotski { |
|
|
namespace codec { |
|
|
namespace codec { |
|
|
|
|
|
|
|
|
const uint32_t SHORT_CODE_LIMIT = 29334498; |
|
|
constexpr uint32_t SHORT_CODE_LIMIT = cases::ALL_CASES_NUM_; |
|
|
|
|
|
|
|
|
class CommonCode; |
|
|
class CommonCode; |
|
|
class ShortCode { |
|
|
class ShortCode { |
|
|
public: |
|
|
public: |
|
|
static void speed_up(bool fast_mode); |
|
|
|
|
|
explicit operator uint32_t() const noexcept; |
|
|
explicit operator uint32_t() const noexcept; |
|
|
static bool check(uint32_t short_code) noexcept; |
|
|
static bool check(uint32_t short_code) noexcept; |
|
|
|
|
|
static void speed_up(bool fast_mode = false) noexcept; |
|
|
friend std::ostream& operator<<(std::ostream &out, ShortCode self); |
|
|
friend std::ostream& operator<<(std::ostream &out, ShortCode self); |
|
|
|
|
|
|
|
|
[[nodiscard]] uint32_t unwrap() const noexcept; |
|
|
[[nodiscard]] uint32_t unwrap() const noexcept; |
|
@ -90,10 +106,11 @@ public: |
|
|
|
|
|
|
|
|
private: |
|
|
private: |
|
|
uint32_t code_; |
|
|
uint32_t code_; |
|
|
static bool fast_available_; |
|
|
static bool fast_available_; // TODO: try to remove it
|
|
|
|
|
|
|
|
|
static uint64_t fast_decode(uint32_t short_code) noexcept; |
|
|
static uint64_t fast_decode(uint32_t short_code) noexcept; |
|
|
static uint32_t fast_encode(uint64_t common_code) noexcept; |
|
|
static uint32_t fast_encode(uint64_t common_code) noexcept; |
|
|
|
|
|
|
|
|
static uint64_t tiny_decode(uint32_t short_code) noexcept; |
|
|
static uint64_t tiny_decode(uint32_t short_code) noexcept; |
|
|
static uint32_t tiny_encode(uint64_t common_code) noexcept; |
|
|
static uint32_t tiny_encode(uint64_t common_code) noexcept; |
|
|
|
|
|
|
|
@ -101,7 +118,7 @@ private: |
|
|
static std::optional<uint32_t> string_decode(const std::string &short_code) noexcept; |
|
|
static std::optional<uint32_t> string_decode(const std::string &short_code) noexcept; |
|
|
}; |
|
|
}; |
|
|
|
|
|
|
|
|
/// CommonCode compare implements.
|
|
|
/// ShortCode compare implements.
|
|
|
inline bool operator==(uint32_t s1, ShortCode s2) noexcept { |
|
|
inline bool operator==(uint32_t s1, ShortCode s2) noexcept { |
|
|
return s1 == s2.unwrap(); |
|
|
return s1 == s2.unwrap(); |
|
|
} |
|
|
} |
|
@ -133,12 +150,12 @@ inline ShortCode ShortCode::unsafe_create(uint32_t short_code) noexcept { |
|
|
return *reinterpret_cast<ShortCode*>(&short_code); // init directly
|
|
|
return *reinterpret_cast<ShortCode*>(&short_code); // init directly
|
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
/// CommonCode create with valid check.
|
|
|
/// ShortCode create with valid check.
|
|
|
inline std::optional<ShortCode> ShortCode::create(uint32_t short_code) noexcept { |
|
|
inline std::optional<ShortCode> ShortCode::create(uint32_t short_code) noexcept { |
|
|
if (ShortCode::check(short_code)) { |
|
|
if (!ShortCode::check(short_code)) { |
|
|
return ShortCode::unsafe_create(short_code); |
|
|
return std::nullopt; |
|
|
} |
|
|
} |
|
|
return std::nullopt; |
|
|
return ShortCode::unsafe_create(short_code); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
/// Output string encoding of ShortCode.
|
|
|
/// Output string encoding of ShortCode.
|
|
|