A simple stop watch which I want to extendSimple stop watchStop Watch applicationStop Watch Application 2.0A thread-safe stop watch utility classTimer stop watch in AngularJSA low tech approach to measuring game speedSimple timer start and stop from consoleExtremely simple timer class in C++A simple Java class for implementing code stop watchesA simple Java class for implementing code stop watches - follow-up

Why does radiocarbon dating only work in nonliving creatures?

Ideal Firearms for time travel or low tech universe exploration

Conversion of space characters into space tokens

Two super-button calculator

Stop autocorrect from autocorrecting a word even if I re-type it after it has been autocorrected

Does the sun cross other spiral arms in its movement around the galaxy's center?

Are there any (natural) scientists in Middle-earth?

Heavy condensation inside car during winter. Tried multiple things, but no results!

What to do if caught in a physical pentest?

What is the last point where one can throw away fruits if one has indicated "not bringing any fruit" on the US customs form when flying to the US?

Expand a recursive pattern

What does Darth Vader think Obi-Wan's referring to when Obi says "If you strike me down..."

Where and/or why is a slanted hyphen used?

A robot surviving on top of a 3x3 platform

Can the jeopardy of being judged be fought against in court?

What does AI software look like, and how is it different from other software?

How to translate "cocotte en papier" in English?

Why can "bubo" ("owl") be feminine or masculine?

Intersection of four circles

How do gelatinous cubes reproduce?

What color is a wolf's coat?

White marks on garage ceiling

Why do new jet engines cost billions to design?

Why do some applications have files with no extension?



A simple stop watch which I want to extend


Simple stop watchStop Watch applicationStop Watch Application 2.0A thread-safe stop watch utility classTimer stop watch in AngularJSA low tech approach to measuring game speedSimple timer start and stop from consoleExtremely simple timer class in C++A simple Java class for implementing code stop watchesA simple Java class for implementing code stop watches - follow-up






.everyoneloves__top-leaderboard:empty,.everyoneloves__mid-leaderboard:empty,.everyoneloves__bot-mid-leaderboard:empty
margin-bottom:0;









2














$begingroup$


I am making a very simple stopwatch functionality as below.



New will start the timer.

Pause will suspend the watch.

Start will restart again.

Session will return the duration.



#include<chrono>
#include<vector>

class Timer
using cppTimer = std::chrono::high_resolution_clock;
using seconds = std::chrono::seconds;
std::vector< int64_t > mvSession;
cppTimer::time_point mtpNow;
cppTimer::time_point mtPause;
seconds mtRestart;
bool mbIsPause;
public:

enum Units
sec,
min,
hrs,
end
;

Timer() noexcept: mbIsPause(false), mtRestart ,mvSession(3)


inline void Now() noexcept
mtpNow = cppTimer::now();
mbIsPause = false;
mtRestart = seconds(0);


inline void Pause() noexcept
if(!mbIsPause)
mtPause = cppTimer::now();
mbIsPause = true;




inline void Start() noexcept
if(mbIsPause)
mtRestart += std::chrono::duration_cast<std::chrono::seconds>(cppTimer::now() - mtPause);
mbIsPause = false;





std::vector< int64_t > SessionTime()

auto now = cppTimer::now();
auto delay = std::chrono::duration_cast<std::chrono::seconds>(now - mtpNow);

if(mbIsPause)
mtRestart += std::chrono::duration_cast<std::chrono::seconds>(cppTimer::now() - mtPause);


auto tp = std::chrono::duration_cast<std::chrono::seconds>(delay - mtRestart); //+ mtRestart ;// SecondsTp(cppTimer::now()) - SecondsTp(mtpNow) + SecondsTp(mtRestart);

mvSession[sec] = tp.count()%60;
mvSession[min] = std::chrono::duration_cast<std::chrono::minutes>(tp).count()%60;
mvSession[hrs] = std::chrono::duration_cast<std::chrono::hours>(tp).count();

return mvSession;


;









share|improve this question












$endgroup$





















    2














    $begingroup$


    I am making a very simple stopwatch functionality as below.



    New will start the timer.

    Pause will suspend the watch.

    Start will restart again.

    Session will return the duration.



    #include<chrono>
    #include<vector>

    class Timer
    using cppTimer = std::chrono::high_resolution_clock;
    using seconds = std::chrono::seconds;
    std::vector< int64_t > mvSession;
    cppTimer::time_point mtpNow;
    cppTimer::time_point mtPause;
    seconds mtRestart;
    bool mbIsPause;
    public:

    enum Units
    sec,
    min,
    hrs,
    end
    ;

    Timer() noexcept: mbIsPause(false), mtRestart ,mvSession(3)


    inline void Now() noexcept
    mtpNow = cppTimer::now();
    mbIsPause = false;
    mtRestart = seconds(0);


    inline void Pause() noexcept
    if(!mbIsPause)
    mtPause = cppTimer::now();
    mbIsPause = true;




    inline void Start() noexcept
    if(mbIsPause)
    mtRestart += std::chrono::duration_cast<std::chrono::seconds>(cppTimer::now() - mtPause);
    mbIsPause = false;





    std::vector< int64_t > SessionTime()

    auto now = cppTimer::now();
    auto delay = std::chrono::duration_cast<std::chrono::seconds>(now - mtpNow);

    if(mbIsPause)
    mtRestart += std::chrono::duration_cast<std::chrono::seconds>(cppTimer::now() - mtPause);


    auto tp = std::chrono::duration_cast<std::chrono::seconds>(delay - mtRestart); //+ mtRestart ;// SecondsTp(cppTimer::now()) - SecondsTp(mtpNow) + SecondsTp(mtRestart);

    mvSession[sec] = tp.count()%60;
    mvSession[min] = std::chrono::duration_cast<std::chrono::minutes>(tp).count()%60;
    mvSession[hrs] = std::chrono::duration_cast<std::chrono::hours>(tp).count();

    return mvSession;


    ;









    share|improve this question












    $endgroup$

















      2












      2








      2


      2



      $begingroup$


      I am making a very simple stopwatch functionality as below.



      New will start the timer.

      Pause will suspend the watch.

      Start will restart again.

      Session will return the duration.



      #include<chrono>
      #include<vector>

      class Timer
      using cppTimer = std::chrono::high_resolution_clock;
      using seconds = std::chrono::seconds;
      std::vector< int64_t > mvSession;
      cppTimer::time_point mtpNow;
      cppTimer::time_point mtPause;
      seconds mtRestart;
      bool mbIsPause;
      public:

      enum Units
      sec,
      min,
      hrs,
      end
      ;

      Timer() noexcept: mbIsPause(false), mtRestart ,mvSession(3)


      inline void Now() noexcept
      mtpNow = cppTimer::now();
      mbIsPause = false;
      mtRestart = seconds(0);


      inline void Pause() noexcept
      if(!mbIsPause)
      mtPause = cppTimer::now();
      mbIsPause = true;




      inline void Start() noexcept
      if(mbIsPause)
      mtRestart += std::chrono::duration_cast<std::chrono::seconds>(cppTimer::now() - mtPause);
      mbIsPause = false;





      std::vector< int64_t > SessionTime()

      auto now = cppTimer::now();
      auto delay = std::chrono::duration_cast<std::chrono::seconds>(now - mtpNow);

      if(mbIsPause)
      mtRestart += std::chrono::duration_cast<std::chrono::seconds>(cppTimer::now() - mtPause);


      auto tp = std::chrono::duration_cast<std::chrono::seconds>(delay - mtRestart); //+ mtRestart ;// SecondsTp(cppTimer::now()) - SecondsTp(mtpNow) + SecondsTp(mtRestart);

      mvSession[sec] = tp.count()%60;
      mvSession[min] = std::chrono::duration_cast<std::chrono::minutes>(tp).count()%60;
      mvSession[hrs] = std::chrono::duration_cast<std::chrono::hours>(tp).count();

      return mvSession;


      ;









      share|improve this question












      $endgroup$




      I am making a very simple stopwatch functionality as below.



      New will start the timer.

      Pause will suspend the watch.

      Start will restart again.

      Session will return the duration.



      #include<chrono>
      #include<vector>

      class Timer
      using cppTimer = std::chrono::high_resolution_clock;
      using seconds = std::chrono::seconds;
      std::vector< int64_t > mvSession;
      cppTimer::time_point mtpNow;
      cppTimer::time_point mtPause;
      seconds mtRestart;
      bool mbIsPause;
      public:

      enum Units
      sec,
      min,
      hrs,
      end
      ;

      Timer() noexcept: mbIsPause(false), mtRestart ,mvSession(3)


      inline void Now() noexcept
      mtpNow = cppTimer::now();
      mbIsPause = false;
      mtRestart = seconds(0);


      inline void Pause() noexcept
      if(!mbIsPause)
      mtPause = cppTimer::now();
      mbIsPause = true;




      inline void Start() noexcept
      if(mbIsPause)
      mtRestart += std::chrono::duration_cast<std::chrono::seconds>(cppTimer::now() - mtPause);
      mbIsPause = false;





      std::vector< int64_t > SessionTime()

      auto now = cppTimer::now();
      auto delay = std::chrono::duration_cast<std::chrono::seconds>(now - mtpNow);

      if(mbIsPause)
      mtRestart += std::chrono::duration_cast<std::chrono::seconds>(cppTimer::now() - mtPause);


      auto tp = std::chrono::duration_cast<std::chrono::seconds>(delay - mtRestart); //+ mtRestart ;// SecondsTp(cppTimer::now()) - SecondsTp(mtpNow) + SecondsTp(mtRestart);

      mvSession[sec] = tp.count()%60;
      mvSession[min] = std::chrono::duration_cast<std::chrono::minutes>(tp).count()%60;
      mvSession[hrs] = std::chrono::duration_cast<std::chrono::hours>(tp).count();

      return mvSession;


      ;






      c++ timer






      share|improve this question
















      share|improve this question













      share|improve this question




      share|improve this question








      edited Aug 12 at 3:42









      Peter Mortensen

      2641 silver badge7 bronze badges




      2641 silver badge7 bronze badges










      asked Aug 11 at 10:10









      MohitMohit

      505 bronze badges




      505 bronze badges























          2 Answers
          2






          active

          oldest

          votes


















          7
















          $begingroup$

          Don't use member variables for temporary storage



          Your class Timer has a member variable mvSession, which is unnecessary. It is only used in SessionTime(), where it is filled in and returned. No other functions use it. Instead, just declare std::vector<int64_t> mvSession(3) inside SessionTime().



          By making it a member variable, you introduced two problems: first, your class now uses more memory than necessary. Second, the function SessionTime() is now no longer reentrant; if two threads would call SessionTime() simultaneously, they would both write to the same mvSession variable, potentially corrupting it.



          Use std::chrono::steady_clock for timers



          The problem with std::chrono::high_resolution_clock() is that it is not guaranteed what kind of clock it is. It can represent wall clock time, which can jump ahead or forwards because of summer and winter time, leap seconds, the computer suspending and resuming. This is not something you want for a timer where you are interested in a simple duration. For this, you want to use std::chrono::steady_clock, which is guaranteed to be a monotonically increasing clock. Also, your function SessionTime() returns the time in a resolution of seconds, so you don't need a high resolution clock anyway.



          Don't use Hungarian notation



          There might be some merits to Hungarian notation, but it really isn't necessary to use it for C++, since the compiler will do type checking for you. Moreover, it's easy to use the wrong prefix, and it's hard to come up with a reasonable prefix when you can have complex types.



          You are already making mistakes in your code. For example, mtpNow and mtPause are both of type cppTimer::time_point. So the prefix should have been the same. And mtRestart has a different type than mtPause, so their prefixes should have been different. I recommend that you just avoid using Hungarian notation altoghether.



          Be consistent with using



          You are declaring using seconds = std::chrono::seconds, and use seconds in a few places, but you also use std::chrono::seconds in a lot of places. Furthermore, you also use std::chrono::minutes and std::chrono::hours, but have not declared an alias for them. In this case, I suggest you don't declare using seconds at all.



          I would keep using cppTimer though, since it basically selects which clock to use. That makes it easier to change the clock later by just changing one line of code. I would write using clock = ... though, to be consistent with the C++ library itself.



          Don't cast to seconds too early



          Instead of seconds mtRestart, use cppTimer::duration mtRestart. This will keep the accuracy of the duration to the same as the clock itself. Only cast durations to seconds or other time intervals until the last moment possible. The same goes for the calculation of delay in SessionTime(), just write:



          auto delay = now - mtpNow;


          Use nouns for variable names, verbs for function names



          A variable holds (the value of) a thing, so its name should naturally be a noun. Functions do stuff, so there names should generally be verbs. The function Now() should actually be named Start(). Your function Start() should probably be named Continue(). The function SessionTime() calculates how long the timer has been running for, so probably should be named GetDuration().



          Conversely, the variables mtPause and mtRestart should be renamed to nouns as well. They are a bit confusing. Sure, you set mtPause in the Pause() function, but it doesn't describe what the value actually means. The same goes for mtRestart. I would instead write:



          clock::time_point StartTime;
          clock::time_point PauseTime;
          clock::duration PausedDuration;
          bool IsPaused;


          Now you can rewrite the function Now() to:



          void Start() 
          StartTime = clock::now();
          IsPaused = false;
          PausedDuration = ;



          Remove mtRestart



          You are using two variables to handle the timer being paused, mtPause and mtRestart. However, you only need one. In the Pause() function, you indeed just record when this function is called. However, when restarting the timer, instead of adding the duration of being paused to mtRestart, just add that duration to mtpNow instead:



          void Start() 
          if(mbIsPause)
          mtpNow += cppTimer::now() - mtPause;
          mbIsPause = false;




          This also simplifies SessionTime():



          std::vector<int64_t> SessionTime() 
          auto end = mbIsPause ? mtPause : cppTimer::now();
          auto tp = std::chrono::duration_cast<std::chrono::seconds>(end - mtpNow);
          ...



          Also, since mtPause is only ever 0 when you didn't pause, you can use that to signal whether the timer is paused instead of having bool mbIsPause.



          Another option is @user673679's suggestion of storing only the start time and the accumulated elapsed time so far. You would then use the start time being equal to as a signal that the timer has not been started.



          Just return a std::chrono::duration



          When you want the elapsed time, I would avoid having the Timer class be responsible for converting the duration to a vector of integers representing hours, minutes and seconds. It reduces the accuracy of your timer. Instead, I would just return a std::chrono::duration, and have the caller decide whether it wants to convert that to something. It also is much more efficient than having to construct a std::vector<int64_t>.



          Try to make it behave reasonable in all situations



          One issue with your code is that it only gives reasonable results for SessionTime() if you have called Now() at least once. You didn't initialize mtpNow, and even if it was value-initialized to zero, then SessionTime() will return the time that has passed since the epoch.



          If you want the Timer to behave like it was started at construction time, then initialize mtpNow to cppTimer::now(). If you want it to behave like it was paused, then ensure both mtpNow and mtPause are initialized to the same value (I suggest just using ), and that mbIsPause is true.



          Make it work like a real stopwatch



          As already suggested by others, it helps to think of a timer as a stopwatch. A real stopwatch starts in a stopped state, showing an elapsed time of zero. Then, you can start and stop the timer mechanism with buttons. Usually, there is a separate button to reset the stopwatch to its initial state. By making the class act like something a lot of people are already familiar with, the code is easier to understand for others.



          Reworked code



          Here is an example of what the code would look like with my suggestions applied, as well as @user673679's way of storing the elapsed time between previous start and stops of the clock:



          #include <chrono>

          class Timer
          using clock = std::chrono::steady_clock;
          clock::time_point StartTime = ;
          clock::duration ElapsedTime = ;

          public:
          bool IsRunning() const
          return StartTime != clock::time_point;


          void Start()
          if(!IsRunning())
          StartTime = clock::now();



          void Stop()
          if(IsRunning())
          ElapsedTime += clock::now() - StartTime;
          StartTime = ;



          void Reset()
          StartTime = ;
          ElapsedTime = ;


          clock::duration GetElapsed()
          auto result = ElapsedTime;
          if(IsRunning())
          result += clock::now() - StartTime;

          return result;

          ;
          ```





          share|improve this answer












          $endgroup$














          • $begingroup$
            That is "system" Hungarian notation (the Microsoft Windows documentation team grossly misinterpreted the original Hungarian notation), not the original Hungarian notation (which is not about types). See e.g. Triangulation episode 277, 4 min 00 secs to 5 min 19 secs.
            $endgroup$
            – Peter Mortensen
            Aug 12 at 1:25



















          3
















          $begingroup$

          • It's fine to prefix member variables with an m, but it's probably best to avoid type prefixes (e.g. mtpNow). It makes code harder to read (you have to know what every abbreviation means), it's a pain to maintain (e.g. mtPause should probably be mtpPause to be consistent). Modern tools eliminate the need for this too (mousing over a variable in Visual Studio will tell me the exact type, not an approximation).


          • Functions defined inline in the class don't need to be declared inline.



          • The naming is very confusing:



            • Now() differs from the standard library (now() returns the current time, and is arguably still a terrible name). Functions names should be commands or questions. Perhaps Restart() or Reset() would be better.


            • Start() is also not ideal. One might expect the function to do what Now() does. I'd suggest calling it Unpause(), which makes the purpose clearer.


            • mtRestart is an odd name for the time spent paused.



          • We don't really need the Now() / Restart() function, since we can just assign a new timer to the old one to do the same thing (e.g. Timer timer; ...; timer = Timer(); // restarted!).


          • There's no reason to keep a vector in the class (we're copying it every time anyway, so we might as well just create a new vector each time).


          • We don't need a vector, since it always has 3 values, it would be neater to return a simple struct, which would allow us to give each value a name. Or...


          • The real solution is to just return an appropriate chrono:: type, and let the user worry about formatting / converting it.


          • I'd suggest naming the class Stopwatch, since that's more specific to the pausable timing functionality we need. We can actually implement this functionality based on a simpler, non-pausable Timer class.



          So overall I'd suggest something more like this (not tested properly):



          #include <chrono>

          template<class ClockT = std::chrono::high_resolution_clock>
          class Timer

          ClockT::time_point m_start;

          public:

          Timer():
          m_start(ClockT::now())




          ClockT::duration GetElapsedTime() const

          return ClockT::now() - m_start;

          ;

          template<class ClockT = std::chrono::high_resolution_clock>
          class Stopwatch

          bool m_paused;
          Timer<ClockT> m_timer;
          ClockT::duration m_timeAccumulated;

          public:

          explicit Stopwatch(bool paused = false):
          m_paused(paused),
          m_timer(),
          m_timeAccumulated(0)




          void Pause()

          m_timeAccumulated += m_timer.GetElapsedTime();
          m_paused = true;


          void Unpause()

          if (m_paused)

          m_timer = Timer<ClockT>();
          m_paused = false;



          ClockT::duration GetElapsedTime() const

          if (m_paused)
          return m_timeAccumulated;

          return m_timeAccumulated + m_timer.GetElapsedTime();

          ;

          #include <iostream>

          int main()

          using seconds_t = std::chrono::duration<float>;

          Stopwatch s;

          for (int i = 0; i != 50; ++i) std::cout << ".";
          std::cout << "n";

          std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

          s = Stopwatch(true);
          std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

          s.Unpause();

          for (int i = 0; i != 50; ++i) std::cout << ".";
          std::cout << "n";

          s.Pause();
          std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

          for (int i = 0; i != 50; ++i) std::cout << ".";
          std::cout << "n";

          std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;






          share|improve this answer










          $endgroup$
















            Your Answer






            StackExchange.ifUsing("editor", function ()
            StackExchange.using("externalEditor", function ()
            StackExchange.using("snippets", function ()
            StackExchange.snippets.init();
            );
            );
            , "code-snippets");

            StackExchange.ready(function()
            var channelOptions =
            tags: "".split(" "),
            id: "196"
            ;
            initTagRenderer("".split(" "), "".split(" "), channelOptions);

            StackExchange.using("externalEditor", function()
            // Have to fire editor after snippets, if snippets enabled
            if (StackExchange.settings.snippets.snippetsEnabled)
            StackExchange.using("snippets", function()
            createEditor();
            );

            else
            createEditor();

            );

            function createEditor()
            StackExchange.prepareEditor(
            heartbeatType: 'answer',
            autoActivateHeartbeat: false,
            convertImagesToLinks: false,
            noModals: true,
            showLowRepImageUploadWarning: true,
            reputationToPostImages: null,
            bindNavPrevention: true,
            postfix: "",
            imageUploader:
            brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
            contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/4.0/"u003ecc by-sa 4.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
            allowUrls: true
            ,
            onDemand: true,
            discardSelector: ".discard-answer"
            ,immediatelyShowMarkdownHelp:true
            );



            );














            draft saved

            draft discarded
















            StackExchange.ready(
            function ()
            StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f225923%2fa-simple-stop-watch-which-i-want-to-extend%23new-answer', 'question_page');

            );

            Post as a guest















            Required, but never shown


























            2 Answers
            2






            active

            oldest

            votes








            2 Answers
            2






            active

            oldest

            votes









            active

            oldest

            votes






            active

            oldest

            votes









            7
















            $begingroup$

            Don't use member variables for temporary storage



            Your class Timer has a member variable mvSession, which is unnecessary. It is only used in SessionTime(), where it is filled in and returned. No other functions use it. Instead, just declare std::vector<int64_t> mvSession(3) inside SessionTime().



            By making it a member variable, you introduced two problems: first, your class now uses more memory than necessary. Second, the function SessionTime() is now no longer reentrant; if two threads would call SessionTime() simultaneously, they would both write to the same mvSession variable, potentially corrupting it.



            Use std::chrono::steady_clock for timers



            The problem with std::chrono::high_resolution_clock() is that it is not guaranteed what kind of clock it is. It can represent wall clock time, which can jump ahead or forwards because of summer and winter time, leap seconds, the computer suspending and resuming. This is not something you want for a timer where you are interested in a simple duration. For this, you want to use std::chrono::steady_clock, which is guaranteed to be a monotonically increasing clock. Also, your function SessionTime() returns the time in a resolution of seconds, so you don't need a high resolution clock anyway.



            Don't use Hungarian notation



            There might be some merits to Hungarian notation, but it really isn't necessary to use it for C++, since the compiler will do type checking for you. Moreover, it's easy to use the wrong prefix, and it's hard to come up with a reasonable prefix when you can have complex types.



            You are already making mistakes in your code. For example, mtpNow and mtPause are both of type cppTimer::time_point. So the prefix should have been the same. And mtRestart has a different type than mtPause, so their prefixes should have been different. I recommend that you just avoid using Hungarian notation altoghether.



            Be consistent with using



            You are declaring using seconds = std::chrono::seconds, and use seconds in a few places, but you also use std::chrono::seconds in a lot of places. Furthermore, you also use std::chrono::minutes and std::chrono::hours, but have not declared an alias for them. In this case, I suggest you don't declare using seconds at all.



            I would keep using cppTimer though, since it basically selects which clock to use. That makes it easier to change the clock later by just changing one line of code. I would write using clock = ... though, to be consistent with the C++ library itself.



            Don't cast to seconds too early



            Instead of seconds mtRestart, use cppTimer::duration mtRestart. This will keep the accuracy of the duration to the same as the clock itself. Only cast durations to seconds or other time intervals until the last moment possible. The same goes for the calculation of delay in SessionTime(), just write:



            auto delay = now - mtpNow;


            Use nouns for variable names, verbs for function names



            A variable holds (the value of) a thing, so its name should naturally be a noun. Functions do stuff, so there names should generally be verbs. The function Now() should actually be named Start(). Your function Start() should probably be named Continue(). The function SessionTime() calculates how long the timer has been running for, so probably should be named GetDuration().



            Conversely, the variables mtPause and mtRestart should be renamed to nouns as well. They are a bit confusing. Sure, you set mtPause in the Pause() function, but it doesn't describe what the value actually means. The same goes for mtRestart. I would instead write:



            clock::time_point StartTime;
            clock::time_point PauseTime;
            clock::duration PausedDuration;
            bool IsPaused;


            Now you can rewrite the function Now() to:



            void Start() 
            StartTime = clock::now();
            IsPaused = false;
            PausedDuration = ;



            Remove mtRestart



            You are using two variables to handle the timer being paused, mtPause and mtRestart. However, you only need one. In the Pause() function, you indeed just record when this function is called. However, when restarting the timer, instead of adding the duration of being paused to mtRestart, just add that duration to mtpNow instead:



            void Start() 
            if(mbIsPause)
            mtpNow += cppTimer::now() - mtPause;
            mbIsPause = false;




            This also simplifies SessionTime():



            std::vector<int64_t> SessionTime() 
            auto end = mbIsPause ? mtPause : cppTimer::now();
            auto tp = std::chrono::duration_cast<std::chrono::seconds>(end - mtpNow);
            ...



            Also, since mtPause is only ever 0 when you didn't pause, you can use that to signal whether the timer is paused instead of having bool mbIsPause.



            Another option is @user673679's suggestion of storing only the start time and the accumulated elapsed time so far. You would then use the start time being equal to as a signal that the timer has not been started.



            Just return a std::chrono::duration



            When you want the elapsed time, I would avoid having the Timer class be responsible for converting the duration to a vector of integers representing hours, minutes and seconds. It reduces the accuracy of your timer. Instead, I would just return a std::chrono::duration, and have the caller decide whether it wants to convert that to something. It also is much more efficient than having to construct a std::vector<int64_t>.



            Try to make it behave reasonable in all situations



            One issue with your code is that it only gives reasonable results for SessionTime() if you have called Now() at least once. You didn't initialize mtpNow, and even if it was value-initialized to zero, then SessionTime() will return the time that has passed since the epoch.



            If you want the Timer to behave like it was started at construction time, then initialize mtpNow to cppTimer::now(). If you want it to behave like it was paused, then ensure both mtpNow and mtPause are initialized to the same value (I suggest just using ), and that mbIsPause is true.



            Make it work like a real stopwatch



            As already suggested by others, it helps to think of a timer as a stopwatch. A real stopwatch starts in a stopped state, showing an elapsed time of zero. Then, you can start and stop the timer mechanism with buttons. Usually, there is a separate button to reset the stopwatch to its initial state. By making the class act like something a lot of people are already familiar with, the code is easier to understand for others.



            Reworked code



            Here is an example of what the code would look like with my suggestions applied, as well as @user673679's way of storing the elapsed time between previous start and stops of the clock:



            #include <chrono>

            class Timer
            using clock = std::chrono::steady_clock;
            clock::time_point StartTime = ;
            clock::duration ElapsedTime = ;

            public:
            bool IsRunning() const
            return StartTime != clock::time_point;


            void Start()
            if(!IsRunning())
            StartTime = clock::now();



            void Stop()
            if(IsRunning())
            ElapsedTime += clock::now() - StartTime;
            StartTime = ;



            void Reset()
            StartTime = ;
            ElapsedTime = ;


            clock::duration GetElapsed()
            auto result = ElapsedTime;
            if(IsRunning())
            result += clock::now() - StartTime;

            return result;

            ;
            ```





            share|improve this answer












            $endgroup$














            • $begingroup$
              That is "system" Hungarian notation (the Microsoft Windows documentation team grossly misinterpreted the original Hungarian notation), not the original Hungarian notation (which is not about types). See e.g. Triangulation episode 277, 4 min 00 secs to 5 min 19 secs.
              $endgroup$
              – Peter Mortensen
              Aug 12 at 1:25
















            7
















            $begingroup$

            Don't use member variables for temporary storage



            Your class Timer has a member variable mvSession, which is unnecessary. It is only used in SessionTime(), where it is filled in and returned. No other functions use it. Instead, just declare std::vector<int64_t> mvSession(3) inside SessionTime().



            By making it a member variable, you introduced two problems: first, your class now uses more memory than necessary. Second, the function SessionTime() is now no longer reentrant; if two threads would call SessionTime() simultaneously, they would both write to the same mvSession variable, potentially corrupting it.



            Use std::chrono::steady_clock for timers



            The problem with std::chrono::high_resolution_clock() is that it is not guaranteed what kind of clock it is. It can represent wall clock time, which can jump ahead or forwards because of summer and winter time, leap seconds, the computer suspending and resuming. This is not something you want for a timer where you are interested in a simple duration. For this, you want to use std::chrono::steady_clock, which is guaranteed to be a monotonically increasing clock. Also, your function SessionTime() returns the time in a resolution of seconds, so you don't need a high resolution clock anyway.



            Don't use Hungarian notation



            There might be some merits to Hungarian notation, but it really isn't necessary to use it for C++, since the compiler will do type checking for you. Moreover, it's easy to use the wrong prefix, and it's hard to come up with a reasonable prefix when you can have complex types.



            You are already making mistakes in your code. For example, mtpNow and mtPause are both of type cppTimer::time_point. So the prefix should have been the same. And mtRestart has a different type than mtPause, so their prefixes should have been different. I recommend that you just avoid using Hungarian notation altoghether.



            Be consistent with using



            You are declaring using seconds = std::chrono::seconds, and use seconds in a few places, but you also use std::chrono::seconds in a lot of places. Furthermore, you also use std::chrono::minutes and std::chrono::hours, but have not declared an alias for them. In this case, I suggest you don't declare using seconds at all.



            I would keep using cppTimer though, since it basically selects which clock to use. That makes it easier to change the clock later by just changing one line of code. I would write using clock = ... though, to be consistent with the C++ library itself.



            Don't cast to seconds too early



            Instead of seconds mtRestart, use cppTimer::duration mtRestart. This will keep the accuracy of the duration to the same as the clock itself. Only cast durations to seconds or other time intervals until the last moment possible. The same goes for the calculation of delay in SessionTime(), just write:



            auto delay = now - mtpNow;


            Use nouns for variable names, verbs for function names



            A variable holds (the value of) a thing, so its name should naturally be a noun. Functions do stuff, so there names should generally be verbs. The function Now() should actually be named Start(). Your function Start() should probably be named Continue(). The function SessionTime() calculates how long the timer has been running for, so probably should be named GetDuration().



            Conversely, the variables mtPause and mtRestart should be renamed to nouns as well. They are a bit confusing. Sure, you set mtPause in the Pause() function, but it doesn't describe what the value actually means. The same goes for mtRestart. I would instead write:



            clock::time_point StartTime;
            clock::time_point PauseTime;
            clock::duration PausedDuration;
            bool IsPaused;


            Now you can rewrite the function Now() to:



            void Start() 
            StartTime = clock::now();
            IsPaused = false;
            PausedDuration = ;



            Remove mtRestart



            You are using two variables to handle the timer being paused, mtPause and mtRestart. However, you only need one. In the Pause() function, you indeed just record when this function is called. However, when restarting the timer, instead of adding the duration of being paused to mtRestart, just add that duration to mtpNow instead:



            void Start() 
            if(mbIsPause)
            mtpNow += cppTimer::now() - mtPause;
            mbIsPause = false;




            This also simplifies SessionTime():



            std::vector<int64_t> SessionTime() 
            auto end = mbIsPause ? mtPause : cppTimer::now();
            auto tp = std::chrono::duration_cast<std::chrono::seconds>(end - mtpNow);
            ...



            Also, since mtPause is only ever 0 when you didn't pause, you can use that to signal whether the timer is paused instead of having bool mbIsPause.



            Another option is @user673679's suggestion of storing only the start time and the accumulated elapsed time so far. You would then use the start time being equal to as a signal that the timer has not been started.



            Just return a std::chrono::duration



            When you want the elapsed time, I would avoid having the Timer class be responsible for converting the duration to a vector of integers representing hours, minutes and seconds. It reduces the accuracy of your timer. Instead, I would just return a std::chrono::duration, and have the caller decide whether it wants to convert that to something. It also is much more efficient than having to construct a std::vector<int64_t>.



            Try to make it behave reasonable in all situations



            One issue with your code is that it only gives reasonable results for SessionTime() if you have called Now() at least once. You didn't initialize mtpNow, and even if it was value-initialized to zero, then SessionTime() will return the time that has passed since the epoch.



            If you want the Timer to behave like it was started at construction time, then initialize mtpNow to cppTimer::now(). If you want it to behave like it was paused, then ensure both mtpNow and mtPause are initialized to the same value (I suggest just using ), and that mbIsPause is true.



            Make it work like a real stopwatch



            As already suggested by others, it helps to think of a timer as a stopwatch. A real stopwatch starts in a stopped state, showing an elapsed time of zero. Then, you can start and stop the timer mechanism with buttons. Usually, there is a separate button to reset the stopwatch to its initial state. By making the class act like something a lot of people are already familiar with, the code is easier to understand for others.



            Reworked code



            Here is an example of what the code would look like with my suggestions applied, as well as @user673679's way of storing the elapsed time between previous start and stops of the clock:



            #include <chrono>

            class Timer
            using clock = std::chrono::steady_clock;
            clock::time_point StartTime = ;
            clock::duration ElapsedTime = ;

            public:
            bool IsRunning() const
            return StartTime != clock::time_point;


            void Start()
            if(!IsRunning())
            StartTime = clock::now();



            void Stop()
            if(IsRunning())
            ElapsedTime += clock::now() - StartTime;
            StartTime = ;



            void Reset()
            StartTime = ;
            ElapsedTime = ;


            clock::duration GetElapsed()
            auto result = ElapsedTime;
            if(IsRunning())
            result += clock::now() - StartTime;

            return result;

            ;
            ```





            share|improve this answer












            $endgroup$














            • $begingroup$
              That is "system" Hungarian notation (the Microsoft Windows documentation team grossly misinterpreted the original Hungarian notation), not the original Hungarian notation (which is not about types). See e.g. Triangulation episode 277, 4 min 00 secs to 5 min 19 secs.
              $endgroup$
              – Peter Mortensen
              Aug 12 at 1:25














            7














            7










            7







            $begingroup$

            Don't use member variables for temporary storage



            Your class Timer has a member variable mvSession, which is unnecessary. It is only used in SessionTime(), where it is filled in and returned. No other functions use it. Instead, just declare std::vector<int64_t> mvSession(3) inside SessionTime().



            By making it a member variable, you introduced two problems: first, your class now uses more memory than necessary. Second, the function SessionTime() is now no longer reentrant; if two threads would call SessionTime() simultaneously, they would both write to the same mvSession variable, potentially corrupting it.



            Use std::chrono::steady_clock for timers



            The problem with std::chrono::high_resolution_clock() is that it is not guaranteed what kind of clock it is. It can represent wall clock time, which can jump ahead or forwards because of summer and winter time, leap seconds, the computer suspending and resuming. This is not something you want for a timer where you are interested in a simple duration. For this, you want to use std::chrono::steady_clock, which is guaranteed to be a monotonically increasing clock. Also, your function SessionTime() returns the time in a resolution of seconds, so you don't need a high resolution clock anyway.



            Don't use Hungarian notation



            There might be some merits to Hungarian notation, but it really isn't necessary to use it for C++, since the compiler will do type checking for you. Moreover, it's easy to use the wrong prefix, and it's hard to come up with a reasonable prefix when you can have complex types.



            You are already making mistakes in your code. For example, mtpNow and mtPause are both of type cppTimer::time_point. So the prefix should have been the same. And mtRestart has a different type than mtPause, so their prefixes should have been different. I recommend that you just avoid using Hungarian notation altoghether.



            Be consistent with using



            You are declaring using seconds = std::chrono::seconds, and use seconds in a few places, but you also use std::chrono::seconds in a lot of places. Furthermore, you also use std::chrono::minutes and std::chrono::hours, but have not declared an alias for them. In this case, I suggest you don't declare using seconds at all.



            I would keep using cppTimer though, since it basically selects which clock to use. That makes it easier to change the clock later by just changing one line of code. I would write using clock = ... though, to be consistent with the C++ library itself.



            Don't cast to seconds too early



            Instead of seconds mtRestart, use cppTimer::duration mtRestart. This will keep the accuracy of the duration to the same as the clock itself. Only cast durations to seconds or other time intervals until the last moment possible. The same goes for the calculation of delay in SessionTime(), just write:



            auto delay = now - mtpNow;


            Use nouns for variable names, verbs for function names



            A variable holds (the value of) a thing, so its name should naturally be a noun. Functions do stuff, so there names should generally be verbs. The function Now() should actually be named Start(). Your function Start() should probably be named Continue(). The function SessionTime() calculates how long the timer has been running for, so probably should be named GetDuration().



            Conversely, the variables mtPause and mtRestart should be renamed to nouns as well. They are a bit confusing. Sure, you set mtPause in the Pause() function, but it doesn't describe what the value actually means. The same goes for mtRestart. I would instead write:



            clock::time_point StartTime;
            clock::time_point PauseTime;
            clock::duration PausedDuration;
            bool IsPaused;


            Now you can rewrite the function Now() to:



            void Start() 
            StartTime = clock::now();
            IsPaused = false;
            PausedDuration = ;



            Remove mtRestart



            You are using two variables to handle the timer being paused, mtPause and mtRestart. However, you only need one. In the Pause() function, you indeed just record when this function is called. However, when restarting the timer, instead of adding the duration of being paused to mtRestart, just add that duration to mtpNow instead:



            void Start() 
            if(mbIsPause)
            mtpNow += cppTimer::now() - mtPause;
            mbIsPause = false;




            This also simplifies SessionTime():



            std::vector<int64_t> SessionTime() 
            auto end = mbIsPause ? mtPause : cppTimer::now();
            auto tp = std::chrono::duration_cast<std::chrono::seconds>(end - mtpNow);
            ...



            Also, since mtPause is only ever 0 when you didn't pause, you can use that to signal whether the timer is paused instead of having bool mbIsPause.



            Another option is @user673679's suggestion of storing only the start time and the accumulated elapsed time so far. You would then use the start time being equal to as a signal that the timer has not been started.



            Just return a std::chrono::duration



            When you want the elapsed time, I would avoid having the Timer class be responsible for converting the duration to a vector of integers representing hours, minutes and seconds. It reduces the accuracy of your timer. Instead, I would just return a std::chrono::duration, and have the caller decide whether it wants to convert that to something. It also is much more efficient than having to construct a std::vector<int64_t>.



            Try to make it behave reasonable in all situations



            One issue with your code is that it only gives reasonable results for SessionTime() if you have called Now() at least once. You didn't initialize mtpNow, and even if it was value-initialized to zero, then SessionTime() will return the time that has passed since the epoch.



            If you want the Timer to behave like it was started at construction time, then initialize mtpNow to cppTimer::now(). If you want it to behave like it was paused, then ensure both mtpNow and mtPause are initialized to the same value (I suggest just using ), and that mbIsPause is true.



            Make it work like a real stopwatch



            As already suggested by others, it helps to think of a timer as a stopwatch. A real stopwatch starts in a stopped state, showing an elapsed time of zero. Then, you can start and stop the timer mechanism with buttons. Usually, there is a separate button to reset the stopwatch to its initial state. By making the class act like something a lot of people are already familiar with, the code is easier to understand for others.



            Reworked code



            Here is an example of what the code would look like with my suggestions applied, as well as @user673679's way of storing the elapsed time between previous start and stops of the clock:



            #include <chrono>

            class Timer
            using clock = std::chrono::steady_clock;
            clock::time_point StartTime = ;
            clock::duration ElapsedTime = ;

            public:
            bool IsRunning() const
            return StartTime != clock::time_point;


            void Start()
            if(!IsRunning())
            StartTime = clock::now();



            void Stop()
            if(IsRunning())
            ElapsedTime += clock::now() - StartTime;
            StartTime = ;



            void Reset()
            StartTime = ;
            ElapsedTime = ;


            clock::duration GetElapsed()
            auto result = ElapsedTime;
            if(IsRunning())
            result += clock::now() - StartTime;

            return result;

            ;
            ```





            share|improve this answer












            $endgroup$



            Don't use member variables for temporary storage



            Your class Timer has a member variable mvSession, which is unnecessary. It is only used in SessionTime(), where it is filled in and returned. No other functions use it. Instead, just declare std::vector<int64_t> mvSession(3) inside SessionTime().



            By making it a member variable, you introduced two problems: first, your class now uses more memory than necessary. Second, the function SessionTime() is now no longer reentrant; if two threads would call SessionTime() simultaneously, they would both write to the same mvSession variable, potentially corrupting it.



            Use std::chrono::steady_clock for timers



            The problem with std::chrono::high_resolution_clock() is that it is not guaranteed what kind of clock it is. It can represent wall clock time, which can jump ahead or forwards because of summer and winter time, leap seconds, the computer suspending and resuming. This is not something you want for a timer where you are interested in a simple duration. For this, you want to use std::chrono::steady_clock, which is guaranteed to be a monotonically increasing clock. Also, your function SessionTime() returns the time in a resolution of seconds, so you don't need a high resolution clock anyway.



            Don't use Hungarian notation



            There might be some merits to Hungarian notation, but it really isn't necessary to use it for C++, since the compiler will do type checking for you. Moreover, it's easy to use the wrong prefix, and it's hard to come up with a reasonable prefix when you can have complex types.



            You are already making mistakes in your code. For example, mtpNow and mtPause are both of type cppTimer::time_point. So the prefix should have been the same. And mtRestart has a different type than mtPause, so their prefixes should have been different. I recommend that you just avoid using Hungarian notation altoghether.



            Be consistent with using



            You are declaring using seconds = std::chrono::seconds, and use seconds in a few places, but you also use std::chrono::seconds in a lot of places. Furthermore, you also use std::chrono::minutes and std::chrono::hours, but have not declared an alias for them. In this case, I suggest you don't declare using seconds at all.



            I would keep using cppTimer though, since it basically selects which clock to use. That makes it easier to change the clock later by just changing one line of code. I would write using clock = ... though, to be consistent with the C++ library itself.



            Don't cast to seconds too early



            Instead of seconds mtRestart, use cppTimer::duration mtRestart. This will keep the accuracy of the duration to the same as the clock itself. Only cast durations to seconds or other time intervals until the last moment possible. The same goes for the calculation of delay in SessionTime(), just write:



            auto delay = now - mtpNow;


            Use nouns for variable names, verbs for function names



            A variable holds (the value of) a thing, so its name should naturally be a noun. Functions do stuff, so there names should generally be verbs. The function Now() should actually be named Start(). Your function Start() should probably be named Continue(). The function SessionTime() calculates how long the timer has been running for, so probably should be named GetDuration().



            Conversely, the variables mtPause and mtRestart should be renamed to nouns as well. They are a bit confusing. Sure, you set mtPause in the Pause() function, but it doesn't describe what the value actually means. The same goes for mtRestart. I would instead write:



            clock::time_point StartTime;
            clock::time_point PauseTime;
            clock::duration PausedDuration;
            bool IsPaused;


            Now you can rewrite the function Now() to:



            void Start() 
            StartTime = clock::now();
            IsPaused = false;
            PausedDuration = ;



            Remove mtRestart



            You are using two variables to handle the timer being paused, mtPause and mtRestart. However, you only need one. In the Pause() function, you indeed just record when this function is called. However, when restarting the timer, instead of adding the duration of being paused to mtRestart, just add that duration to mtpNow instead:



            void Start() 
            if(mbIsPause)
            mtpNow += cppTimer::now() - mtPause;
            mbIsPause = false;




            This also simplifies SessionTime():



            std::vector<int64_t> SessionTime() 
            auto end = mbIsPause ? mtPause : cppTimer::now();
            auto tp = std::chrono::duration_cast<std::chrono::seconds>(end - mtpNow);
            ...



            Also, since mtPause is only ever 0 when you didn't pause, you can use that to signal whether the timer is paused instead of having bool mbIsPause.



            Another option is @user673679's suggestion of storing only the start time and the accumulated elapsed time so far. You would then use the start time being equal to as a signal that the timer has not been started.



            Just return a std::chrono::duration



            When you want the elapsed time, I would avoid having the Timer class be responsible for converting the duration to a vector of integers representing hours, minutes and seconds. It reduces the accuracy of your timer. Instead, I would just return a std::chrono::duration, and have the caller decide whether it wants to convert that to something. It also is much more efficient than having to construct a std::vector<int64_t>.



            Try to make it behave reasonable in all situations



            One issue with your code is that it only gives reasonable results for SessionTime() if you have called Now() at least once. You didn't initialize mtpNow, and even if it was value-initialized to zero, then SessionTime() will return the time that has passed since the epoch.



            If you want the Timer to behave like it was started at construction time, then initialize mtpNow to cppTimer::now(). If you want it to behave like it was paused, then ensure both mtpNow and mtPause are initialized to the same value (I suggest just using ), and that mbIsPause is true.



            Make it work like a real stopwatch



            As already suggested by others, it helps to think of a timer as a stopwatch. A real stopwatch starts in a stopped state, showing an elapsed time of zero. Then, you can start and stop the timer mechanism with buttons. Usually, there is a separate button to reset the stopwatch to its initial state. By making the class act like something a lot of people are already familiar with, the code is easier to understand for others.



            Reworked code



            Here is an example of what the code would look like with my suggestions applied, as well as @user673679's way of storing the elapsed time between previous start and stops of the clock:



            #include <chrono>

            class Timer
            using clock = std::chrono::steady_clock;
            clock::time_point StartTime = ;
            clock::duration ElapsedTime = ;

            public:
            bool IsRunning() const
            return StartTime != clock::time_point;


            void Start()
            if(!IsRunning())
            StartTime = clock::now();



            void Stop()
            if(IsRunning())
            ElapsedTime += clock::now() - StartTime;
            StartTime = ;



            void Reset()
            StartTime = ;
            ElapsedTime = ;


            clock::duration GetElapsed()
            auto result = ElapsedTime;
            if(IsRunning())
            result += clock::now() - StartTime;

            return result;

            ;
            ```






            share|improve this answer















            share|improve this answer




            share|improve this answer








            edited Aug 11 at 20:51

























            answered Aug 11 at 13:19









            G. SliepenG. Sliepen

            2,3905 silver badges14 bronze badges




            2,3905 silver badges14 bronze badges














            • $begingroup$
              That is "system" Hungarian notation (the Microsoft Windows documentation team grossly misinterpreted the original Hungarian notation), not the original Hungarian notation (which is not about types). See e.g. Triangulation episode 277, 4 min 00 secs to 5 min 19 secs.
              $endgroup$
              – Peter Mortensen
              Aug 12 at 1:25

















            • $begingroup$
              That is "system" Hungarian notation (the Microsoft Windows documentation team grossly misinterpreted the original Hungarian notation), not the original Hungarian notation (which is not about types). See e.g. Triangulation episode 277, 4 min 00 secs to 5 min 19 secs.
              $endgroup$
              – Peter Mortensen
              Aug 12 at 1:25
















            $begingroup$
            That is "system" Hungarian notation (the Microsoft Windows documentation team grossly misinterpreted the original Hungarian notation), not the original Hungarian notation (which is not about types). See e.g. Triangulation episode 277, 4 min 00 secs to 5 min 19 secs.
            $endgroup$
            – Peter Mortensen
            Aug 12 at 1:25





            $begingroup$
            That is "system" Hungarian notation (the Microsoft Windows documentation team grossly misinterpreted the original Hungarian notation), not the original Hungarian notation (which is not about types). See e.g. Triangulation episode 277, 4 min 00 secs to 5 min 19 secs.
            $endgroup$
            – Peter Mortensen
            Aug 12 at 1:25














            3
















            $begingroup$

            • It's fine to prefix member variables with an m, but it's probably best to avoid type prefixes (e.g. mtpNow). It makes code harder to read (you have to know what every abbreviation means), it's a pain to maintain (e.g. mtPause should probably be mtpPause to be consistent). Modern tools eliminate the need for this too (mousing over a variable in Visual Studio will tell me the exact type, not an approximation).


            • Functions defined inline in the class don't need to be declared inline.



            • The naming is very confusing:



              • Now() differs from the standard library (now() returns the current time, and is arguably still a terrible name). Functions names should be commands or questions. Perhaps Restart() or Reset() would be better.


              • Start() is also not ideal. One might expect the function to do what Now() does. I'd suggest calling it Unpause(), which makes the purpose clearer.


              • mtRestart is an odd name for the time spent paused.



            • We don't really need the Now() / Restart() function, since we can just assign a new timer to the old one to do the same thing (e.g. Timer timer; ...; timer = Timer(); // restarted!).


            • There's no reason to keep a vector in the class (we're copying it every time anyway, so we might as well just create a new vector each time).


            • We don't need a vector, since it always has 3 values, it would be neater to return a simple struct, which would allow us to give each value a name. Or...


            • The real solution is to just return an appropriate chrono:: type, and let the user worry about formatting / converting it.


            • I'd suggest naming the class Stopwatch, since that's more specific to the pausable timing functionality we need. We can actually implement this functionality based on a simpler, non-pausable Timer class.



            So overall I'd suggest something more like this (not tested properly):



            #include <chrono>

            template<class ClockT = std::chrono::high_resolution_clock>
            class Timer

            ClockT::time_point m_start;

            public:

            Timer():
            m_start(ClockT::now())




            ClockT::duration GetElapsedTime() const

            return ClockT::now() - m_start;

            ;

            template<class ClockT = std::chrono::high_resolution_clock>
            class Stopwatch

            bool m_paused;
            Timer<ClockT> m_timer;
            ClockT::duration m_timeAccumulated;

            public:

            explicit Stopwatch(bool paused = false):
            m_paused(paused),
            m_timer(),
            m_timeAccumulated(0)




            void Pause()

            m_timeAccumulated += m_timer.GetElapsedTime();
            m_paused = true;


            void Unpause()

            if (m_paused)

            m_timer = Timer<ClockT>();
            m_paused = false;



            ClockT::duration GetElapsedTime() const

            if (m_paused)
            return m_timeAccumulated;

            return m_timeAccumulated + m_timer.GetElapsedTime();

            ;

            #include <iostream>

            int main()

            using seconds_t = std::chrono::duration<float>;

            Stopwatch s;

            for (int i = 0; i != 50; ++i) std::cout << ".";
            std::cout << "n";

            std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

            s = Stopwatch(true);
            std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

            s.Unpause();

            for (int i = 0; i != 50; ++i) std::cout << ".";
            std::cout << "n";

            s.Pause();
            std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

            for (int i = 0; i != 50; ++i) std::cout << ".";
            std::cout << "n";

            std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;






            share|improve this answer










            $endgroup$



















              3
















              $begingroup$

              • It's fine to prefix member variables with an m, but it's probably best to avoid type prefixes (e.g. mtpNow). It makes code harder to read (you have to know what every abbreviation means), it's a pain to maintain (e.g. mtPause should probably be mtpPause to be consistent). Modern tools eliminate the need for this too (mousing over a variable in Visual Studio will tell me the exact type, not an approximation).


              • Functions defined inline in the class don't need to be declared inline.



              • The naming is very confusing:



                • Now() differs from the standard library (now() returns the current time, and is arguably still a terrible name). Functions names should be commands or questions. Perhaps Restart() or Reset() would be better.


                • Start() is also not ideal. One might expect the function to do what Now() does. I'd suggest calling it Unpause(), which makes the purpose clearer.


                • mtRestart is an odd name for the time spent paused.



              • We don't really need the Now() / Restart() function, since we can just assign a new timer to the old one to do the same thing (e.g. Timer timer; ...; timer = Timer(); // restarted!).


              • There's no reason to keep a vector in the class (we're copying it every time anyway, so we might as well just create a new vector each time).


              • We don't need a vector, since it always has 3 values, it would be neater to return a simple struct, which would allow us to give each value a name. Or...


              • The real solution is to just return an appropriate chrono:: type, and let the user worry about formatting / converting it.


              • I'd suggest naming the class Stopwatch, since that's more specific to the pausable timing functionality we need. We can actually implement this functionality based on a simpler, non-pausable Timer class.



              So overall I'd suggest something more like this (not tested properly):



              #include <chrono>

              template<class ClockT = std::chrono::high_resolution_clock>
              class Timer

              ClockT::time_point m_start;

              public:

              Timer():
              m_start(ClockT::now())




              ClockT::duration GetElapsedTime() const

              return ClockT::now() - m_start;

              ;

              template<class ClockT = std::chrono::high_resolution_clock>
              class Stopwatch

              bool m_paused;
              Timer<ClockT> m_timer;
              ClockT::duration m_timeAccumulated;

              public:

              explicit Stopwatch(bool paused = false):
              m_paused(paused),
              m_timer(),
              m_timeAccumulated(0)




              void Pause()

              m_timeAccumulated += m_timer.GetElapsedTime();
              m_paused = true;


              void Unpause()

              if (m_paused)

              m_timer = Timer<ClockT>();
              m_paused = false;



              ClockT::duration GetElapsedTime() const

              if (m_paused)
              return m_timeAccumulated;

              return m_timeAccumulated + m_timer.GetElapsedTime();

              ;

              #include <iostream>

              int main()

              using seconds_t = std::chrono::duration<float>;

              Stopwatch s;

              for (int i = 0; i != 50; ++i) std::cout << ".";
              std::cout << "n";

              std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

              s = Stopwatch(true);
              std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

              s.Unpause();

              for (int i = 0; i != 50; ++i) std::cout << ".";
              std::cout << "n";

              s.Pause();
              std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

              for (int i = 0; i != 50; ++i) std::cout << ".";
              std::cout << "n";

              std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;






              share|improve this answer










              $endgroup$

















                3














                3










                3







                $begingroup$

                • It's fine to prefix member variables with an m, but it's probably best to avoid type prefixes (e.g. mtpNow). It makes code harder to read (you have to know what every abbreviation means), it's a pain to maintain (e.g. mtPause should probably be mtpPause to be consistent). Modern tools eliminate the need for this too (mousing over a variable in Visual Studio will tell me the exact type, not an approximation).


                • Functions defined inline in the class don't need to be declared inline.



                • The naming is very confusing:



                  • Now() differs from the standard library (now() returns the current time, and is arguably still a terrible name). Functions names should be commands or questions. Perhaps Restart() or Reset() would be better.


                  • Start() is also not ideal. One might expect the function to do what Now() does. I'd suggest calling it Unpause(), which makes the purpose clearer.


                  • mtRestart is an odd name for the time spent paused.



                • We don't really need the Now() / Restart() function, since we can just assign a new timer to the old one to do the same thing (e.g. Timer timer; ...; timer = Timer(); // restarted!).


                • There's no reason to keep a vector in the class (we're copying it every time anyway, so we might as well just create a new vector each time).


                • We don't need a vector, since it always has 3 values, it would be neater to return a simple struct, which would allow us to give each value a name. Or...


                • The real solution is to just return an appropriate chrono:: type, and let the user worry about formatting / converting it.


                • I'd suggest naming the class Stopwatch, since that's more specific to the pausable timing functionality we need. We can actually implement this functionality based on a simpler, non-pausable Timer class.



                So overall I'd suggest something more like this (not tested properly):



                #include <chrono>

                template<class ClockT = std::chrono::high_resolution_clock>
                class Timer

                ClockT::time_point m_start;

                public:

                Timer():
                m_start(ClockT::now())




                ClockT::duration GetElapsedTime() const

                return ClockT::now() - m_start;

                ;

                template<class ClockT = std::chrono::high_resolution_clock>
                class Stopwatch

                bool m_paused;
                Timer<ClockT> m_timer;
                ClockT::duration m_timeAccumulated;

                public:

                explicit Stopwatch(bool paused = false):
                m_paused(paused),
                m_timer(),
                m_timeAccumulated(0)




                void Pause()

                m_timeAccumulated += m_timer.GetElapsedTime();
                m_paused = true;


                void Unpause()

                if (m_paused)

                m_timer = Timer<ClockT>();
                m_paused = false;



                ClockT::duration GetElapsedTime() const

                if (m_paused)
                return m_timeAccumulated;

                return m_timeAccumulated + m_timer.GetElapsedTime();

                ;

                #include <iostream>

                int main()

                using seconds_t = std::chrono::duration<float>;

                Stopwatch s;

                for (int i = 0; i != 50; ++i) std::cout << ".";
                std::cout << "n";

                std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

                s = Stopwatch(true);
                std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

                s.Unpause();

                for (int i = 0; i != 50; ++i) std::cout << ".";
                std::cout << "n";

                s.Pause();
                std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

                for (int i = 0; i != 50; ++i) std::cout << ".";
                std::cout << "n";

                std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;






                share|improve this answer










                $endgroup$



                • It's fine to prefix member variables with an m, but it's probably best to avoid type prefixes (e.g. mtpNow). It makes code harder to read (you have to know what every abbreviation means), it's a pain to maintain (e.g. mtPause should probably be mtpPause to be consistent). Modern tools eliminate the need for this too (mousing over a variable in Visual Studio will tell me the exact type, not an approximation).


                • Functions defined inline in the class don't need to be declared inline.



                • The naming is very confusing:



                  • Now() differs from the standard library (now() returns the current time, and is arguably still a terrible name). Functions names should be commands or questions. Perhaps Restart() or Reset() would be better.


                  • Start() is also not ideal. One might expect the function to do what Now() does. I'd suggest calling it Unpause(), which makes the purpose clearer.


                  • mtRestart is an odd name for the time spent paused.



                • We don't really need the Now() / Restart() function, since we can just assign a new timer to the old one to do the same thing (e.g. Timer timer; ...; timer = Timer(); // restarted!).


                • There's no reason to keep a vector in the class (we're copying it every time anyway, so we might as well just create a new vector each time).


                • We don't need a vector, since it always has 3 values, it would be neater to return a simple struct, which would allow us to give each value a name. Or...


                • The real solution is to just return an appropriate chrono:: type, and let the user worry about formatting / converting it.


                • I'd suggest naming the class Stopwatch, since that's more specific to the pausable timing functionality we need. We can actually implement this functionality based on a simpler, non-pausable Timer class.



                So overall I'd suggest something more like this (not tested properly):



                #include <chrono>

                template<class ClockT = std::chrono::high_resolution_clock>
                class Timer

                ClockT::time_point m_start;

                public:

                Timer():
                m_start(ClockT::now())




                ClockT::duration GetElapsedTime() const

                return ClockT::now() - m_start;

                ;

                template<class ClockT = std::chrono::high_resolution_clock>
                class Stopwatch

                bool m_paused;
                Timer<ClockT> m_timer;
                ClockT::duration m_timeAccumulated;

                public:

                explicit Stopwatch(bool paused = false):
                m_paused(paused),
                m_timer(),
                m_timeAccumulated(0)




                void Pause()

                m_timeAccumulated += m_timer.GetElapsedTime();
                m_paused = true;


                void Unpause()

                if (m_paused)

                m_timer = Timer<ClockT>();
                m_paused = false;



                ClockT::duration GetElapsedTime() const

                if (m_paused)
                return m_timeAccumulated;

                return m_timeAccumulated + m_timer.GetElapsedTime();

                ;

                #include <iostream>

                int main()

                using seconds_t = std::chrono::duration<float>;

                Stopwatch s;

                for (int i = 0; i != 50; ++i) std::cout << ".";
                std::cout << "n";

                std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

                s = Stopwatch(true);
                std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

                s.Unpause();

                for (int i = 0; i != 50; ++i) std::cout << ".";
                std::cout << "n";

                s.Pause();
                std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;

                for (int i = 0; i != 50; ++i) std::cout << ".";
                std::cout << "n";

                std::cout << std::chrono::duration_cast<seconds_t>(s.GetElapsedTime()).count() << std::endl;







                share|improve this answer













                share|improve this answer




                share|improve this answer










                answered Aug 11 at 13:47









                user673679user673679

                5,6291 gold badge15 silver badges40 bronze badges




                5,6291 gold badge15 silver badges40 bronze badges































                    draft saved

                    draft discarded















































                    Thanks for contributing an answer to Code Review Stack Exchange!


                    • Please be sure to answer the question. Provide details and share your research!

                    But avoid


                    • Asking for help, clarification, or responding to other answers.

                    • Making statements based on opinion; back them up with references or personal experience.

                    Use MathJax to format equations. MathJax reference.


                    To learn more, see our tips on writing great answers.




                    draft saved


                    draft discarded














                    StackExchange.ready(
                    function ()
                    StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fcodereview.stackexchange.com%2fquestions%2f225923%2fa-simple-stop-watch-which-i-want-to-extend%23new-answer', 'question_page');

                    );

                    Post as a guest















                    Required, but never shown





















































                    Required, but never shown














                    Required, but never shown












                    Required, but never shown







                    Required, but never shown

































                    Required, but never shown














                    Required, but never shown












                    Required, but never shown







                    Required, but never shown









                    Popular posts from this blog

                    Tamil (spriik) Luke uk diar | Nawigatjuun

                    Align equal signs while including text over equalitiesAMS align: left aligned text/math plus multicolumn alignmentMultiple alignmentsAligning equations in multiple placesNumbering and aligning an equation with multiple columnsHow to align one equation with another multline equationUsing \ in environments inside the begintabularxNumber equations and preserving alignment of equal signsHow can I align equations to the left and to the right?Double equation alignment problem within align enviromentAligned within align: Why are they right-aligned?

                    Where does the image of a data connector as a sharp metal spike originate from?Where does the concept of infected people turning into zombies only after death originate from?Where does the motif of a reanimated human head originate?Where did the notion that Dragons could speak originate?Where does the archetypal image of the 'Grey' alien come from?Where did the suffix '-Man' originate?Where does the notion of being injured or killed by an illusion originate?Where did the term “sophont” originate?Where does the trope of magic spells being driven by advanced technology originate from?Where did the term “the living impaired” originate?