Timestamp.cpp详解[1-2-2]

#include <muduo/base/Timestamp.h>

#include <sys/time.h>
#include <stdio.h>
#define __STDC_FORMAT_MACROS
#include <inttypes.h>
#undef __STDC_FORMAT_MACROS

#include <boost/static_assert.hpp>

using namespace muduo;

BOOST_STATIC_ASSERT(sizeof(Timestamp) == sizeof(int64_t));


/*
以微秒初始化microSecondsSinceEpoch_
*/
Timestamp::Timestamp(int64_t microseconds)
  : microSecondsSinceEpoch_(microseconds)
{
}


/*
将微秒microSecondsSinceEpoch_ 转换为[秒seconds +  微秒microseconds]
*/
string Timestamp::toString() const
{
  char buf[32] = {0};
  int64_t seconds = microSecondsSinceEpoch_ / kMicroSecondsPerSecond;
  int64_t microseconds = microSecondsSinceEpoch_ % kMicroSecondsPerSecond;
  snprintf(buf, sizeof(buf)-1, "%" PRId64 ".%06" PRId64 "", seconds, microseconds);
  return buf;
}


/*
将微秒microSecondsSinceEpoch_ 转换为[time_t]
http://blog.csdn.net/csuwzc/article/details/6912104
再利用gmtime_r 将time_t 时间做转换存于 tm_time
*/
string Timestamp::toFormattedString() const
{
  char buf[32] = {0};
  time_t seconds = static_cast<time_t>(microSecondsSinceEpoch_ / kMicroSecondsPerSecond);
  int microseconds = static_cast<int>(microSecondsSinceEpoch_ % kMicroSecondsPerSecond);
  struct tm tm_time;
  gmtime_r(&seconds, &tm_time);

  snprintf(buf, sizeof(buf), "%4d%02d%02d %02d:%02d:%02d.%06d",
      tm_time.tm_year + 1900, tm_time.tm_mon + 1, tm_time.tm_mday,
      tm_time.tm_hour, tm_time.tm_min, tm_time.tm_sec,
      microseconds);
  return buf;
}


/*
将当前时间以微秒存Timestamp类的变量microSecondsSinceEpoch 中;
调用explicit Timestamp(int64_t) 返回
*/
Timestamp Timestamp::now()
{
  struct timeval tv;
  gettimeofday(&tv, NULL);
  int64_t seconds = tv.tv_sec;
  return Timestamp(seconds * kMicroSecondsPerSecond + tv.tv_usec);
}


/*
将0 存Timestamp类的变量microSecondsSinceEpoch 中;
调用Timestamp() 返回
*/
Timestamp Timestamp::invalid()
{
  return Timestamp();
}

【1】Boost学习笔记 BOOST_STATIC_ASSERT

Boost中提供了一个编译时断言宏 BOOST_STATIC_ASSERT ,这个宏很简单,但里面也有一些值得学习的地方。
首先来看看真正做事的代码
template  <bool x> struct STATIC_ASSERTION_FAILURE;
 
template  <> struct STATIC_ASSERTION_FAILURE<true> { enum { value = 1 }; };
 
template <int x> struct static_assert_test{};
注意这里定义了一个模板STATIC_ASSERTION_FAILURE;利用模板的偏特化对true的时候给出了一个特化版本,而对于当false的时候并没有定义。这就是实现编译时断言的关键所在。其实你可以直接使用
STATIC_ASSERTION_FAILURE; 但是直接使用不方便。应为它是一个类。
STATIC_ASSERTION_FAILURE<false> ; 如果想这样它是不会断言失败的。你必须
STATIC_ASSERTION_FAILURE<false> obj; 要明确定义一个对象编译器才会明确告诉你,你断言错了。而且这样的模板形式显然不符合我们使用断言宏的习惯。所以boost给出了一个方便的宏的形式。还有个问题
template <int x> struct static_assert_test{}; 有什么用?一会儿就会看到。
下面看看平时真正使用的那个宏
#define  BOOST_STATIC_ASSERT( B ) /
   typedef ::boost::static_assert_test</
      sizeof(::boost::STATIC_ASSERTION_FAILURE< (bool)( B ) >)>/
         BOOST_JOIN(boost_static_assert_typedef_, __COUNTER__)
这个宏是针对VC的,关于这个一会儿再解释。
首先当然从最里面开始剖析sizeof(::boost::STATIC_ASSERTION_FAILURE< (bool)( B ) >),首先要知道sizeof()实际上是工作再编译期的,它会检查::boost::STATIC_ASSERTION_FAILURE< (bool)( B ) >的大小,如果B = false,由于没有定义关于false的特化版本,sizeof()不能取得它的大小所以不能编译通过。还要注意一点这里用的typedeftypedef也是再编译是工作的,而且在运行是不占用资源。
然后看最后一句 BOOST_JOIN(boost_static_assert_typedef_, __COUNTER__) ,也就是被定义的类型名。
在这里的作用就是boost_static_assert_typedef_##__COUNTER__也就是
boost_static_assert_typedef___COUNTER__ 。注意了,这里__COUNTER_其实也是一个宏,一个VC专有的宏,每使用它一次它就会增加1,比如说你说第10次使用它,那么它就会变成
boost_static_assert_typedef_9( 最开始___COUNTER__为0),但这又有什么用呢,编译错误时它可以告诉你是第几个BOOST_STATIC_ASSERT出了问题,但在我的编译器上显然不做不到,应为我的编译器会直接找出问题的关键所在:显示:STATIC_ASSERTION_FAILURE出错。对于其他编译器这里用的时__LINE__,这个标准宏,它表示在文件中所在的行。
最后有一点就是BOOST_STATIC_ASSERT是放在全局命名空间中的,所以使用的时候不用加boost::限定名。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值