Accessing static Data and Functions in Legacy C
http://www.renaissancesoftware.net/blog/archives/450 It’s a new year; last year was a leap year; so the quadrennial reports of leap year bugs are coming in. Apologies are in the press from Apple, TomTom, and Microsoft. Trains we stopped f
http://www.renaissancesoftware.net/blog/archives/450
It’s a new year; last year was a leap year; so the quadrennial reports of leap year bugs are coming in. Apologies are in the press from Apple, TomTom, and Microsoft. Trains we stopped from running in China. Somehow calling them glitches seems to make it someone else’s fault, something out of their control. How long have leap years been around? Julius Caesar introduced Leap Years in the Roman empire over 2000 years ago. The Gregorian calendar has been around since 1682. This is not a new idea, or a new bug.
I’m going to try to take one excuse away from the programmers that create these bugs by answering a question that comes up all the time, “How do I test static functions in my C code?”
In code developed using TDD, static functions are tested indirectly through the public interface. As I mentioned in a this article TDD is a code rot radar. It helps you see design problems. Needing direct access to hidden functions and data in C is a sign of code rot. It is time to refactor.
But what about existing legacy code that has statics? It is probably way past the time for idealism and time for some pragmatism. In this article and the next, we’ll look at how to get your code
untouched into the test harness and access those pesky <span>static </span>
variables
and functions.
If you don’t mind touching your code, you could change all mentions of static
to STATIC
.
Then using the preprocessor, STATIC
can
he set to static
during
production and to nothing for test, making the names globally accessible. In gcc you would use these command line options
-
For production builds use
-dSTATIC=static
-
For unit test builds use
-dSTATIC
Let’s look at two options that, at least for access to statics, you will not have to touch your code to get it under test. First is <span>#include</span>
-ing
your .c in the test file. In the next article we’ll build a test adapter .c that give access to the hidden parts.
We are going to revisit code that is similar to the original code responsible for the Zune Bug. I rewrote the code to avoid attracting any lawyers but it is structured similarly to the original Zune driver, complete with static data and functions that must be correct for the function to work.
The header file provides a data structure and initialization function for figuring out the information about the date. Here is the header:
typedef struct Date { int daysSince1980; int year; int dayOfYear; int month; int dayOfMonth; int dayOfWeek; } Date; void Date_Init(Date *, int daysSince1980); enum { SUN = 0, MON, TUE, WED, THU, FRI, SAT }; enum { JAN = 0, FEB, MAR, APR, MAY, JUN, JUL, AUG, SEP, OCT, NOV, DEC };
Date_Init
populates
the Date
instance
passed in. Ignoring the fact that I can probably fully test this by rigging the daysSince1980
,
and inspecting the contents of the resultingDate
structure,
we are going to see how we can directly test the hidden functions and data.
Date_Init
has
three hidden helper functions.
void Date_Init(Date* date, int daysSince1980) { date->daysSince1980 = daysSince1980; FirstSetYearAndDayOfYear(date); ThenSetMonthAndDayOfMonth(date); FinallySetDayOfWeek(date); }
Date_Init
is
the tip of the iceberg. All the interesting stuff is happening in the hidden data and functions:
#include "Date.h" #includeenum { STARTING_YEAR = 1980, STARTING_WEEKDAY = TUE }; static const int nonLeapYearDaysPerMonth[12] = { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }; static const int leapYearDaysPerMonth[12] = { 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }; static bool IsLeapYear(int year) { if (year % 400 == 0) return true; if (year % 100 == 0) return false; if (year % 4 == 0) return true; return false; } static int GetDaysInYear(int year) { if (IsLeapYear(year)) return 366; else return 365; } static void FirstSetYearAndDayOfYear(Date * date) { int days = date->daysSince1980; int year = STARTING_YEAR; int daysInYear = GetDaysInYear(year); while (days > daysInYear) { year++; days -= daysInYear; daysInYear = GetDaysInYear(year); } date->dayOfYear = days; date->year = year; } static void ThenSetMonthAndDayOfMonth(Date * date) { int month = 0; int days = date->dayOfYear; const int * daysPerMonth = nonLeapYearDaysPerMonth; if (IsLeapYear(date->year)) daysPerMonth = leapYearDaysPerMonth; while (days > daysPerMonth[month]) { days -= daysPerMonth[month]; month++; } date->month = month + 1; date->dayOfMonth = days; } static void FinallySetDayOfWeek(Date * date) { date->dayOfWeek =((date->daysSince1980-1)+STARTING_WEEKDAY)%7; } void Date_Init(Date* date, int daysSince1980) { date->daysSince1980 = daysSince1980; FirstSetYearAndDayOfYear(date); ThenSetMonthAndDayOfMonth(date); FinallySetDayOfWeek(date); }
Let’s say you want to check the days per month vectors. You might want to write a test to check the months against the handy poem we use here in the US: Thirty days, has September, April, June and November; all the rest have thirty-one, except for February. It has 28 except in leap year it has 29.
If you started by writing this test…
TEST(Date, sept_has_30_days) { LONGS_EQUAL(30, nonLeapYearDaysPerMonth[SEP]); }
… you get this error:
DateTest.cpp:41: error: 'nonLeapYearDaysPerMonth' was not declared in this scope
Let’s get access to the hidden statics in the test case by including Date.c
instead
ofDate.h
in DateTest.cpp
.
The full test case file looks like this now:
#include "CppUTest/TestHarness.h" extern "C" { #include "Date.c" } TEST_GROUP(Date) { }; TEST(Date, sept_has_30_days) { LONGS_EQUAL(30, nonLeapYearDaysPerMonth[SEP]); }
With a little refactoring days per month vectors can be checked like this:
#include "CppUTest/TestHarness.h" extern "C" { #include "Date.c" } TEST_GROUP(Date) { const int * daysPerYearVector; void setup() { daysPerYearVector = nonLeapYearDaysPerMonth; } void itsLeapYear() { daysPerYearVector = leapYearDaysPerMonth; } void CheckNumberOfDaysPerMonth(int month, int days) { LONGS_EQUAL(days, daysPerYearVector[month]); } void ThirtyDaysHasSeptEtc() { CheckNumberOfDaysPerMonth(SEP, 30); CheckNumberOfDaysPerMonth(APR, 30); CheckNumberOfDaysPerMonth(JUN, 30); CheckNumberOfDaysPerMonth(NOV, 30); CheckNumberOfDaysPerMonth(OCT, 31); CheckNumberOfDaysPerMonth(DEC, 31); CheckNumberOfDaysPerMonth(JAN, 31); CheckNumberOfDaysPerMonth(MAR, 31); CheckNumberOfDaysPerMonth(MAY, 31); CheckNumberOfDaysPerMonth(JUL, 31); CheckNumberOfDaysPerMonth(AUG, 31); } void ExceptFebruaryHas(int days) { CheckNumberOfDaysPerMonth(FEB, days); } }; TEST(Date, non_leap_year_day_per_month_table) { ThirtyDaysHasSeptEtc(); ExceptFebruaryHas(28); } TEST(Date, leap_year_day_per_month_table) { itsLeapYear(); ThirtyDaysHasSeptEtc(); ExceptFebruaryHas(28); }
You have access to all the hidden stuff, so you can write the test for the static functions:
IsLeapYear()
, GetDaysInYear()
, FirstSetYearAndDayOfYear()
,ThenSetMonthAndDayOfMonth()
,
and FinallySetDayOfWeek()
.
If Date
been
an abstract data type, with its data structure hidden in the .c
file,
the tests would all have access to structure members hidden from the rest of the world.
There is a downside to this approach, which probably does not matter in this case, but could in others. You can only have one test file that can include a given .c
file.
In the next article we’ll solve that problem.
Have you heard of any interesting leap year bugs? Did you prevent your own?

Alat AI Hot

Undresser.AI Undress
Apl berkuasa AI untuk mencipta foto bogel yang realistik

AI Clothes Remover
Alat AI dalam talian untuk mengeluarkan pakaian daripada foto.

Undress AI Tool
Gambar buka pakaian secara percuma

Clothoff.io
Penyingkiran pakaian AI

AI Hentai Generator
Menjana ai hentai secara percuma.

Artikel Panas

Alat panas

Notepad++7.3.1
Editor kod yang mudah digunakan dan percuma

SublimeText3 versi Cina
Versi Cina, sangat mudah digunakan

Hantar Studio 13.0.1
Persekitaran pembangunan bersepadu PHP yang berkuasa

Dreamweaver CS6
Alat pembangunan web visual

SublimeText3 versi Mac
Perisian penyuntingan kod peringkat Tuhan (SublimeText3)

Topik panas

Folder data mengandungi data sistem dan program, seperti tetapan perisian dan pakej pemasangan Setiap folder dalam folder Data mewakili jenis folder storan data yang berbeza, tidak kira sama ada fail Data merujuk kepada nama fail Data atau sambungan data , semuanya adalah fail data yang disesuaikan oleh sistem atau program Data ialah fail sandaran untuk penyimpanan data Secara umumnya, ia boleh dibuka dengan meidaplayer, notepad atau word.

Peranan dan penggunaan statik dalam bahasa C: 1. Skop pembolehubah; 3. Fungsi dalaman 4. Ubah suai fungsi; Jika terdapat kata kunci statik di hadapan pembolehubah, maka skop pembolehubah adalah terhad kepada fail di mana ia diisytiharkan Dalam erti kata lain, pembolehubah adalah "skop peringkat fail", yang sangat berguna untuk mencegah masalah "definisi pendua" pembolehubah; 2. Kitaran hayat, pembolehubah statik dimulakan sekali apabila program mula dilaksanakan, dan dimusnahkan apabila program tamat, dsb.

1. statik Sila lihat program berikut dahulu: publicclassHello{publicstaticvoidmain(String[]args){//(1)System.out.println("Hello, world!");//(2)}} Pernah melihat ini Program segmen biasa kepada kebanyakan orang yang telah mempelajari Java. Walaupun anda belum mempelajari Java tetapi telah mempelajari bahasa peringkat tinggi yang lain, seperti C, anda seharusnya dapat memahami maksud kod ini. Ia hanya mengeluarkan "Hello, dunia" dan tidak mempunyai kegunaan lain Walau bagaimanapun, ia menunjukkan tujuan utama kata kunci statik.

Senario aplikasi praktikal dan kemahiran penggunaan kata kunci statik dalam bahasa C 1. Gambaran keseluruhan statik ialah kata kunci dalam bahasa C, digunakan untuk mengubah suai pembolehubah dan fungsi. Fungsinya adalah untuk mengubah kitaran hayat dan keterlihatannya semasa program berjalan, menjadikan pembolehubah dan fungsi statik. Artikel ini akan memperkenalkan senario aplikasi praktikal dan teknik penggunaan kata kunci statik, dan menggambarkannya melalui contoh kod tertentu. 2. Pembolehubah statik memanjangkan kitaran hayat pembolehubah Menggunakan kata kunci statik untuk mengubah suai pembolehubah tempatan boleh memanjangkan kitaran hayatnya.

Fungsi statik: 1. Pembolehubah; 3. Kegunaan lain 5. Pengoptimuman prestasi; Pengoptimuman Reka Letak Memori; 11. Elakkan pemulaan berulang; Pengenalan terperinci: 1. Pembolehubah, pembolehubah statik Apabila pembolehubah diisytiharkan sebagai statik, ia tergolong dalam peringkat kelas, bukan peringkat contoh, yang bermaksud bahawa tidak kira berapa banyak objek yang dicipta, hanya satu pembolehubah statik wujud, dan semua objek. kongsi pembolehubah Statik ini dan sebagainya.

Penyelesaian kepada data beban mysql yang bercelaru: 1. Cari pernyataan SQL dengan aksara yang bercelaru;

Operator SQLAND&OROperator AND dan OR digunakan untuk menapis rekod berdasarkan lebih daripada satu syarat. DAN dan ATAU menggabungkan dua atau lebih syarat dalam subnyata WHERE. Operator AND memaparkan rekod jika kedua-dua syarat pertama dan kedua adalah benar. Operator OR memaparkan rekod jika sama ada syarat pertama atau syarat kedua adalah benar. Jadual "Orang": LastNameFirstNameAddressCityAdamsJohnOxfordStreetLondonBushGeorgeFifthAvenueNewYorkCarter

Perbezaannya ialah: 1. xdata biasanya merujuk kepada pembolehubah bebas, manakala data merujuk kepada keseluruhan set data 2. xdata digunakan terutamanya untuk membina model analisis data, manakala data digunakan untuk analisis data dan statistik; untuk Analisis regresi, analisis varians, pemodelan ramalan, data boleh dianalisis menggunakan pelbagai kaedah statistik 4. xdata biasanya memerlukan prapemprosesan data, dan data boleh mengandungi data asal yang lengkap;
