C实现与 uint64_t 相同功能的类

实现与 uint64_t 相同的类,如果平台不支持 uint64_t 的话,可以代替之。
目前只完成部分功能,其他功能敬请期待。

uint64.hpp

#include <endian.h>
#include <cstdint>
#include <type_traits>
#include <array>

#define MC_BEGIN_NAMESPACE namespace mc {
#define MC_END_NAMESPACE }

MC_BEGIN_NAMESPACE

#if __BYTE_ORDER == __BIG_ENDIAN
struct maybe_big_endian : std::true_type {};
#elif __BYTE_ORDER == __LITTLE_ENDIAN
struct maybe_big_endian : std::false_type {};
#else
#error "Endianness not defined!"
#endif

template<typename Array, bool>
struct uint64_data : public Array
{
protected:
 uint32_t& first() { return (*this)[0]; }
 uint32_t& second() { return (*this)[1]; }
 uint32_t first() const { return (*this)[0]; }
 uint32_t second() const { return (*this)[1]; }
};

template<typename Array>
struct uint64_data<Array, true> : public Array
{
protected:
 uint32_t& first() { return (*this)[1]; }
 uint32_t& second() { return (*this)[0]; }
 uint32_t first() const { return (*this)[1]; }
 uint32_t second() const { return (*this)[0]; }
};

class uint64 : public uint64_data
<std::array<uint32_t, 2>, maybe_big_endian::value>
{
public:
 uint64() = default;
 //explicit
 uint64(uint32_t v);
 uint64(const uint64& o);
 ~uint64() = default;
 uint64& operator+=(const uint64& v) noexcept;
 uint64& operator<<=(unsigned int n) noexcept;
 uint64& operator>>=(unsigned int n) noexcept;
 operator uint32_t() { return first(); }
 friend void swap(uint64& l, uint64& r);
};

inline uint64 operator+(const uint64& l, const uint64& r)
{ auto tmp = l; return tmp += r; }

inline uint64 operator>>(const uint64& l, unsigned int n)
{ auto tmp = l; return tmp >>= n; }

inline uint64 operator<<(const uint64& l, unsigned int n)
{ auto tmp = l; return tmp <<= n; }

MC_END_NAMESPACE

uint64.cpp

#include "uint64.hpp"

MC_BEGIN_NAMESPACE

uint64::uint64(uint32_t v)
{
 first() = v;
 second() = 0u;
}

uint64::uint64(const uint64& o)
{
 *this = o;
}

uint64& uint64::operator+=(const uint64& o) noexcept
{
 second() += o.second(); // 先计算 second,预防 (this == &o) 的情况
 uint32_t old = first();
 if ((first() += o.first()) < old) {
  ++second();
 }
 return *this;
}

uint64& uint64::operator<<=(unsigned int n) noexcept
{
 if (n < 32) {
  second() = (second() << n) | (first() >> (32 - n));
  first() <<= n;
 } else if (n < 64) {
  second() = first() << (n - 32);
  first() = 0u;
 } else /*if (n >= 64)*/ {
  second() = first() = 0u;
 }
 return *this;
}

uint64& uint64::operator>>=(unsigned int n) noexcept
{
 if (n < 32) {
  first() = (first() >> n) | (second() << (32 - n));
  second() >>= n;
 } else if (n < 64) {
  first() = second() >> (n - 32);
  second() = 0u;
 } else /*if (n >= 64)*/ {
  second() = first() = 0u;
 }
 return *this;
}

void swap(uint64& l, uint64& r)
{
 if (&l != &r) {
  auto tmp  = l.first();
  l.first() = r.first();
  r.first() = tmp;
  tmp    = l.second();
  l.second() = r.second();
  r.second() = tmp;
 }
}

MC_END_NAMESPACE

test.cpp

#include <cstdint>
#include <cstdio>
#include "uint64.hpp"

#if 1
 typedef mc::uint64 U64;
 inline void ptype() {std::printf("使用 mc::uint64\n");}
#else
 typedef std::uint64_t U64;
 inline void ptype() {std::printf("使用 std::uint64_t\n");}
#endif

void frm(const char* str) {
 std::printf("%20s", str);
}

void data_hex(const U64& v) {
 const uint8_t* p = (const uint8_t*)&v;
 for (int i = 0; i < 8; ++i) {
  if (i == 4) std::printf(" ");
  std::printf("%02x", p[i]);
 }
 std::printf("\n");
}

void test() {
 uint32_t v = 0xffffffff;
 U64 a = v;
 frm("(a = 0xffffffff) => ");
 data_hex(a);

 frm("(a >>= 1) => ");
 data_hex(a >>= 1);

 a = v;
 frm("(a <<= 1) => ");
 data_hex(a <<= 1);

 a = v;
 frm("(a += a) => ");
 data_hex(a += a);
}

int main() {
 ptype();
 if (mc::maybe_big_endian::value) {
  std::printf("主机字节序是 big-endian\n");
 } else {
  std::printf("主机字节序是 little-endian\n");
 }
 for (int i = 0; i < 20; ++i)
  std::printf(" ");
 if (mc::maybe_big_endian::value)
  std::printf("H <<<< L H <<<< L\n");
 else
  std::printf("L >>>> H L >>>> H\n");
 test();
 return 0;
}

功能还在逐步完善中,小伙伴们记得关注。

(0)

相关推荐

  • 浅谈C语言编程中的布尔bool数据类型

    我们知道在C++里有专门的bool类型,用来表示真或假.但是在C语言里没有这样的类型(至少我是一直这么认为的),表达式的值0为假,非0为真.所以条件判断语句( if(-).while(-) )非常灵活,甚至一个指针类型都可以是条件表达式. 为了使程序更清晰,我们常常会给出如下的宏定义: typedef int BOOL; #define TRUE 1 #define FALSE 0 这是最常见的写法,能被任何C语言编译器认可. 今天我在一段程序里看见这么一行 #include ,这个陌生的头文件

  • 深入解析C语言中常数的数据类型

    废话不多说,上代码 复制代码 代码如下: //编译环境:codeblocks+gcc#include <stdio.h>#include <stdint.h>int Fun(){    uint64_t y;    uint32_t x1, x2; //y = 3000 * 24000000 / 1000;//常数默认作为32位数据,临时运算结果也是32位,溢出错误    //y = (uint64_t)3000 * (uint64_t)24000000 / 1000;//常数强制

  • C/C++与Java各数据类型所占字节数的详细比较

    C/C++的数据类型: 一,整型 Turbo C:   [signed] int 2Byte//有符号数,-32768~32767   unsigned int 2Byte //无符号数,只能表示整数0~65535 [signed] short [int] 2Byte unsigned short [int] 2 Byte long [int] 4 Byte unsigned long [int] 4 Byte Visual C++ 6.0: [signed] int 4Byte   unsig

  • java和c/c++ 数据类型长度的比较

    1. c语言中的整数类型有char, short, int, long等几种, 下面是C语言对每种数据类型长度的规定: (a). short和long类型的长度不相同 (b). int类型通常同具体机器的物理字长相同 (c). short通常是16bits, int通常是16bits or 32bits每种编译器可以根据硬件的不同自由确定, 但是short和int必须最少是16bits, 而long类型必须最少是32bits, 并且short必须比int和long类型要短. 2. sizeof(

  • C实现与 uint64_t 相同功能的类

    实现与 uint64_t 相同的类,如果平台不支持 uint64_t 的话,可以代替之. 目前只完成部分功能,其他功能敬请期待. uint64.hpp #include <endian.h> #include <cstdint> #include <type_traits> #include <array> #define MC_BEGIN_NAMESPACE namespace mc { #define MC_END_NAMESPACE } MC_BEGI

  • ASP实现类似hashMap功能的类

    java中的hashMap存取数据非常方便,可惜ASP中没有类似的类.作者在开发程序中需要类似的数据类型,于是构造了一个能基本类似hashMap功能的类,可以实现键值存取操作等,存取的数据可以为ASP 中的任何基本类型. 下面是程序的代码,贴到一个空的ASP中可以直接运行. <% set jb51 = new Jb '给mp对象赋值 jb51.putv "a","jb51.net" jb51.putv "b","www.jb51.

  • Android开发实现的IntentUtil跳转多功能工具类【包含视频、音频、图片、摄像头等操作功能】

    本文实例讲述了Android开发实现的IntentUtil跳转多功能工具类.分享给大家供大家参考,具体如下: 说明:此工具类是本人开发中总结下来的,还有其它的跳转亲给我留言,希望大家一起把这个工具类打造成最全的跳转工具,谢谢! package com.android.chat.utils; import java.io.File; import java.io.Serializable; import android.app.Activity; import android.content.Co

  • PHP发送邮件确认验证注册功能示例【修改别人邮件类】

    本文实例讲述了PHP发送邮件确认验证注册功能.分享给大家供大家参考,具体如下: 类库: require "class.phpmailer.php"; require "class.smtp.php"; class PHP_Mailer { protected $mail; public function __construct() { $mail = new PHPMailer; $mail->SMTPDebug = 3; $mail->isSMTP()

  • 使用 Javascript 实现浏览器推送提醒功能的示例

    本篇文章内容简单,速读只需两三分钟,通过这两三分钟的时间你就可以给自己的网站实现推送提醒的功能 Notification 类 简单明了,这个类就是负责推送消息的,只要用户当前没有关闭页面,及时是在使用其他程序,浏览器也能够将消息推送给用户 请求权限 我们在手机上都收到过消息推送,在接收推送之前我们会先将消息推送权限开放给应用.在浏览器中也一样,在使用浏览器推送之前,需要先获取权限 Notification.requestPermission().then(permission => { cons

  • Mootools 1.2教程 类(一)

    简单地讲,一个类就是一个容器,这个容器包含了一些变量集合和操作这些变量的函数,以便实现特定的功能.在一个内容牵涉较多的项目中,类会显得难以置信的有用. 变量 在前面一系列的课程中,我们已经学习过了Hash对象中键值对(key/value pair)的使用方式,因此,下面的这个例子中创建了一个类,它只包含了一些变量,你看起来可能会觉得非常的熟悉: 参考代码: 复制代码 代码如下: // 创建一个名为class_one的类 // 包含两个内部变量 var Class_one = new Class(

  • Servlet+JavaBean+JSP打造Java Web注册与登录功能

    采用Java Web所实现的MVC结构图如下,其中控制器部分采用Servlet来实现,模型部分采用JavaBean来实现,而大部分的视图采用Jsp页面来实现. 思想基础 JSP+JavaBean两层结构工作原理应该是比较熟悉的,也比较好理解. 但是有一点必须要清楚就是用户通过浏览器来发送网页的请求,此请求到达服务器后在服务器端查找对应的网页,如果是首次请求(第二次就不用解释执行了),对于JSP来说要生成Servlet,然后通过Servlet引擎来执行 Servlet,把调用JavaBean的结果

  • Java多线程基础——Lock类

    之前已经说道,JVM提供了synchronized关键字来实现对变量的同步访问以及用wait和notify来实现线程间通信.在jdk1.5以后,JAVA提供了Lock类来实现和synchronized一样的功能,并且还提供了Condition来显示线程间通信. Lock类是Java类来提供的功能,丰富的api使得Lock类的同步功能比synchronized的同步更强大.本文章的所有代码均在Lock类例子的代码 本文主要介绍一下内容: Lock类 Lock类其他功能 Condition类 Con

  • Java案例之随机验证码功能实现实例

    实现的功能比较简单,就是随机产生了四个字符然后输出.效果图如下,下面我会详细说一下实现这个功能用到了那些知识点,并且会把 这些知识点详细的介绍出来.哈哈 ,大神勿喷,对于初学Java的人帮助应该蛮大的.嘿嘿. 先上效果图: 源代码: RandomGen.java(实现产生验证码功能的类) package verificationcode; import java.util.Random; public class RandomGen { //生成四位不重复的验证码 public static S

  • python类中super()和__init__()的区别

    单继承时super()和__init__()实现的功能是类似的 class Base(object): def __init__(self): print 'Base create' class childA(Base): def __init__(self): print 'creat A ', Base.__init__(self) class childB(Base): def __init__(self): print 'creat B ', super(childB, self).__

随机推荐