08-11-2016 06:34 AM
Hi Guys,
I'm creating a VI that will allow me to log data at time intervals of .1, .2, .3, .4, .5, .6, .7, .8, .9, 1, 2, 3, 4 etc seconds or 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 20, 30 etc seconds depending on my input from my main VI.
In the VI I have attached I have a Hold Down case and a Toggle case. I've gone through the Hold Down case with highlight execution and it seems to be ok. My problems start when using the Toggle case. If I initiate the Stress Time In control at 0, it increments to .1, .2, .3, .4, .5, .6, .7, .8 .9 and 1 as expected. However, when transitioning from adding .1 into the case structure where it is supposed to start adding 1's, it ends up adding a .1 and a 1 for the first time and then continues to add 1 as expected. I've traced it back in the code and when I turn on highlight execution with a 1 coming from the shift register, it is input into the "Less than" function in the bottom case structure and it outputs a True when it should output a False. My top case structure also outputs a zero when it should start outputting 1.
Can someone please run this code and see if the problem exists for them to preserve my sanity? Or am I just being an idiot and missing something obvious because of a crossed wire or something?
Thanks for your help
Solved! Go to Solution.
08-11-2016 06:51 AM - edited 08-11-2016 06:54 AM
From the sounds of it, the problem is to do with DBL floating point comparisons. The way that a DBL represents 1 is the best approximation it can store in 32-bits. It isn't necessarily exactly 1.0...it might be stored as 1.000000000000000000000000001. If you change the display precision to a really high number of decimal places, you'll see the tiny difference in the values and it will make sense as to why the result of the comparison isn't what you're expecting.
You also have to be really careful when checking if a DBL 0 is equal to 0. It might be 0 at first glance, but it could also be -0 or 0.0000000000000000000001
08-11-2016 06:58 AM
The problem is with Floating Point representation which you used for comparision.
http://digital.ni.com/public.nsf/allkb/B01682241DD825948625665100663F61
http://digital.ni.com/public.nsf/allkb/49D72E8658F6AEE48625758A00710BB4?OpenDocument
08-11-2016 07:58 AM
Hi Guys,
Thank you for the responses. I've gone through the links posted above and have read through the proposed solutions. Is there any easier way that I could just reduce the precision of the floating point to just a few digits? If I understand correctly, I can't use integers because my values aren't whole numbers to begin with. Or is it better to just use the range and coerce function to make the code work? Thank you.
08-11-2016 08:04 AM
08-11-2016 08:06 AM
You basically have several choices:
Of these, the last is likely to be (by far) the fastest, and will "remind" you that fractional values are typically "approximations" (you can represent 0.5, 0.25, 0.125, etc. exactly, but not 0.1, for example).
Bob Schor
08-11-2016 08:33 AM
Multiplied both by 10 and its working correctly now! Thank you everyone!
08-11-2016 08:50 AM
Here's an old post by Altenbach that shows a way to perform an Almost Equal on doubles by ignoring some amount of bit tolerance.
Unofficial Forum Rules and Guidelines
Get going with G! - LabVIEW Wiki.
17 Part Blog on Automotive CAN bus. - Hooovahh - LabVIEW Overlord
08-11-2016 09:26 AM - edited 08-11-2016 10:10 AM
(Sorry, I am a bit late to the party because I am on the west coast, but here are my thoughts).
Yes, The core reason for your observation is the limitations of DBL representation, but I think you are looking in the wrong place. Your code is just a plain monstrosity and I am sure that your problem can be solved with 5% of the code you currently have. Try to find a more elegant way to do all this. Having dozens of case structure and add primitives cannot be the solution.
As a first step, explain what the output as a function of input should be. What is the range of allowed inputs (e.g. can it be negative?).
As a second step, modify the code so we can actually test it. Currently your code gets trapped in the while loop and no output is ever produced because the loops will run forever. This cannot be right!
To do range checking, it is typcailly easiest to threshold the input value into an array of boundaries.
Also having clean code is a big step towards readability. It is very difficult to follow all these crisscrossing wires and the jumble of randomly placed comparison and boolean operations. Also stick to one datatype, you have many coercion dots for no good reason at all. You have several diagram constants with the same value. One each is enough, just branch the wire!
In any case, please tell us what the two cases are supposed to output as a function of the input. Maybe even a simple graph or table of some representative "stress time out" vs "stress time in".
I think you are digging yourself into a hole here and multiplying by 10 is just a bandaid that needs to be ripped of and replaced once you need to adapt the code to deal with e.g. 0.01 second increments. Instead of actually fixing the problem, you just sweep it under the rug.
08-11-2016 10:07 AM
Hi,
Thank you for the reply. This VI is a sub-VI in a VI which does data collection and logs it against the stress time. I had the stop condition set to false while I was testing so that I wouldn't have to go in and out of the main VI to see what was happening when I was using highlight execution.In use within the main VI, this VI's stop condition is set to True and I have the shift registers connected to my While loop of my main VI. Apologies but I cannot send on the code of my main VI.
I'm making the code for a colleague and we will never have to measure below an interval of .1 seconds or above 10000 seconds so I wrote the code with those limits in mind.
Apologies for the messy code. I'm usually a stickler for having clean readable code but that went out the window for this time measuring VI. I guess some background to the function of the code is as follows: I have 2 possible types of experiment, Hold Down and Toggle. I want my Toggle test to initially log more frequent values than the equivalent Hold Down test but after 1 second has elapsed, they are both able to be the same interval.
So in my main VI with a Toggle experiment selected, I want to measure @t0, wait for .1 seconds, measure, wait .1 seconds etc until 1 second of total experimental time has elapsed. When 1 second has elapsed I want to measure, wait for 1 second, measure, and wait for 1 second etc etc until the total experimental time is 10 seconds. When 10 seconds has elapsed I want to measure, wait for 10 seconds, measure, and wait for 10 seconds etc etc until the total experimental time is 100 seconds. And I keep doing this until I get to 10000 second intervals between measurements and when this point is reached, a constant measurement interval of 10000 seconds is output from the loop.
If I select a Hold Down test, my code performs the same function but it cuts out the intervals from 0-1 second and jumps straight to 1 second intervals.
I'd love to hear ways to improve and alter this because I spent a loooong time trying to figure out a way to do it to suit my needs.