Önceki yazılarda, bir C programı hakkında feyizli bir takım konular üzerinde konuşmuştuk. Merhaba dünya yazılımının anatomisinden sonra ibretlik veri tiplerini, değişkenler ile onlara atanacak sabitleri, ardından da alımlı kodun olmazsa olmazı yorum satırlarını incelemiştik. Bu defa ise, C dilindeki bir diğer önemli başlık olan operatörleri inceleyeceğiz.
Dünyada çeşit çeşit operatör vardır ancak ne mutlu bize ki C dilindeki operatörler çeşitleri sınırlı sayıda. 🙂 Neyse, “operatörlere özgü bir yazı” olması, ilk bakışta biraz garip guraba gelebilir ama, operatörler özellikle gömülü programlamada “çok bilinçli şekilde kullanılmak zorunda” olduğundan değer verilmesi gereken önemli bir konu. Bizim de niyetimiz operatörlerin hak ettiği saygınlığı kazanmasına yardımcı olabilmek.
Bazen aynı amaç, farklı operatörler ile gerçeklenebilir. Hatta genelde durum böyledir. Bu aşamada, hangisini seçmenin daha akıllıca olacağına karar vermenizi sağlayacak birtakım ince noktalara burada değinmeye çalışacağım. Operatörlerin doğru kullanımında dahi bazı kazan/kaybet (trade-off) durumları olduğundan, yine ibretlerle dolu bir yazının şanlı başlangıcında olduğumuzu hissediyorum. Haydi hayırlısı…
Aritmetik Operatörler
C dilinde aritmetik operatörler, bizleri çocukluğumuza, ilkokul sıralarına götürür 🙂
Not: Operatörlerin uygulandığı değişken veya sabitlere genel olarak operand denebilir. A=10, B=20 için;
Operatör | Tanım | Örnek |
---|---|---|
+ | iki operandı toplar | A + B = 30 |
– | Soldaki operanddan sağdaki operandı çıkarır. | A – B = -10 |
* | operandları çarpar | A * B = 200 |
/ | soldaki operandı sağdaki operanda böler | B / A = 2 |
% | Mod operatörü, bölmeden sonra kalanı verir | B % A = 0 |
++ | Bir artırma operatörü | A++ = 11 |
— | Bir azaltma operatörü | A– = 9 |
Aritmetik operatörleri etkin şekilde öğrenmenin en iyi yolu, bu operatörleri kullanarak programlar yazmak ve bunları çalıştırarak sonuçları görmektir. Benim derdim, operatörlerin ne iş yaptığını anlatmak olmamalı diye düşünüyorum. Herhangi bir C dersinde/kitabında bu zaten alenen anlatılıyor. Aşağıdaki örnek yazılımı yazmamın ardından asıl anlatılması gerken ince noktalara geçeceğiz.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 |
#include <stdio.h> #include <stdint.h> #define NUM_A (10.0) #define NUM_B (20.0) #define TOPLA(A,B) (A+B) #define CIKAR(A,B) (A-B) #define CARP(A,B) (A*B) #define BOL(A,B) (A/B) #define MOD(A,B) (A%B) int main() { uint32_t i; printf("Toplama: %fnCikarma : %fnCarpma: %fnBolme: %fnMod: %dn",TOPLA(NUM_A,NUM_B),CIKAR(NUM_A,NUM_B),CARP(NUM_A,NUM_B),BOL(NUM_A,NUM_B),MOD((uint32_t)NUM_A,(uint32_t)NUM_B)); i=30; printf("++ sonra kullanilinca i:%dn",i++); printf("bir sonraki adimda i:%dn",i); i=30; printf("++ once kullanilinca i:%dn",++i); printf("bir sonraki adimda i:%dn",i); getchar(); //kullanici bir giris yapana kadar bekler return 0; } |
Yukarıdaki kod çalıştırıldığında, çıktısı aşağıdaki gibi olur:
++ sonra kullanilinca i:30
bir sonraki adimda i:31
++ once kullanilinca i:31
bir sonraki adimda i:31
Önemli bir nokta olarak i++ yazdırıldığında, önce i’nin değerinin yazdırılıp ardından bir artırıldığında dikkat ediniz. Aynı şekilde ++i yazdırıldığında, önce i’nin değerinin bir artırıldığında ardından bu değerin yazdırıldığına çok dikkat ediniz. İnananlar için burada büyük ibretler var. ++ opratörünü bu bilinçle kullanmakta faydalar var.
Aritmetik operatörlerle ilgili önemli bir diğer nokta da öncelik sırası kavramı. Öncelik sırasına takılmamak ve kodu okunur kılmak adına, ardışıl aritmetik işlemlerde parantez kullanaraktan sorunu teğet geçmekte fayda var. Örneğin 13+227-12/3 ile ne demek istediğimi anlayamayabilirsiniz, demek istediğimi yanlış diyor da olabilirim ama 12+(227)-(12/3) ile ne demek istediğimiz çok açık olur. C dilinde malum her zaman parantezin içi öncelikli oluyor. Hem hataları önlemek için hem de toplum içerisinde anlaşılabilir olmak için bu alışkanlığı edinmekte faiderler, feyizler var.
Aritmetik operatörlerle ilgili bir diğer önemli nokta da, yerli yersiz kullanılmamaları. Kimi zaman (hatta çoğu zaman), bir işi yapmanın birden fazla yolu oluyor. Örneğin 8 bitlik iki sayıyı alıp yanyana koyup 16 bitlik tek bir sayı elde etmek istediğimiz yerler oluyor. Bunu yapmak aşağıdaki şekillerde mümkün:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 |
uint8_t lowByte; uint8_t highByte; uint16_t myWord; //yöntem 0 myWord = (highByte*256) + lowByte; //yöntem 1 myWord = (highByte<<8) + lowByte; //yöntem 3 myWord = (highByte<<8) | lowByte; //yöntem 4; typedef union { uint8_t au8Data[2]; uint16_t u16Data; }tu_DataGroup16; tu_DataGroup16 u_DataGroup16; u_DataGroup16.au8Data[0] = lowByte; u_DataGroup16.au8Data[1] = highByte; myWord = u_DataGroup16.u16Data; |
Bazı platformlarda toplama, çarpma gibi aritmetik işlemler atomik değildir, yani bu operatörlerin işletilmesi 1 saat darbesi süresinden (1 cycle) fazla sürer. Halbuki her işlemcide AND,OR işlemleri tek cycle’da yapılır. Buna göre eğer yukarıdaki kodu MSP430 serisi mikrokontrolörler için yazıyorsanız, yöntem 0 sizi canınızdan bezdirir çünkü eğer HW multiplicator olmayan bir modelde çarpma 130 cycle kadar sürer, HW multiplicator olan bir modelde bile çarpma 8 cycle sürer. Daha bunun toplaması ve ataması da var. Halbuki Yöntem 3’te tüm olay hepi topu 3 cycle’da biter. Yöntem 4’ün ise çok sanatsal bir amacı var, umarım anlamışsınızdır ancak anlamadıysanız dahi ileride detaylıca anlatacağım zaten.
Neyse demek ki yerli yersiz aritmetik operatör kullanmamak lazımmış. Gerçi Arm Cortex M3 mimarisinde aritmetik operatörler de 1 cycle sürüyor (non-fp için). Orada elinizi korkak alıştırmayabilirsiniz ama yine de gün gelir o kodu 8 bitlik bir mikrokontrolöre taşımanız gerekirse başınız ağrır. En güzeli, en baştan çiçek gibi bilinçle yazmak 🙂
İlişkisel Operatörler/ Karşılaştırma Operatörleri
İlişkisel operatörler sağ ve solundaki operandlar belirli bir mantıksal ilişkiye göre karşılaştırılır ve eğer ilişki doğru ise doğru döndürülür, yanlışsa yanlış. 😀
Operatör | Tanım | Örnek |
---|---|---|
== | İki operand birbirine eşitse, sonuç doğru(true,1) olur | (A == B) |
!= | İki operand birbirine eşit değilse, sonuç doğru(true,1) olur | (A != B) |
> | Soldaki operand sağdakinden büyük ise, sonuç doğru(true,1) olur | (A > B) |
< | Soldaki operand sağdakinden küçük ise, sonuç doğru(true,1) olur | (A < B) |
>= | Soldaki operand sağdakinden büyük ya da eşit ise, sonuç doğru(true,1) olur | (A >= B) |
<= | Soldaki operand sağdakinden küçük ya da eşit ise, sonuç doğru(true,1) olur | (A <= B) |
Bu kısımla ilgili verilecek ibretlik mesajı önceki kısımlarda vermiştik. Ancak tekrar bir hatırlatma yapmak gerekirse;karşılaştırma operatörleri genellikle if,while gibi koşul/döngü işlemlerinde kullanılır ve doğal olarak en çok hata da buralarda yapılır. Örneğin sık yapılan hatalardan biri if(age == 18) yazacakken yanlışlıkla if(age=18) yazmaktır. Bu durumda age değişkenini 18 ile karşılaştırmak yerine, age değişkenine 18 atamış ve dönüş olarak true almış olursunuz. If içinde atama yapmak mümkün olduğundan derleyici buna hata vermez, uyarı verebilir. Bu hatayı hiç yapmamanın yolu if(18 == age) formatını kullanmaktır. Yani sabiti sola, değişkeni sağa almaktır. Bu durumda yanlışlıkla if(18 = age) yazsanız bile, derleyici size sabite değişken atanmaz diye hata verir. Bu arada 18 yazdım ama, örnek anlaşılsın diyerekten sayıyı doğrudan yazdım 🙂 Yoksa MAGIC_NUMBER kullanmak yok 😉 Önerdiğim formatta yazacak olursam eğer if(ageOfDriverCandidate == AGE_LIMIT_TO_DRIVE) yerine if(AGE_LIMIT_TO_DRIVE == ageOfDriverCandidate) yazmakta fayda var. Ya da bu hususta sık hata yapıyorsanız, bir makro yaratarak, ve onu bir kez test ederek bu sorunu tarihin tozlu sayfalarına gömebilirsiniz :
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
#define IS_EQUAL(ARG1,ARG2) (ARG1==ARG2) #define ESIT_MI(OP1,OP2) (OP1 == OP2) //English #define AGE_LIMIT_TO_DRIVE (18U) uint8_t ageOfDriverCandidate; .. if(IS_EQUAL(AGE_LIMIT_TO_DRIVE ,ageOfDriverCandidate)) { //rockie detected } .. //Turkce #define EHLIYET_YAS_SINIRI (18U) uint8_t surucuAdayininYasi; .. if(ESIT_MI(EHLIYET_YAS_SINIRI ,surucuAdayininYasi)) { //ahanda caylak geldi } |
Mantıksal Operatörler
Bazen birden fazla koşul aynı anda değerlendirilmek istenebilir. Bu durumda mantıksal ilişkiye göre (ve,veya,değil) koşullar değerlendirilebilir.
Operatör | Tanım | Örnek |
---|---|---|
&& | Mantıksal VE(AND) operatörü çağırılır. Eğer iki taraf da sıfırdan farklı ise, sonuç doğru(true) olur. | (A && B) |
|| | Mantıksal VEYA(OR) operatörü çağırılır. Eğer iki taraftan herhangi biri sıfırdan farklı ise, sonuç doğru(true) olur. | (A || B) |
! | Mantıksal DEĞİL(NOT) operatörü çağırılır. Operandın mantıksal tersi alınır; operand 1 ise sonuç 0, operand 0 ise sonuç 1 olur. | !(A && B) |
Mantıksal operatörler birleştirilerek kullanılabilir. Ancak insan aklının almayacağı uzunlukta bir koşul söz konusu ise, o koşul ya indirgenmeli, ya da yorum satırları ile açıklanmalıdır. İnsan aklı derinlik olarak ortalama 7 seviye ilişkiye kadar hafızada tutabilir. Fazlası olmaz 🙂 if içinde 9 tane koşul yazmamakta fayda var.
1 2 3 4 |
if((isUserAuthenticated && (userAge>=AGE_LIMIT_TO_DRIVE)) || isUserBilal) { //do not touch } |
Bitsel/İkili (Bitwise/Binary) Operatörler
Esasen bitsel operatörlerin ciğerini BURADA, çiçek gibi açıkladık. Hatta elektronik olarak nasıl gerçeklendiğini de konuştuk. Ancak tabi ki olayı bir de yazılım perspektifinde incelemekte sayısız feyizler var. Zira gömülü sistemlerde kimi zaman bir bitin bile hesabı yapılır. Hesabının yapılmadığı yerde bile, kullanılan mikrokontrolörün bir registerındaki bir biti 1 ya da 0’a çekmek için bitsel operatörler kullanılabilir. Bu söylediğim ilk aşamada bir anlam ifade etmemiş olabilir, ancak ileride değineceğimiz üzere mikrokontolördeki donanımı kullanabilmek için, genişliği genelde işlemcinin adres yolu genişliği kadar olan (8/16/32 bit), ayırılmış özel bellek alanlarına (register) bir takım veriler yazmak gerekir. Belleği verimli kullanmak için bir register içindeki her bitin ayrı ayrı işlevi olur. Bu nedenle registerlara bitsel olarak erişmek gerekir. Burada da genellikle bitsel operatörler kullanılır. Yani mevzu, o kadar önemli ki, ateş ediyor =)
Misal A = 60; ve B = 13; olsun. Bunların ikili tabanda gösterimi şöyle olur:
A = 0011 1100
B = 0000 1101
Buna göre
A&B = 0000 1100
A|B = 0011 1101
A^B = 0011 0001
~A = 1100 0011
Operatör | Tanım | Örnek |
---|---|---|
& | Bitsel VE Operatörü | (A & B) = 12, yani 0000 1100 |
| | Bitsel VEYA Operatörü | (A | B) = 61, yani 0011 1101 |
^ | Bitsel XOR Operatörü | (A ^ B) = 49, yani 0011 0001 |
~ | Tümleyen operatörü. Operandın 1 olan bitlerini 0, 0 olan bitlerini de 1 yapar. | (~A ) = -61, yani 1100 0011 |
<< | Sola kaydırma operatörü. Soldaki operandın bitleri, sağ operand kadar sola kaydırılır ve her kaydırmada en sola 0 eklenir. (Karışık oldu, örnek üzerinden görmekte fayda var) | A << 2 = 240 yani 1111 0000 |
>> | Sağa kaydırma operatörü. Soldaki operandın bitleri, sağ operand kadar sağa kaydırılır ve her kaydırmada en sağa 0 eklenir.(Karışık oldu, örnek üzerinden görmekte fayda var) | A >> 2 = 15 yani 0000 1111 |
Şimdi gelelim fasülyenin faydalarına 🙂 Yani her yerde yazmayan önemli bilgilere…
& (VE) operatörü ile maskeleme yapılarak bir sayının 1 olan bitlerini 0’a çekmek yani temizlemek mümkündür. Misal:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
/*Asagidakilerin hepsi PORTB'nin 3. bitini 0'a çeker.Bu işlem gerçekte PIC mikrodenetleyicilerde B portunun 3 pinini lojik sıfıra çeker, yani o pinde 0V görülür. */ PORTB = PORTB & 0b11110111; //ya da ayni sey: PORTB = PORTB & 0xF7; //ya da ayni sey: PORTB &= ~(1<<3); //ya da macro yazalim yine ayni sey #define CLEAR_BIT(DATA,BIT_INDEX) do(DATA&= ~(1<<BIT_INDEX);)while(0) CLEAR_BIT(PORTB ,3); |
| (VEYA) operatörü ile maskeleme yapılarak bir sayının 0 olan bitlerini 1’e çekmek yani set etmek! (kurmak) mümkündür. Misal:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
/*Asagidakilerin hepsi PORTB'nin 3. bitini 1'e çeker.Bu işlem gerçekte PIC mikrodenetleyicilerde B portunun 3 pinini lojik bire çeker, yani o pinde 5V görülür. */ PORTB = PORTB | 0b00001000; //ya da ayni sey: PORTB = PORTB | 0x08; //ya da ayni sey: PORTB |= (1<<3); //ya da macro yazalim yine ayni sey #define SET_BIT(DATA,BIT_INDEX) do(DATA|= (1<<BIT_INDEX);)while(0) SET_BIT(PORTB ,3); |
^ (XOR) operatörü candır! Bu operatör ile maskeleme yaparak bir sayının herhangi bir bitini (ya da bitlerini) terslemek mümkündür.
1 2 3 4 5 6 7 8 9 10 11 12 13 |
/*Asagidakilerin hepsi PORTB'nin 3. bitini tersler.Bu işlem gerçekte PIC mikrodenetleyicilerde B portunun 3 pin lojik birde ise bu pini lojik sıfıra çeker, yani o pinde 5V varken artık 0V görülür. Benzer şekild epin lojik sıfırda ise bu pini lojik bire çeker, yani o pinde 0V varken artık 5V görülür */ PORTB = PORTB ^ 0b00001000; //ya da ayni sey: PORTB = PORTB ^ 0x08; //ya da ayni sey: PORTB ^= (1<<3); //ya da macro yazalim yine ayni sey #define TOGGLE_BIT(DATA,BIT_INDEX) do(DATA^= (1<<BIT_INDEX);)while(0) TOGGLE_BIT(PORTB ,3); |
Ek olarak kaydırma operatörünün, cyclic olup olmadığı her platformda mutlaka test edilmelidir. Normalde cyclic olmaz, ama test etmekte fayda var =)
Yazdığım örneklerde zaten kaydırma operatörünün kullanımı da var. Bir de ^=, +=, |= gibi örneklerden atama operatörünün diğer bitsel ve aritmetik operatörlerle birleştirilebildiğini de göstermiş olduk diye düşünüyorum.
Bu gün bu yazımı Mustafa Keser üstadımızın sözü ile sonlandırıyorum.
“Eveeeeet, bir proğramın daha sonuna geldiiik. Tekrar görüşmek üzere efeem.”
Önceki Sayfa Sonraki Sayfa
Merhaba Özen Hocam,
Yazılarınızı sırayla özümseye özümseye okuyorum. Yazılarınızın devamını da bekliyoruz. Ülkemizde gömülü sistemlerde sizin gibi değerli insanların olması bize gurur veriyor. Umarım sizin gibi bir çok üstada da örnek olur. Yazıların anlatımı da çok güzel. Çok teşekkür ederiz.
Gözüme birkaç yer takıldı.
*Bir azaltma operatörü tek eksi ile değilde 2 eksi ile olmalı
*Yöntem4 de union içinde “uint8_t au8Data[2]; “olarak tanımlamışsınız ancak programın devamında u8Data olarak kullanılmış
*| (VEYA) operatörü ile maskeleme kısmında 2. yöntemde araya ve operatörü girmiş.
*rockie out rookie in 🙂
Merhaba Fatih Bey,
Öncelikle geri bildirimleriniz için ve yorumlarınız için çok teşekkür ederim. Bulduğunuz hataları paylaştığınız için ayrıca teşekkür ederim. İlki dışında diğerlerini düzelttim. İlkinde de yazıda — (iki eksi ile) yazdım ancak font dolayısıyla tek eksi gibi gözüküyor. Onun için kısa zamanda yazıların fontunu değiştireceğim 🙂
Yeni yazıları da olabildiğince hızlı şekilde eklemeye çalışacağım 🙂