|
using System;
using Pfz.Extensions.MonitorLockExtensions;
namespace Pfz.Threading
{
/// <summary>
/// Class that returns the value of DateTime.Now.Ticks when GetValue is called
/// but also guarantees that two calls will generate different values, even if
/// they are done one just after the other.
/// </summary>
public static class TicksOrIncrement
{
private static readonly object _lock = new object();
private static long _lastValue = DateTime.Now.Ticks;
private static readonly Random _random = new Random();
private static int _randomIncrement = 1000;
/// <summary>
/// Gets or sets the random value used as the increment factor when
/// two calls are made in the same microsecond.
/// </summary>
public static int RandomIncrement
{
get
{
return _randomIncrement;
}
set
{
if (value < 1)
throw new ArgumentOutOfRangeException("value", "The minimum accepted value is 1.");
_randomIncrement = value;
}
}
private static int _totalProcesses = 1;
/// <summary>
/// Gets or sets a value that is seen as the total number of processes using
/// TicksOrIncrement. So, together with ProcessId, values are guaranteed to
/// be exclusive.
/// </summary>
public static int TotalProcesses
{
get
{
return _totalProcesses;
}
set
{
if (value < 1)
throw new ArgumentOutOfRangeException("value", "TotalProcesses must be at least 1.");
_totalProcesses = value;
}
}
private static int _processId;
/// <summary>
/// Gets or sets the processid used to guarantee that never two process using
/// TicksOrIncrement will get the same value.
/// </summary>
public static int ProcessId
{
get
{
return _processId;
}
set
{
if (value < 0 || value >= _totalProcesses)
throw new ArgumentOutOfRangeException("value", "ProcessId must be >=0 and <=TotalProcesses.");
_processId = value;
}
}
/// <summary>
/// Gets a new DateTime.Now.Ticks value or a random incremented value if
/// this is a call done at the same microsecond of the last one.
/// </summary>
public static long GetNext()
{
long ticks = ((DateTime.Now.Ticks / _totalProcesses) * _totalProcesses ) + _processId;
_lock.UnabortableLock
(
delegate
{
if (ticks <= _lastValue)
{
int addValue = (1 + _random.Next(_randomIncrement)) * _totalProcesses;
ticks = (((_lastValue + addValue) / _totalProcesses) * _totalProcesses) + _processId;
}
_lastValue = ticks;
}
);
return ticks;
}
}
}
|
By viewing downloads associated with this article you agree to the Terms of Service and the article's licence.
If a file you wish to view isn't highlighted, and is a text file (not binary), please
let us know and we'll add colourisation support for it.
I started to program computers when I was 11 years old, as a hobbyist, programming in AMOS Basic and Blitz Basic for Amiga.
At 12 I had my first try with assembler, but it was too difficult at the time. Then, in the same year, I learned C and, after learning C, I was finally able to learn assembler (for Motorola 680x0).
Not sure, but probably between 12 and 13, I started to learn C++. I always programmed "in an object oriented way", but using function pointers instead of virtual methods.
At 15 I started to learn Pascal at school and to use Delphi. At 16 I started my first internship (using Delphi). At 18 I started to work professionally using C++ and since then I've developed my programming skills as a professional developer in C++ and C#, generally creating libraries that help other developers do their work easier, faster and with less errors.
Want more info or simply want to contact me?
Take a look at:
http://paulozemek.azurewebsites.net/
Or e-mail me at: paulozemek@outlook.com
Codeproject MVP 2012, 2015 & 2016
Microsoft MVP 2013-2014 (in October 2014 I started working at Microsoft, so I can't be a Microsoft MVP anymore).