我需要使用最小数量的字节/字符打包带有UTC日期时间的字符串。我只需要精确到秒。使用.NET 4.0,最节省空间的方法是什么呢?蜱虫似乎并不那么小。
所有想法都赞赏。 感谢。
编辑:感谢Joel Coehoorn,打包/拆包是最好的。谢谢!这是一些证据:DateTimeOffset nowStamp = DateTimeOffset.UtcNow;
Console.WriteLine( nowStamp.ToString() ); // 9/9/2011 2:17:17 PM +00:00
Console.WriteLine( nowStamp.ToString( "u" ) ); // 2011-09-09 14:17:17Z
Console.WriteLine( nowStamp.Ticks.ToString() ); // 634511746376767889
Console.WriteLine( PackDate( nowStamp ) ); // 7R9qTgAAAAA=
Console.WriteLine( UnpackDate( PackDate( nowStamp ) ) ); // 9/9/2011 2:17:17 PM +00:00
答案 0 :(得分:2)
也许是unix时间(1970年1月1日以来的秒而不是毫秒)的变体base64编码。
//Helpers
private static DateTime Jan1st1970 = new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc);
public static long toUnixTime(this DateTime d)
{
return (long)((d.ToUniversalTime() - Jan1st1970).TotalMilliseconds);
}
public static string Base64Encode(long toEncode)
{
return Convert.ToBase64String(BitConverter.GetBytes(toEncode));
}
//Encode
public static string PackDate(DateTime toPack)
{
return Base64Encode(toPack.toUnixTime()/1000);
}
//Decode
public static DateTime UnpackDate(string toUnpack)
{
long time = BitConverter.ToInt64(Convert.FromBase64String(toUnpack),0);
return Jan1st1970.AddSeconds(time); //you may or may not want a "ToLocaltime()" call here.
}
请注意,所有这些都是在没有IDE的帮助下完成的 - 上面可能存在一两个错误。但它应该让你开始。
这应该会产生一个固定宽度的字符串。由于我们只做了几秒而不是几毫秒,你可能会发现你总是在结果中有一些你不需要的额外填充。你甚至可以使用int而不是long,这会将字符串切成两半。但要小心剥去那个填充物,因为越接近1970年,数字越小,但越大,越大,你就越需要它。您需要确定,您的日期值将适合新的较小范围进行任何修剪。例如,当前日期适合在int中,但即使28年后也不会。 UInt32将为您带来更远的未来,但阻止您在1970年之前使用日期。
答案 1 :(得分:0)
如果你需要保存一些字节,并且确定日期时间限制,那么这个解决方案就可以了:
internal class Program
{
private static DateTime _lbound = new DateTime(2011, 1, 1).ToUniversalTime();
private static DateTime _ubound = new DateTime(2013, 1, 1).ToUniversalTime();
private static int Pack(DateTime utcTime)
{
var totalSeconds = (_ubound - _lbound).TotalSeconds;
return (int) (utcTime - _lbound).TotalSeconds;
}
private static DateTime Unpack(int packedTime)
{
return _lbound.AddSeconds(packedTime);
}
private static void Check(DateTime time)
{
var unpacked = Unpack(Pack(time));
var areEquals = Math.Abs((time - unpacked).TotalSeconds) < 1.0;
Console.WriteLine("Verify: {0} - {1}", time, areEquals);
}
static void Main(string[] args)
{
Check(_lbound);
Check(_ubound);
Check(DateTime.UtcNow);
}
}
它将适合时间表示,在4个字节(int)中定义的时间范围(从2011年到2013年)具有1秒精度。但是,从维护的角度来看,IMO真的很糟糕。