C# Müsahibə sualları (Interview Questions by Ilkin Rufullayev)

1) C# nədir ?

C# Microsoft tərəfindən 2001-ci ildə yaradılan obyekt yönümlü proqramlaşdırma dilidir. C# proqramlaşdırma dili C və C++ dillərinin üstün tərəflərini alaraq yaradılıb. Hazırda .Net platformunda ən çox istifadə edilən proqramlaşdırma dilidir. C# ilə web proqramlaşdırma,oyun proqramlaşdırma,masaüstü proqramlar yazmaq və hətta mobile proqramlaşdırma da mümkündür.

2) Sinif (class) nədir?

Siniflər dəyişənlərı,metodları,propertiləri yığdığımız bir yerdir. Bu metodlari,dəyişənləri istifadə etmək üçün isə həmin sinifdən obyekt yaradılmalıdır. Sinifə bir nümunə olaraq İnsanı göstərə bilərik. İnsandakı xüsusiyyətlər məs. göz rəngi,adı,yaşı dəyişənlərimiz,onun danışması,gülməsi,qaçması kimi feili hərəkətləri isə metodlarımız olsun. Hər insan bu xüsusiyyətlərə fərd olaraq sahibdir. Proqramlaşdırmada da siniflərdə yazdığımız xüsusiyyətlər, metodlar obyektlərə aid edilir.Obyektlər bizim fərdlərimizdir.

class Person{
  public string eyeColor;
  public string name;
  public int age;
  
  void Talk(){
    //codes
  }
  
  void Laugh(){
    //codes
  }
}

3) Obyekt (object) nədir?

Obyekt bir sinifdən yaradılan və yaddaşda yer tutan bir verilən strukturudur. Obyekt sinifin nümunəsidir,sinifin içərisində tanımlanılan metodlara, dəyişənlərə və s. obyektlər vasitəsilə çatmaq mümkündür.Obyektlər siniflərdə yazdığımız xüsusiyyətlərə,metodlara sahib olur. Sinifdən bir obyekt yaratmaq üçün new açar sözündən istifadə edilir. Məsələn aşağıdakı nümunədə Person sinifə məxsus age dəyişəni obj obyekti vasitəsi ilə çağırılıb və 5 dəyəri mənimsədilib.

  Person obj = new Person();
  obj.age = 5;

4) C# propertilər nədir?

C#-da propertilər dəyişənlərə çox bənzəyir lakin adi dəyişənlərdən fərqli olaraq get və set bloklarına malikdir. Bu bloklar dəyərin oxunması və mənimsədilməsi zamanı çalışır. Məsələn , propertiyə dəyər mənimsədərkən set blokları çalışır , set blokları vasitəsi ilə mənimsədilən dəyər üzərində müəyyən əməliyyatlar icra etdikdən sonra dəyəri get bloklarına ötürüb ekrana çap edə bilərik.

  class Test{
    private int number;
    public int Number {get{ return number; } set{ number = value;} } //value menimsetdiyimiz deyeri (12) temsil edir
  }
  
  static void Main(string[] args){
    Test test = new Test();
    test.Number = 12; // Set blokları çalışdı
    Console.WriteLine(test.Number); // Get blokları çalışdı
  }
  

5) OOP prinsipləri hansılardır?

Obyekt yönümlü proqramlaşdırmada 4 əsas prinsip mövcuddur.

  • Encapsulation (Kapsullama)
  • Inheritance (Mirasalma)
  • Abstraction (Mücərrədlik)
  • Polymorphism (Çox yönümlülük)

Encapsulation (Kapsullama)
Encapsulation siniflərdə yazdığımız dəyişənlərin əlçatılabilirliyini propertilər vasitəsilə məhdudlaşdırmaq və istifadəçilərin hər dəyişənə rahatlıqla çata bilməsinə maneə olmaq məqsədi ilə istifadə edilir. Məsələn, aşağıdakı nümunədə:

 Person obj = new Person();
 obj.age = 5;

age dəyişəninə 5 dəyərini mənimsətdik. Burada hər şey normal görsənir,yaxşı bəs əgər 5 deyil -5 mənimsətsəydik? Texniki baxımdan bunu etməyimiz heç bir xəta səbəbi olmayacaqdı,çünki age dəyişənimizin tipi int-dir və int mənfi dəyərlər ala bilir. Amma insanın yaşı mənfi ola bilməz. Belə halların qarşısını almaq üçün age dəyişənini kapsullayaq.

class Person{
    private int age;
    public int Age 
    {
      get{ return  age; } 
      set
      {
        if(value > 0)
          age = value;
      }
   } 
}

Bildiyimiz kimi propertilerdeki set bloku dəyər mənimsədən zaman işə düşür,biz də həmin dəyərin 0-dan böyük olduğu vəziyyəti yoxlayıb neticeni ona görə qaytardıq,artıq bu halda istifadəçi age dəyişəninə mənfi dəyərlər mənimsədə bilməz (əks halda 0 nəticəsi alacağıq).

Inheritance (Mirasalma)
Bizim siniflərimiz sayca çox olacaq.Bənzər xüsusiyyətlərə sahib siniflərin içərisində eyni kodu təkrar-təkrar yazmaq yerinə həmin kodları üst sinifdən (Base class) miras almaq daha əlverişli olacaq. Məsələn,

class Employee{
  public string eyeColor;
  public string name;
  public int age;
}

class Manager{
  public string eyeColor;
  public string name;
  public int age;
}


şəklində EmployeeManager sinifləri üçün dəyişənləri bu şəkildə yaratmağımızda heç bir problem yoxdur amma fikir verdinizsə eyni kodları yazdıq üstəlik bu xüsusiyyətlər işçi ya da menecer olmağından asılı olmayaraq hər bir insana aiddir. İndi isə bu kodları qısaldaq :

class Person{
  public string eyeColor;
  public string name;
  public int age;
}

class Employee:Person{
}

class Manager:Person{
}

daha əvvəl EmployeeManager siniflərində yazdığımız eyni kodların hər insana məxsus olduğunu demişdik,indi o dəyişənləri Person adında bir sinifə yazıb yuxarıdakı qayda ilə həm Employee həm də Manager siniflərində istifadə edə bilərik. Beləliklə, EmployeeManager sinifləri Person sinifindən onun xüsusiyyətlərini miras almış oldu.

Onu da unutmayaq ki,alt siniflər (nümunədə EmployeeManager) yalnız bir sinifdən miras ala bilər.

Abstraction (Mücərrədlik)
Abstract siniflər üst sinif (Base class) olaraq istifadə edilən mücərrəd siniflərdir. Abstract siniflərin bəzi xüsusiyyətlərinə baxaq.

  • Üst sinifdə var olan metod və propertilərin alt siniflərə görə müxtəlif cür işlədiyi vəziyyətlərdə istifadə olunur.
  • Bu siniflərdə abstract açar sözü ilə işarələnən metodlar və propertilər onu miras alan alt siniflər tərəfindən mütləq implement olunmalıdır!
  • Abstract metodlar və propertilər abstract siniflərdə yazılmalıdır.
  • Abstract metodlar və propertilər private ola bilməz!
  • Abstract metodlar yazıldığı üst siniflərdə yalnız imza hissəsi ilə qeyd olunur. Onların tam hissəsi implement olunduqları alt siniflərdə yazılır.
  • Abstract siniflərdə abstract olmayan metodlar da yazıla bilər.
  • Abstract siniflərdən obyekt yaradıla bilməz!

Nümunə:

abstract class Car{
  public abstract void Start();
}

class Audi:Car{
 public override void Start()
 {
    throw new NotImplementedException();
 }
}

Baxmayaraq ki , abstract siniflərdən obyekt yaranmır amma onlar obyekt yaradarkən referans olaraq istifadə edilə bilir.

Car audi = new Audi();

Polymorphism (Çox yönümlülük)
Üst sinifin referans , onun alt siniflərinin isə obyekt olaraq yaradıldığı vəziyyətdir.

Məsələn:

  Animal a1 = new Bird();

bu nümunədə Animal sinifini üst sinif olaraq Bird sinifini isə onun alt sinifi olaraq düşünsək, yuxarıdakı kimi obyekt yaratmaq mümkündür. Amma bunun tərsi mümkün deyil, eyni ilə hər quşun heyvan olması,amma hər heyvanın quş olmaması kimi.

Bird a1 = new Animal(); //error

nümunələri artırmaq olar:

Animal a2 = new Tiger();
Animal a3 = new Cat();

Burada üst sinif olan Animal hər obyektə görə müxtəlif davranış göstərir yəni çox yönümlü olur.

6) C#-da hansı növ siniflər (class) mövcuddur?

C#-da aşağıdakı növdə siniflər (class) mövcuddur:

  • Abstract class
  • Static class
  • Sealed class
  • Partial class

(Hər biri haqqında ətraflı məlumat verilib)

7) Abstract classlar və interfacelər arasındakı fərqlər nələrdir?

Abstract class Interface
Constructoru olur Constructoru olmur
Static dəyərlər ala bilər Static dəyərlər ala bilməz
Abstract classlardakı elementlər bütün access modifierslərlə işlənə bilər (abstract metodlar private ola bilmaz) Access modifiersi yalnız public ola bilər
Bir sinif yalnız bir abstract classı miras ala bilər Bir sinif birdən çox interfeysi miras ala bilər
Bir çox sinif eyni tipdən və ortaq davranış göstərirsə abstract sinif base class olaraq istifadə edilər Bir çox sinif yalnız ortaq metodlar istifadə edirsə interfeys istifadə etmək lazımdır
Abstract classlar metod,properti,fields,consts və s. elementlər ala bilir Interfeyslər yalnız metodlarla işlənir
Abstract classdan miras alan alt siniflər yalnızca bu sinifdəki abstract açar sözlü metodları implement etmelidir Interfeysdən miras alan siniflər interfeysin bütün metodlarını implement etmelidir

8) Virtual metodlar nədir?

Bir metodun virtual açar sözü ilə işarələnməsi o metodun alt siniflərdə dəyişdirilib fərqli davranış göstərə biləcəyi mənasına gəlir.

 public class Car
 {
    public virtual void Start()
    {
        Console.WriteLine("Car started");
    }
 }

 public class Audi : Car
 {
    public override void Start()
    {
        Console.WriteLine("Audi started");
    }
 }

Yuxarıdakı nümunədə base classda olan Start metodu onun alt sinifində fərqli şəkildə davranış göstərdi, bunun üçün override açar sözündən istifadə etmək lazımdır. Onu da unutmayaq ki, metodu override etmək kimi bir məcburiyyətimiz yoxdur,yəni Start metodunu Audi sinifində dəyişmək kimi bir məcburiyyətimiz yox idi.

9) Metodun override edilməsi nədir?

Metodun override edilməsinə metodun əzilməsi də deyirlər, mənası da üst siniflərdə olan metodun alt siniflərdə dəyişdirilərək istifadə edilməsidir. Unutmayaq ki,yalnız virtual açar sözü ilə işarələnmiş metodları əzə (dəyişdirə) bilərik.

10) Metodun overload edilməsi nədir?

Bu sual metodun override edilməsi ilə qarışdırılmamalıdır. Override metodu əzmək idisə, overload metodun artıq yüklənməsidir. Yəni bir sinifdə var olan bir metodu həmin sinifdə eyni ad altında yenidən yarada bilərik amma necə? Bunun üçün bəzi şərtlər var təbii ki birəbir eyni imza ilə 2 eyni metodu yarada bilmərik.

class Math{
  public void Sum(int a,int b){
    //code
  }

  public void Sum(int a,int b){ //error 2 eyni imzaya sahib metod ola bilməz
    //code
  }
}

O zaman nə edəcəyik,təbii ki, imzalarını dəyişdirəcəyik. Məsələn, birinci metod 2 parametr alıbsa 2-ci Sum metodu 3 parametr ala bilər yaxud ilk metodun parametrləri int tipində olub 2-ci metodun parametrləri double ola bilər. Bu hallarda eyni ad ilə xəta almadan metodlarımızı yarada bilərik,buna da metodun overload edilməsi deyilir.

class Math{
  void Sum(int a, int b)
  {
  //code
  }
  void Sum(int a, int b, int c)
  {
  //code
  }
}

yaxud

class Math{
  void Sum(double a, double b)
  {
  //code
  }
  void Sum(int a, int b)
  {
  //code
  }
}

hətta geri dönüş tipini də dəyişərək etmək mümkündür

class Math{
  void Sum(double a, double b)
  {
  //code
  }
  int Sum(int a, int b)
  {
  //code
  }
}

11) Static metodlar nədir?

Static metodlar obyekt ilə deyil birbaşa sinifin adı ilə müraciət edilən metodlardır. Bu metodlar çağırılan zaman obyekt yaradılmadığından constructor işə düşməyəcək. Static metodlar static açar sözü ilə yaradılır və yalnız static classlarda mövcud ola bilir.

public static class Math{
  public static int Sum(int num1,int num2){
    return num1+num2;
  }
}

Çağırılan zaman aşağıdakı şəkildə sinifin adı ilə çağırılır.

Math.Sum(2,3);

12) Sealed classlar nədir?

Sealed açar sözü ilə tanımlanmış classlar miras alma xüsusiyyətini bloklayır. Yəni bu şəkildə tanımladığımız classlardan artıq miras ala bilmərik.

public sealed class Car{
  //methods & properties
}

public class Audi:Car{ //error sealed classlardan miras almaq olmaz
}

13) Access modifierlar nədir və hansılardır?

Access Modifier-lar kodda xarici müdaxilənin sərhədlərini müəyyən etmək üçün istifadə olunan əsas ifadələrdir. C#-da aşağıdakı access modiferlar mövcuddur:

  • Private — Bir dəyərin private olaraq tanımlanması , o dəyərin yalnız aid olduğu sinifdən əlçatılan olması mənasına gəlir.
  • Public — Bir dəyərin public olaraq tanımlanması , o dəyərin istənilən yerdən əlçatılan olması deməkdir. Public-də məhdudiyyət yoxdur.
  • Protected — Bir dəyərin protected olaraq tanımlanması , o dəyərin aid olduğu sinifdən və o sinifi miras alan alt siniflərdən əlçatılan olması deməkdir.
  • Internal — Bir dəyərin internal olaraq tanımlanması , o dəyərin yalnız eyni proyektdən əlçatılan olması deməkdir.
  • Protected Internal — Bir dəyərin protected internal olaraq tanımlanması , o dəyərin aid olduğu sinif , onun alt sinifləri və hətta onun digər proyektdəki alt sinifindən əlçatılan olması deməkdir.

14) Value type və referance type nədir və hansılardır?

C#-da tiplər yaddaşda tutduğu yerə görə 2 yerə bölünür value tiplərreferance tiplər. Value typelar yaddaşda stack adlanan hissədə tutulur. Value typelar aşağıdakılardır:

  • int
  • double
  • decimal
  • float
  • byte
  • bool
  • char
  • struct
  • enum

Referans tiplər isə yaddaşın heap hissəsində saxlanılır, lakin onların stackda unikal adresləri olur,heapdakı referans tipə keçid məhz stack bölməsindəki unikal adresdən keçir. Referans typelar əsasən new açar sözü ilə yaradılan obyektlərdir.

  • Object
  • Class
  • Interface
  • string
  • array (hər tipdən olan arraylar bura daxildir)
  • List

ValueVsRef

Şəkildən gördüyümüz kimi referans tiplərin adresləri stackda saxlanılır

int num1;
int num2 = 12;

num1 = num2; // num1-in yeni dəyəri 12 olacaq
num2 = 19;  // num2-nin yeni dəyəri 19 olacaq amma num1-in dəyəri dəyişməyəcək,çünki o bir value typedır!
string[] cities = new string[]{"Bakı","Masallı","Sumqayıt"}; // array referans tipdir heapda saxlanılacaq,amma unikal adresi stackdadır

string[] cities2 = new string[]{"Gəncə","Lənkəran","Şamaxı"}; // başqa bir referans tipli obyekt

cities = cities2; // İki referansı tipi bir-birinə mənimsədən zaman onların stackdəki adreslərini eyni referansa yönəltmiş oluruq

valref2

cities = cities2 nəticəsi şəkildəki kimi olacaq , 2 adres də eyni referansa yönələcək. Nümunəyə davam edək..

Referans tipli dəyişkənlərin bir-birinə mənimsədilməsi ilə yaddaşdakı adreslərin kopyalanmasına Shallow Copy deyilir.

Obyektlərin kopyalanması əməliyyatına Deep Copy deyilir

cities[0] = "Naxçıvan";

Bu zaman hər iki adres eyni referansı göstərdiyindən cities[0]="Naxçıvan" yazdıqdan sonra eyni dəyər cities2 arrayında da dəyicək.

Console.WriteLine(cities2[0]); // Naxçıvan

Bəs cities arrayının heapdakı referansına ({"Bakı","Masallı","Sumqayıt"}) nə olacaq ? Bu zaman Garbage Collector dediyimiz obyekt heapda adresi olmayan bütün referansları siləcək.

15) Garbage collector nədir?

Obyekt yönümlü proqramlaşdırma dillərində mövcud olan Garbage Collectorun vəzifəsi heap bölməsində işini tamamlamış obyektlərin referanslarını təmizləməkdir. Bir sinifdən obyekt yaratdığımız zaman heap bölməsində onun üçün yer ayırılır,bu obyektin işi bitdikdən sonra Garbage Collector avtomatik olaraq həmin obyektin referansını heapdan təmizləyir.

Garbage Collector yalnız heap bölməsində fəaliyyət göstərir!

Garbage collectoru bizim işə salmağımıza ehtiyac yoxdur,o avtomatik olaraq işə düşür amma özümüz Garbage collectoru çağırmaq istəsək bu qayda ilə çağıra bilərik.

GC.Collect();

16) Constructor metod nədir?

Bir classdan obyekt yaradılarkən ilk işə düşən metodlardır. Constuctor public olmalı, class adı ilə eyni olmalıdır və constuctor metodların geri dönüş tipləri yoxdur.

MyClass myObj = new MyClass();  

new-dən sonra MyClass() referansını çağırdıq və bildiyimiz kimi () mötərizə C# və Java kimi obyekt yönümlü dillərdə yalnız metodlarda istifadə olunur. Myclass-dan sonra gələn mötərizələr bəs hansı metodu aktivləşdirir? Məhz constructor metodları.. Beləliklə, constuctor metodlar new açar sözü ilə obyket yaradılarkən ilk işə düşən metodlardır. Obyekt yaradılarkən constructor metodlar mütləq işə düşür !

class MyClass{
  public MyClass(){
    // obyekt yaradılarken ilk bura çalışır
  }
}

Hər classda biz yazmasaq da, default olaraq boş bir constuctor mövcuddur. Bir classda istənilən qədər constructor yaradıla bilər,parametr ala bilər və overload edilə bilər. Constructor metod private olarsa, həmin sinifdən obyekt yaratmaq mümkün olmayacaq,çünki obyekt yaradılarkən mütləq constuctor metod işə düşməlidir,amma access modifiersi private olarsa , constructor metoda kənardan əlçatmaq mümkün olmayacağından ötrü obyekt yarada bilməyəcəyik.

17) Destructor metod nədir?

Constructor metodun əksidir,bir classdan yaradılan obyekt yaddaşdan təmizlənərkən yaddaşdan silinmədən son dəfə çalışan metoddur.

C# dilində destructor sadəcə classlarda mövcuddur və hər bir classın sadəcə 1 destructoru olur! Destructor metod parametr ala bilməz buna görə də overload da edilə bilməz! Bir obyekt hansı şərtlərdə yox edilir (yaddaşdan silinir)?

  • Əgər obyekt hər hansı bir referans tərəfindən işarələnmirsə.. (həmin obyektin stackdakı adresi artıq o referansı göstərmirsə)
  • Obyektin yaradıldığı və istifadə edildiyi scope bitibsə.. (bu hal obyekt kənar bir scopeda istifadə edilməyibsə keçərlidir)
  • Obyket bir daha əlçatılan deyilsə..

Bu hallarda obyekt Garbage Collector tərəfindən yaddaşdan silinir,silinmədən öncə son bir dəfə destructor metod işə düşür.

Destructor metodun tanımlanması:

class MyClass{

  ~MyClass(){
    // obyekt imha edilərkən son dəfə çalışacaq kodlar  
  }
}

C# classlarında biz yaratmadığımız təqdirdə default olaraq destructor mövcud olmur!

18) C#-da Dispose və Finalize metodları arasındakı fərq nədir?

Hər iki metod yaddaşda istifadə edilməyən obyekti sərbəst buraxmaq üçün istifadə olunur. Yəni artıq istifadə olunmayan obyektləri silir.

Amma aralarında bəzi fərqlər mövcuddur:

  • Hər ikisi yaddaşdakı obyekti təmizləmək üçün istifadə edilir ancaq:
    • Dispose metodu yaddaşdakı obyekti təmizləmək üçün manuel olaraq istifadə edilir.
    • Finalize metodu isə yaddaşdakı obyekti təmizləmək üçün Garbage Collector tərəfindən istifadə edilir.
  • Dispose metodu manuel olaraq istifadə edildiyindən Garbage Collectora görə daha performanslı çalışır.
  • Digər tərəfdən Finalize metodu Garbage Collector tərəfindən çağırılırdığından performans baxımından yavaşdır.
  • Garbage collectorun nə zaman çalışacağı naməlumdur, bu da təhlükəsizlik baxımından risklidir.

Bu səbəblərdən C# -da Dispose metodunu istifadə etməyimiz tövsiyə edilir.

19) C#-da using açar sözü nə üçün istifadə olunur?

C#-da using açar sözü 2 məqamda istifadə olunur.

  • Başqa bir namespace-də yazılmış class və metodları import etmək üçün:
  • Dispose metodunu manuel yazmaq yerine avtomatik olaraq çağırmaq üçün:

Birinci nümunədə Models folderi altında olan Context sinifini cari sinifində istifadə etmək istəyiriksə,həmin Context sinifini burada import edirik.

using Models;
Context con = new Context();

Ikinci nümunədə isə biz Dispose metodunun manuel olaraq çağırıldığını qeyd etmişdik..

MyClass myObj = new MyObj();
((IDisposable)myObj).Dispose();

Hər obyekt yaratdıqdan sonra Dispose metodunu bu cür manuel olaraq çağırmaq yerinə using açar sözündən istifadə edə bilərik

using (FileStream stream = File.OpenRead("e"))
{
  //kodlar
}

Daha qısa yazsaq..

using FileStream stream = File.OpenRead("e");

using açar sözündən istifadə edərək manuel olaraq Dispose sinifini çağırmağa ehtiyac yoxdur

using açar sözü ancaq və ancaq IDisposable interfeysindən implement alan siniflərlə istifadə edilə bilir!

20) Namespace nədir?

C# -da namespacelar classların sərhədləridir/adresləridir əgər həmin classı başqa sərhəd daxilində istifadə etsək,adresini using açar sözü vasitəsilə bəyan etməliyik.

using Models.Entities;

Product prod = new();

Bu nümunədə /Models/Entities adresindəki/namespaceindəki sinifləri import etdik və artıq cari sinifdə istifadə edə bilərik.

21) Managed və Unmanaged kod nədir?

Managed kod .Net framework içərisindəki CLR(Common Language Runtime) tərəfindən idarə olunan koddur. Managed code C# kimi .Net dəstəkli dillər tərəfindən yazılır ve Intermediate Language (IL) olaraq compile edilir,daha sonra CLR tərəfindən idarə olunur. CLR yaddaşın idarəsi, xəta idarəsi (exception handling) və digər sistem səviyyəli xidmətləri bu kodlarda (managed) üzərinə götürür.

Unmanaged code isə .Net frameworkdən və CLR nəzarətindən kənar kodlardır,hansı ki bu kodlar .Net dəstəkli dillərdən birində yazılmayıb,bu cür kodlar birbaşa əməliyyat sistemi tərəfindən compile edilir. Bu kodlar CLR-ın nezaretinden kənar olduğundan belə kodlarda xəta idarəsi,yaddaşın idarəsi və s. xidmətlər proqramçı tərəfindən manuel olaraq həyata keçirilməlidir.

22) C#-da class və struct arasındakı fərqlər nələrdir?

C#-da həm siniflər, həm də structlar fərdi data tiplərini müəyyən etmək üçün istifadə olunur, lakin onların bəzi əsas fərqləri var.

Class Struct
Class referance typedır və Heapda saxlanılır Structlar isə value typedır və Stackdə saxlanılır
Classlar inheritance və polymorphismi dəstəkləyir Structlar isə inheritance və polymorphismi dəstəkləmir
Classlar Abstract olaraq tanımlana bilərlər Structlar abstract ola bilməz
Classların bütün üzvləri default olaraq private olur Structlarda isə üzvlər default olaraq public olur
Classlar referance type olduğundan yaddaş idarəsində Garbage Collectordan istifadə edə bilir Structlar value type olduğundan Garbage Collectordan istifadə edə bilmir
Classlar böyük və mürəkkəb modellər üçün daha uyğundur Structlar isə kiçik həcmli modellər üçün uyğundur.

23) Enum nədir?

Proqram içində istifadə edilən sabitlərə mənalı adlar verilməsi məqsədilə bu sabitləri bir qrup altında toplaya bilərik. Bu cür qruplara enum (enumeration) deyilir. Enumlar value typedır ,buna görə də yaddaşın stack bölməsində saxlanılır. Enumun istifadəsinə aid bir nümunə aşağıdakı kimidir:

public enum DaysOfWeek
{
    Monday,
    Tuesday,
    Wednesday,
    Thursday,
    Friday,
    Saturday,
    Sunday
}

Enumlarda ilk deyer default olaraq 0-dan başlayır, amma bunu dəyişmək mümkündür.

public enum DaysOfWeek
{
    Monday = 1,
    Tuesday,
    Wednesday,
    Thursday,
    Friday,
    Saturday,
    Sunday
}

Çağırarkən isə bu cür çağırılacaq:

DaysOfWeek day = DaysOfWeek.Monday;

24) C# -da boxing və unboxing nədir?

Boxing - bir value tipin referans tipə çevirilməsi, unboxing isə referans tipin value tipə çevirilməsidir.

Boxing

  • Value tip (int,float,char və s.) dəyişənin referans tipə (object) çevirilməsidir.
int num = 10;
Object obj = num; // boxing

Üstdəki kod nəticəsində num dəyişəni value tip olduğundan stackda saxlanılacaq, obj dəyişəni isə referans tip olduğundan heap bölməsində tutulacaq.

Unboxing

  • Referans tip dəyişənin value tipə çevirilməsidir.
Object obj = 41;
int i = (int)obj; // unboxing zamanı xeta ilə qarşılaşsaq casting istifade edeceyik

25) C# -da partial classlar hansılardır?

Siniflərin həddindən artıq böyüdüyü hallarda həmin sinifi bölüb idarə etməkdir. Partial classlar bizə bir sinifi birdən çox sinifə bölməyimizə, constructor , metod, dəyişən , properti və s. səliqəli şəkildə ayırmağımıza imkan verir.Fiziki olaraq birdən çox parçaya bölünmüş siniflər çalışma zamanında tək bir sinif kimi hərəkət edir.

public partial class Car
{
    public Car()
    {
       //codes
    }
}

public partial class Car
{
    public string Name { get; set; }
    public string Color { get; set; }
}

public partial class Car
{
    void Start()
    {
        //codes
    }
}

Yuxarıdakı nümunədə Car sinifini 3 yerə bölüb constructor,properti və metodlara görə ayırdıq. Buna baxmayaraq Car sinifi vahid bir sinif kimi davranır. Fərqli partial classlarda eyni adlı metod yaxud dəyişən yarada bilmərik eynilə normal siniflərdə olduğu kimi. Unutmayaq partial classlar normal classlar kimi davranır,normal classlarda edilən hər əməliyyat burada da edilə bilər.

26) Ref və out açar sözləri arasındakı fərq nədir?

Ref açar sözü referans sözünün qısaltmasıdır,parametrlərin referansını metoda vermək üçün istifadə edilir.Bir metoda parametr ötürdüyümüz zaman yalnız dəyəri kopyalanar, ötürdüyümüz parametrin referansını kopyalamaq üçün ref açar sözündən istifadə edə bilərik.

Əvvəlcə ref istifadə etmədən kodu aşağıdakı şəkildə yazaq:

int a = 41;
Console.WriteLine(a); // 41

ChangeNumber(a);
Console.WriteLine(a); // 41

void ChangeNumber(int num){
    num = 100;
}

Metod vasitəsilə dəyişənin qiymətini dəyişdirmək istəsək bu zaman ref açar sözündən istifadə edə bilərik. Biz a dəyişəninə reference-nı(adressini) metoda ötür demiş oluruq.

int a = 41;
Console.WriteLine(a); // 41

ChangeNumber(ref a);

Console.WriteLine(a); // 100

void ChangeNumber(ref int num){
    num = 100;
}

Out açar sözü də eyni işi görür, yəni parametrin referansını metodlara verməkdə istifadə olunur. outref -in təməl fərqi odur ki, ref açar sözünü istifadə ediriksə,parametrin başlanğıc dəyəri verilməlidir,out-da isə belə bir məcburiyyət yoxdur.

int a;

ChangeNumber(out a);

Console.WriteLine(a); // 100

void ChangeNumber(out int num)
{
    num = 100;
}

27) Const və readonly arasındakı fərq nədir?

Bütün proqram boyunca dəyərin sabit qalmasını istədiyimiz dəyişənləri tanımlayan zaman const və ya readonly açar sözlərindən istifadə edə bilərik. Hər ikisinin gördüyü iş eynidir,yəni sabit parametr yaratmağımıza kömək edir,amma aralarında bəzi fərqlər mövcuddur.

  • Const

    • Tanımlandığı anda dəyəri verilməlidir
    • Const ilə tanımlanan dəyişənlər statikdir və onlara sinifin adı ilə müraciət olunur
    • Classdan yaradılmış obyekt vasitəsilə müraciət edilə bilməz
  • Readonly

    • Dəyərini tanımladıqdan sonra metod yaxud constructorda verə bilərik
    • Classdan yaradılan obyekt ilə müraciət edilə bilir
public class Example
{
    public const float PI = 3.14f; // mütləq dəyəri verilməlidir
    public readonly string variable; // dəyəri constructorda verildi

    public Example(string variable)
    {
        this.variable = variable;
    }
    public Example(){}
}

Bu dəyişənlərə müraciət edən zaman :

Example.PI // const dəyər sinifin adı ilə çağırılır
Example obj = new();
obj.variable; // readonly dəyər obyekt ilə çağırılır

Hər iki dəyişən də sabit olduğundan onların dəyərini dəyişə bilmərik!

28) Birdən çox catch bloku çalışa bilərmi?

Try bloku ilə birlikdə birdən çox catch bloku tanımlana bilər ,amma sadəcə exceptionu tutan ilk catch bloku çalışacaq. Əgər proqram exception atarsa, bu exception sıra ilə catch bloklarında yoxlanılacaq və yalnız uyğun catch bloku işə düşəcək, əgər heç bir catch bloku uyğun deyilsə, heç biri işə düşməyəcək.
Əgər uyğun bir exception üçün birdən çox catch bloku yazılarsa, bu zaman runtime error alacağıq.

try
{
    int a = 4;
    Console.WriteLine(a / 0);
}
catch (DivideByZeroException ex)
{
    Console.WriteLine(ex.Message);
}
catch (DivideByZeroException ex) // error eyni exception üçün birdən çox catch bloku yazıla bilməz 
{
    Console.WriteLine(ex.Message);
}

29) Jagged arraylar hansılardır?

Jagged arraylar içərisində hər biri müxtəlif ölçülü array ola bilən elementlər saxlayan arraylardır(massivlərdir). Başqa bir sözlə jagged arraydakı hər array müxtəlif ölçüdə ola bilər.

int[][] jaggedArray = new int[3][];
jaggedArray[0] = new int[5];
jaggedArray[1] = new int[3];
jaggedArray[2] = new int[2];

Yuxarıdakı nümunədə tipi int olan jagged arrayın içərisində 3 array verilmiş və bu 3 arrayın hər biri müxtəlif ölçüdə qeyd edilmişdir. Adətən matrikslərdə və hər sətirin fərqli sayda sütunları olan senaryolarda istifadə edilə bilər.

30) Constructor zənciri nədir?

Constructor zənciri bir sinifin birdən çox constuctorundan birini çağıraraq bir constructordan digərinə keçmə əməliyyatıdır.

C# -da bir constructordan digərinə keçmək üçün base() yaxud this() açar sözlərindən istifadə edilə bilər.

  • this() açar sözü bir class daxilindəki constructor metodlarda birindən digərinə parametr göndərmək üçün istifadə olunur
public class Person
{
    public string FirstName { get; set; }
    public string LastName { get; set; }

    public Person(string lastName)
    {
       this.LastName = lastName;
    }

    public Person(string firstName, string lastName) :this(lastName) // lastName diger constructora göndərilib orada dəyər alacaq
    {
        this.FirstName = firstName;
    }
}
  • base() açar sözü isə adından da anlaşıldığı kimi base classın (üst sinifin) constructoruna parametr göndərmək üçün istifadə olunur.
class Employee : Person
{
   public Employee(string firstName, string lastName) : base(firstName, lastName) // üst sinifə ötürülüb orada dəyər alacaq
   {
   }
}

31) C# -da Genericlər nədir?

Genericlər classlarda,metodlarda və ya delegatelərdə bəlli bir tipə bağlı qalmadan kod yazmağı təmin edən xüsusiyyətlərdir. Genericlər birdən çox data tipi ilə çalışan ortaq kodun yazılmasını təmin edir.Eyni kodun başqa tiplərlə (string,int..) işləməsini təmin edir. Genericlərin istifadəsində classın yaxud metodun tipi bəlli deyil,runtime sırasında məlum edilir.

public class ExampleGeneric<T>
{
    public void GetValue(T entity)
    {
        Console.WriteLine(typeof(T));
        Console.WriteLine(entity);
    }
}

Yuxarıdakı kodda T bizim məhcul tipimizdir, çağırılan zaman T -nin yerinə məlum tipi istifadə ediləcək

ExampleGeneric<string> example = new(); // artıq tipimizin string olduğu məlumdur 
example.GetValue("Hello");

Çağırılan zaman tip məlum olur və artıq Generic classda T olan yerlər seçdiyim tiplə əvəz olunur,nümunədə T -lər string ilə əvəz olundu. Aşağıdakı nümunədə isə generic metod istifadə edilmişdir

List<T> Calculate<T>(T[] entities)
{
    List<T> result = new();
    foreach (var item in entities)
    {
        result.Add(item);
    }
    return result;
}

32) C# -da is və as operatorları arasındakı fərq nədir?

Is və as operatorları C# dilində obyektin tipinin yoxlanılması üçün istifadə edilir.

  • Is operatoru bir obyektin hər hansı bir tipə aid olub olmadığını yoxlayır və geriyə true yaxud false qaytarır.
object obj = "Hello";
bool result = obj is string;
Console.WriteLine(result);  // true
  • As operatoru isə bir obyektin hər hansı bir tipə çevirilib çevirilməyəcəyini yoxlayır.Əgər çevirilə bilirsə, obyektin həmin tipdəki dəyərini,çevirilə bilməzsə, null qaytarır.
object obj = "Hello";
string str = obj as string;
Console.WriteLine(str);  // Hello

33) C# -da tuple nədir?

Tuple birdən çox tipdəki elementlərin bir-biriylə əlaqəli olaraq tutulduğu bir data tipidir. Tuplelar bir dəyişən adı altında birdən çox datanı toplamaq üçün istifadə edilirlər və bu datalara müstəqil olaraq çatmaq mümkündür.

(int, string) tuple = (1, "Hello");

Bu datalara dəyişən adından sonra nöqtə qoyub Item1 , Item2 .. şəklində çata bilərik.

(int, string) tuple = (1, "Hello");
Console.WriteLine(tuple.Item1); // 1
Console.WriteLine(tuple.Item2); // "Hello"

Ayrıca çatdığımız dataları başqa bir dəyişənə ata bilərik

int firstElement = tuple.Item1;
string secondElement = tuple.Item2;

Tuplelar dataların bir-biriylə əlaqəli olaraq saxlanılmasını və dataların ayrı bir element olaraq çağırılmasını asanlaşdırırlar.

34) Array və ArrayList arasındakı fərqlər nələrdir?

Array və ArrayList dataların toplanması üçün istifadə edilən tiplərdir.Hər ikisi birdən çox dataların toplanmasında istifadə edilsə də,fərqləri çoxdur.

  • Arraylar
    • Əvvəlcədən müəyyən edilmiş tipdə,müəyyən edilmiş sayda məlumat toplayır
    • Array yaratmaq üçün mütləq arrayın saxlayacağı datanın tipi və saxlanılacaq məlumatların sayı əvvəlcədən müəyyən edilməlidir
    • Arraya data əlavə etmək yaxud çıxarmaq olmaz
    • Null dəyərlər ala bilməz
int[] numbers = new int[3];
numbers[0] = 1;
numbers[1] = 2;
numbers[2] = 3;
  • ArrayList
    • ArrayList üçün isə əvvəlcədən tip müəyyən edilmir və ArrayListlərin həcmi dinamikdir,yəni məlumat əlavə etdikcə yaxud sildikcə dəyişir
    • ArrayListə məlumat əlavə etmək yaxud çıxarmaq mümkündür
    • Null dəyərlər ala bilər
ArrayList numbers = new ArrayList();
numbers.Add(1);
numbers.Add(2);
numbers.Add(3);

ArrayListin performansı Arraya görə daha yavaşdır,həmçinin ArrayList məlumatları object tipində qəbul etdiyindən müxtəlif tipdən məlumatları saxlaya bilər. Digər tərəfdən arraylar seçilən tipdə məlumatları toplayır,hansının istifadə ediləcəyi ehtiyaca görə dəyişir.

35) Break və continue açar sözləri arasındakı fərqlər nələrdir?

Break və continue açar sözləri dövrlərdə (for,while) istifadə edilirlər.

  • Break açar sözü dövrü sonlandırır və dövrdən tamamilə çıxır.
for (int i = 0; i < 10; i++)
{
    if (i == 3)
        break; // 3-cü iterasiyada dövr tamamilə sonlanır
    Console.WriteLine(i); // 0 1 2
}
  • Continue açar sözü isə dövrün cari iterasiyasını atlayır və növbəti iterasiyaya keçir.
for (int i = 0; i < 10; i++)
{
    if (i == 3)
        continue; // 3-cü iterasiya keçilir və dövr davam edilir
    Console.WriteLine(i); // 0 1 2 4 5 6 7 8 9
}

36) C# -da error handling (xeta idarəsi) üçün istifadə edilən açar sözlər hansılardır?

Aşağıdakı açar sözlər C# -da error handling zamanı istifadə edilə bilər

  • try xəta ehtimalı olan kod bu blokda yazılır
  • catch try blokunda xəta baş verərsə burada yazdığımız kodlar çalışacaq
  • finally xəta olub olmamasından asılı olmayaraq çalışacaq kod blokudur
  • throw xəta yaratmaq üçün istifadə edilir (əsasən catch blokunda)

Xəta baş verməsi anında xətanın idarə edilməsində istifadə edəcəyimiz açar sözlər bunlardır, nümunəyə baxaq:

try
{
  int a = 2;
  int b = 0;
  int c = a / b;
  Console.WriteLine(c);
}
catch (Exception ex)
{
  throw new Exception(ex.Message);
}
finally
{
   Console.WriteLine("BB");
}

37) C# -da komment əlavə etmənin yolları hansılardır?

  • Single-Line Comment: Tək sətirlik komment üçün // simvolu istifadə edilir
// This is a single-line comment
  • Multi-Line Comment: Çox sətirlik komment üçün /* */ istifadə edilir
/*
This is
multi-line 
comment
*/
  • XML Documentation Comments: Xml kodlarındakı commentlər üçün /// istifadə olunur
/// <summary>
/// This is an XML documentation comment
/// </summary>

38) C# -da nullable tiplər necə istifadə edilir?

Bildiyimi kimi C# -da value typelar null dəyərlər ala bilmir, nullable tiplər value typelara null dəyərlər ötürə bilməyimizi təmin edir və aşağıdakı kimi yazılır.

int a = null; // error value type null ola bilmez
Nullable<int> b = null; // okay

Bu yazılışı daha da qısaldıb aşağıdakı kimi yaza bilerik

int? b = null;

Bu nullable dəyişənlərinin dəyərinin olub olmadığını yoxlamaq üçün HasValue propertisindən istifadə edə bilərik

if(b.HasValue) // b null deyilsə true əks halda false
    // codes

Dəyərini almaq üçün isə Value propertisindən istifadə ediə bilərik

if(b.HasValue)
    Console.WriteLine(b.Value);

Bunlarla yanaşı onu da bilməkdə fayda var ki, var tipi heç vaxt null ola bilməz!

39) C# -da String və StringBuilder arasındakı fərq nədir?

  • String mətn, text şəklində dəyərlər saxlamaq üçün istifadə etdiyimiz bir tipdir. Mətn tipli dəyərlərin içərisində axtarış, qarşılaşdırma və s. əməliyyatlar icrə edə bilərik.
  • Amma String sinifi immutable (dəyişdirilə bilməz) olduğundan cari string tipli dəyişənimizdə etdiyimiz dəyişikliklər üçün yeni bir String obyekti yaradılır, dəyişikliklərimiz orada qeyd edilir və əvvəlki stringin referansı qırılır.
// String sinifi
string myString = "Salam, Dünya!";
myString = myString.Replace("Dünya", "C#"); // heapda yeni bir obyekt yaradıldı
Console.WriteLine(myString);
  • StringBuilder sinifi də mətn tipli dəyərləri saxlamaq üçün istifadə edilir.
  • Təməl fərq odur ki, StringBuilder sinifi, String kimi immutable deyil, yəni cari dəyərin üzərində dəyişikliklər etdiyimiz zaman yeni bir obyekt yaradılmır.
// StringBuilder sinifi
StringBuilder myStringBuilder = new StringBuilder("Salam, Dünya!");
myStringBuilder.Replace("Dünya", "C#"); 
Console.WriteLine(myStringBuilder.ToString());
  • Bonus olaraq, StringBuffer sinifi haqqında da məlumat verək, demək olar ki, StringBuilder ilə eynidir.
  • StringBuffer-in təməl fərqi sinxron şəklində çalışması və 2 threadın eyni vaxtda eyni metoda daxil olmasına icazə verməməsidir.

40) Array.CopyTo() və Array.Clone() metodları arasındakı fərq nədir?

Hər iki metod С# -da arrayları kopyalamaq üçün istifadə edilir

  • Array.CopyTo() metodu əsas arraydakı elementləri seçdiyimiz arraya kopyalamaqda istifadə olunur
  • Bu metodla seçdiyimiz arrayın tipini və həcmini ötürməliyik, əgər əsas arrayın həcmi seçdiyimiz arrayın həcmindən böyük olsa, ArgumentException xətasını alacağıq.
int[] currentArray = new int[] { 1, 2, 3 };
int[] selectedArray = new int[3]; // həcmi və tipi əsas arraydakılarla eyni olmalıdır

// System.Array.CopyTo() 
currentArray.CopyTo(selectedArray, 0);
Console.WriteLine(string.Join(", ", selectedArray)); // 1, 2, 3

Əks halda

int[] currentArray = new int[] { 1, 2, 3 };
int[] selectedArray = new int[2]; // həcmi cari arraydakından aşağıdır

// System.Array.CopyTo() 
currentArray.CopyTo(selectedArray, 0); // ArgumentException
Console.WriteLine(string.Join(", ", selectedArray)); 
  • Array.Clone() metodu isə əsas arrayın bir kopyasın yaradır və geriyə kopyasını yaratdığı arrayı dönür.
  • Bu metod əsas arrayın bir kopyasını yaratdığından əsas arrayda ediləcək dəyişikliklərə məruz qalmayacaq.
int[] currentArray = new int[] { 1, 2, 3 };
int[] selectedArray = (int[])currentArray.Clone(); // həcmini ötürməyimizə ehtiyac yoxdur

Console.WriteLine(string.Join(", ", selectedArray)); // 1, 2, 3

41) C# -da extension metodlar nədir?

Extension metodlar class ya da structu dəyişdirmədən onlara metod əlavə edə bilməyimizi təmin edir. Bəzi classlar əlçatılmaz ola bilər, həmin classların içərisinə müdaxilə etmədən onlara aid metodlar yaza bilərik.

  • Extension metod yazmaq üçün bir neçə qayda var
    • Extension metodlar static classda static metod olaraq yazılmalıdır
    • Genişləndiriləcək class həmin extension metoda metodun ilk parametri olaraq verilib önünə this açar sözü ilə yazılmalıdır.
    • Extension metodda müəyyən edilmiş parametrlərdən yalnız 1-i this açar sözlə müəyyən edilə bilər.

Bir extension metod nümunəsi yazaq, fərz edək ki, belə bir classımız var və biz o classa müdaxilə etmədən ona aid metod yazmaq istəyirik

class Math{
    public int Sum(int num1,int num2){
      return num1 + num2;
    }
}

Indi extension metodu yazaq

public static class MathExtension{ // istediyimiz adı vere bilerik
    public static int Multiply(this Math math,int num1,int num2){ // ilk parametr genişledilecek class və önüne this açar sözü
      return num1 * num2;
    }
}

Artıq classdan obyekt yaradılan zaman hər iki metoda çata bilərik

Math math = new Math();
math.Sum(3,4);
math.Multiply(4,2);

Extension metodlar statik olaraq yazılmağına baxmayaraq obyekt adı ilə çağırılır

Extension metodlar daha çox C# -ın hazır metodlarına əlavə metod yazmaq istədiyimiz zaman source koda müdaxilə etmədən bunu etməyimizə imkan verir.

42) Async və await açar sözləri nə üçün istifadə edilir?

Bu açar sözlər yazdığımız kodların asinxron şəkildə çalışması üçün istifadə edilir. Asinxron proqramlaşdırmanı anlamaq üçün əvvəlcə sinxron proqramlaşdırmaya baxaq, sinxron proqramlaşdırmada yazdığımız kodlar yuxarıdan aşağıya doğru compile edilir və ona görə nəticə dönür. Yəni bir əməliyyat bitdikdən sonra digərinə keçilir.

Asinxron proqramlaşdırmada isə əməliyyatlar eyni anda başlayır.

sync

Bir metodu asinxron şəkildə yazmaq üçün asyncawait açar sözlərindən istifadə etməliyik.

public async Task<IEnumerable<Student>> GetAll(){
    return await _context.Students.ToListAsync();
}

Yuxarıdakı metodda databasadan məlumatlar gətiriləcək, amma məlumatların sayına görə bu əməliyyat bir neçə saniyə çəkə bilər asinxron proqramlaşdırma həmin bir neçə saniyə ərzində digər kodların çalışmasına, bir neçə saniyə bitib məlumatlar gələndə isə yenidən buranın çalışmasına imkan verir.

await açar sözü gözləniləcək yeri müəyyən edir, yəni koda 'Siz davam edin, datalar gelende men xeber edecem' deyir. await açar sözünü istifadə etmək üçün metodun imza hissəsində async açar sözünü istifadə edib geri dönüş tipini isə Task<> arasında yazmalıyıq

Bəs nə zaman kodları asinxron yazaq?

  • Əgər metodun uzaq bağlantısı varsa, (Database,Cloud ..)
  • Yaxud Http request göndərirsə, bu zaman metodu asinxron yazmağımız daha yaxşı olar.

Onu da bilmək də fayda var ki, asinxron proqramlaşdırma performansı artırmır, sadəcə saniyədə cavablandırılacaq request sayını artırır.

43) C# -da Indexer nədir?

Indexer bir classa array kimi indeks ötürməyimizi təmin edir və sinifin içərisindəki elementlərə çatmaq üçün array istifadə edilir.

Indexer bildiyimiz propertiyə çox bənzəyir eynən onun kimi get və set bloklarına sahibdir. Classdakı array yaxud listlərin elementlərini get və set blokları ilə istifadə etməyimizə yarayır.

class Indexer
{
    int[] numbers = new int[5];
    public int this[int index]
    {
        get
        {
            return numbers[index];
        }
        set
        {
            numbers[index] = value;
        }
    }
}

Indexer propertiyə çox bənzəyir tək fərq this[] sözüdür həmçinin indexerlara ad verilmir.

Çağırılarkən bu cür istifadə olunur

Indexer example = new Indexer();
example[0] = 3; // bu zaman Indexerin set blokları çalışır
example[3] = 4;
Console.WriteLine(example[0]); // indexerin get blokları burada işə düşür

example[0] = 3; dediyimiz zaman bu deyer indexerə gedir orada set bloklarına düşür və numbers[0] = 3; şəklində arraya ötürülür. Console.WriteLine(example[0]); bu zaman isə indexerin get blokları çalışır və return numbers[0]; şəklində nəticə qayıdır.

44) C# -da Linq nədir?

LINQ (Language Integrated Query), C# ve diger .NET dillərində dataları sorğulamak üçün istifadə edilən bir xüsusiyyətdir.

Linq databasedən, collectionlardan, xml fayllarından gələn dataları sorğulamağımızı asanlaşdırır.

linq

Linq sorğuları vasitəsilə gələn məlumatları filterdən keçirib, qruplayıb istifadə edə bilərik. Linq sorğuları sorğunun gerçəkləşdirildiyi yerə uyğun olaraq fərqli sintakslarda istifadə edə bilər, məsələn Sql-dən gələn datalar üçün sql-ə bənzər sintaksis istifadə edə bilirkən,xml-dən gələn datalar üçün ona uyğun bir sintaksisdə yazıla bilir.

List<int> nums = new() { 1, 2, 3, 4, 5, 6, 7, 8, 9 };

foreach (var item in nums.Where(x => x > 5))
{
    Console.WriteLine(item);
}

Bu nümunədə collectionda gələn məlumatların 5-dən böyük olanlarını ekrana yazdırdıq.

45) C# -da delegate nədir?

Delegatelər bir ya da daha çox metodu yaddaşda saxlayan referans tipli obyektlərdir. Delegate sözü bizim dilə təmsilçi olaraq çevirilə bilər yəni bir neçə metodu təmsil etdiyi adından məlumdur.

Delegatelər metodun imzası şəklində tanımlanır, tanımlanarkən delegate açar sözündən istifadə edilir. Və həmin delegate-ə yazdığımız metodları += operatoru ilə əlavə edə bilirik.

public delegate int OperationHandler(int x,int y); // Delegatelərin sonuna Handler əlavə etmək adətdəndir)

Indi bu delegate-ə əlavə etmək üçün metodlar yazaq.

public delegate int OperationHandler(int x, int y);
public class Operations
{
    public int Sum(int x, int y) => x + y;
    public int Substract(int num, int num2) => num - num2;
    public int Multiply(int a, int b) => a * b;
    public int DividedBy(int x, int y) => x / y;
}

Çağırılan zaman bu cür çağırılacaq

Operations obj = new();
OperationHandler operation = obj.Sum;
operation(4,2); // 6

// delegate-ə digər metodları da əlavə edək
operation += obj.Substract;
operation += obj.Multiply;
operation += obj.DividedBy;

operation(4,2); // 2 netice en son elave edilen metodun cavabını yazdıracaq

Delegate-ə metod əlavə etmək üçün +=, metod çıxarmaq üçün -=

Bütün metodların neticesini görmek üçün aşağıdakı kimi bir kod yaza bilerik

Operations obj = new Operations();
OperationHandler handler = obj.Sum;
handler += obj.Substract;
handler += obj.Multiply;
handler += obj.DividedBy;

List<int> results = new List<int>();
Delegate[] delegates = handler.GetInvocationList();

foreach (Delegate del in delegates)
{
    results.Add((int)del.DynamicInvoke(5, 2));
}

foreach (int result in results)
{
    Console.Write(result + " "); // 7 3 10 2
}

46) C# -da anonim tipler (anonymous type) nədir?

C# -da anonim tip (anonymous type) kodda açıq adı olmayan runtime zamanı tanımlanan bir tipdir. Xüsusi bir sinif yaratmadan sanki elə bir sinif varmış kimi, dəyər ötürə bildiyimiz bir tipdir.

 var Person = new { Name = "Ilkin", LastName = "Rufullayev" }; // new açar sözü ilə bu cür tanımlanır, içərisində istədiyimiz adda propertilər yaza bilerik
 Console.WriteLine(Person.Name); // Ilkin

Anonim tiplər sayəsində xüsusi bir hal üçün hər dəfə bir class yaratmalı deyilik. Əgər anonim tip istifadə etməsək üstdəki kod bu cür olacaqdı.

class Person{
  public string Name {get;set;}
  public string LastName {get;set;}
}

var value = new Person(){ Name = "Ilkin", LastName = "Rufullayev"};
Console.WriteLine(value.Name);

Əsasən Linq sorğularını sadələşdirmək və bunun üçün ayrıca sinif yaratmadan əməliyyatları icra etməkdə istifadə edilir.

47) Solid prinsipləri hansılardır?

Bu prinsiplər yazdığımız kodun anlaşıqlı , yeniliklərə açıq , minimum dərəcədə təkrarlı olmasını təmin edən prinsiplər toplusudur.

  • Bu prinsiplərin məqsədi

    • Yazdığımız kodun gələcəkdə yeni tələblərə tez adaptasiya olması
    • Kodda ciddi bir dəyişiklik etmədən yeni xüsusiyyətlərin əlavə edilə bilməsi
    • Kod təkrarçılığının qarşısının alına bilməsi
  • Solid prinsipləri aşağıdakılardır:

    • Single Responsibility Principle (SRP)- Tək məsuliyyət prinsipi
    • Open closed Principle (OCP)- Açıq qapalı prinsip
    • Liskov substitution Principle (LSP)- Liskovun əvəzetmə prinsipi
    • Interface segregation principle(ISP)- İnterfeys ayrılma prinsipi
    • Dependency inversion principle(DIP)- Asılılğın tərsinə çevrilmə prinsipi

Single Responsibility Principle — (Tək məsuliyyət prinsipi)

Bu prinsipin məqsədi bir sinifin ya da metodun bir işi görməsidir. Bir sinifdə ancaq bir məqsəd üçün kodlar yazıla bilər,məsələn CRUD əməliyyatlarını yazdığımız bir sinif olsun, bu sinifimizin məqsədi yalnız CRUD əməliyyatlarını icra etməsidir.Bu sinifdə biz başqa məqsədlər üçün, məsələn eyni zamanda modelə qoşulmaq üçün metod yazsaq, o zaman “Single responsibility”-nin prinsipini pozmuş olarıq

Open Closed Prinsiple — (Açıq qapalı prinsipi)

Bu prinsipin məqsədi sinifin dəyişikliklərə bağlı ,amma yeniliklərə açıq olmasıdır. Yəni proyektimiz yeniliklərə açıq olmalıdır, amma eyni zamanda təməl sinifimiz dəyişdirilməməlidir.

Liskov substitution Principle — (Liskovun əvəzetmə prinsipi)

Bütün classlarda eyni olacaq əsas metodların base classa yazılıb, vəziyyətə görə dəyişəcək metodların isə interfacelərə yazılaraq ehtiyac duyulduqda çağırılmasıdır. Lazımsız implementlərin qarşısını almaq üçün istifadə olunur.

Interface segregation principle — (İnterfeys ayrılma prinsipi)

Bir interfeysə yalnız bir işi görən metodların yığılaraq qarışıqlığın qarşısını alınmasıdır. Tək məsuliyyət prinsipinə bənzəyən bu prinsipdə bir işi görən metodlar bir interfeysə yazılır, beləcə implement zamanı lazımsız metodlar çağırılmamış olur.

Dependency inversion principle — (Asılılğın tərsinə çevrilmə prinsipi)

Bu prinsipə görə üst səviyyə siniflər , metodlar və modullar alt səviyyəli siniflərdən asılı olmamalıdır.Alt siniflərdə edilən hər hansı bir dəyişiklik üst siniflərə təsir etməməlidir.

48) Json və Xml arasındakı fərqlər nələrdir?

Fərqli kompüter sistemləri və məlumat bazaları müxtəlif məlumat formatlarına malikdir. Bu məlumatlar bir-biri ilə uyğunsuz ola bilər. JSON və XML internetdəki müxtəlif sistemlər arasında məlumat mübadiləsi üçün iki ümumi formatdır.

Json (Javascript Object Notation) JSON strukturlaşdırılmış məlumatları saxlayan bir formatdır və ümumiyyətlə server və müştəri arasında məlumat mübadiləsi üçün istifadə olunur. JSON-un bəzi məlumat növləri Number, Boolean, String, Array, Obyekt, Nulldur. JSON sintaksisi açar və dəyər (Key - Value) cütlərindən ibarətdir və məlumatlar vergüllə ayrılır. Aşağıdakı bir JSON nümunəsidir.

{
  "Id": "S001",
  "Ad": "Ann"
}

Json istifadə etmənin üstünlükləri

  • Bütün brauzerlər tərəfindən dəstəklənir
  • Oxumaq və yazmaq asandır
  • Şəbəkə bağlantısından istifadə edərək strukturlaşdırılmış məlumatları ötürməyə imkan verir
  • Müasir proqramlaşdırma dilləri ilə istifadə edilə bilər

XML (Extensible Markup Language) html-ə çox bənzəyən etiketli bir sintaksisə sahibdir, amma html-i əvəz etmir və onun kimi veb səhifənin quruluşu üçün deyil.XML daha ümumi məqsədlidir. XML-in vacib üstünlüyü proqramçıların özlərinə etiketlər yarada bilməsidir. XML-in əsas üstünlüyü isə ondan ibarətdir ki, iki fərqli sistem arasında körpü kimi istifadə edilə bilər. Məsələn, bir bankda daha köhnə bir kompüter sistemi ola bilər. XML onu yeni bir sistemlə bağlamaq və məlumat mübadiləsi üçün istifadə edilə bilər. Bu iki sistem tamamilə fərqli olsa da, məlumat mübadiləsi mümkündür. XML böyük məlumat toplusu üçün uyğun deyil. Bu vəziyyətdə bir verilənlər bazası istifadə edilməlidir. Məlumat mürəkkəbləşdikdə XML oxumaq çətin ola bilər. Aşağıdakı bir Xml nümunəsidir

<food>     
  <name>Belgian Waffles</name>     
  <price>$5.95</price>     
  <description>
  Two of our famous Belgian Waffles with plenty of real maple syrup 
  </description>     
  <calories>650</calories> 
</food>

Xml istifadə etmənin üstünlükləri

  • XML-in köməyi ilə müxtəlif platformalar arasında sürətli məlumat mübadiləsi apara bilərik
  • XML platformanın dəyişdirilməsi prosesini asanlaşdırır

Xml və Json arasındakı oxşarlıqlar

  • JSON və XML ikisi də veb əlaqəli texnologiyalardır
  • Hər ikisi məlumatları təsvir etmək üçün istifadə edilə bilir
  • Hər ikisi məlumat mübadiləsi üçün istifadə edilə bilir
  • Hər ikisi platforma dəyişikliklərini dəstəkləyir
  • Hər ikisi də bir çox proqramlaşdırma dilləri tərəfindən istifadə olunur

Xml və Json arasındakı fərqlər

  • JSON XML-dən daha sürətli və kiçik həcmlidir
  • JSON XML-dən daha sadə və oxunması asandır, Xml isə daha geniş və mürəkkəbdir
  • JSON Javascript-dən yarandığı halda, XML işarələmə dillərindən yaranmışdır
  • Ümumiyyətlə, JSON XML-dən üstün tutulur
  • RSS səhifələri, Sitemap faylları və bəzi API proqramları XML-in üstünlüklə istifadə edildiyi sahələrdir

49) Lazy Loading və Eager Loading nədir?

Entity framework-ün yazılan Linq sorğularına bağlı olaraq məlumatı database-dən gətirmək üçün loading tipləri vardır. Bunlar Lazy Loading və Eager Loading adlanır.

Lazy Loading

Səhifədə yer alan bir obyektin ehtiyac duyulmadığı təqdirdə çağırılmaması prinsipi ilə çalışır. Yəni bir obyekt nümunəsinin ehtiyac duyulana qədər alınmaması və gözlədilməsi məqsədi daşıyır. Bu məqsədlə datalar sorğuya bağlı olaraq gətirilir və bütün dataların gətirilməsi yerinə ehtiyac duyulduqca təkrar-təkrar database-ə sorğu göndərilir.

Eager Loading

Lazy loadingin tam əksi istiqamətdə çalışır. İstifadə edəcəyimiz obyektləri, ehtiyac anından da əvvəl yaradır və gözlədir. Eager loading Linq sorğusu işə düşəndə bütün dataları yükləyib yaddaşda saxlayır.(default loading tipidir)

İki üsulun da üstünlükləri və mənfi cəhətləri var. Bunlar yazdığımız proyektin ehtiyaclarına görə, database-in böyüklüyünə görə və database-dəki əlaqələrin çoxluğuna görə dəyişir. Lazy loading ilə entitylər ehtiyac olduqca gətirilir.Eager loadingə görə database-ə daha çox bağlanır və istək göndərir burada da proqram üçün əlavə yüklənmə vardır. Eager loading isə tək sorğuda məlumatları alır. Lazy loading və Eager loading arasındakı işləmə sürəti database-dəki dataların sayından asılıdır. Dataların sayı artdıqca Lazy loading database-ə təkrar-təkrar bağlandığı üçün sürəti azalır.

Performans Lazy Loading Eager Loading
100 data üçün orta hesabla sürət 1sn 1sn
1000 data üçün orta hesabla sürət 2.5sn 2sn
5000 data üçün orta hesabla sürət 7.5sn 7sn
10000 data üçün orta hesabla sürət 9.5sn 6.3sn

50) Dependency inversion və dependency injection arasındakı fərq nədir?

Dependency inversion Solid prinsiplerinin sonuncusudur, üst səviyyəli modulların alt səviyyəli modullardan asılı olmamasını deyir. (necə ki,rəhbər işçidən yox işçi rəhbərdən asılıdır)

Dependency inversion sadəcə bir prinsipdir, bu prinsipi isə dependency injection vasitəsi ilə həyata keçirəcəyik.

Dependency injection - Asılılıq yaradan hissələrin ayrılması və kənardan verilməsi ilə sistemdəki asılılığın minimuma endirilməsi prosesidir.

  • Bu 3 yolla edilə bilər
    • Constructor Injection
    • Property Injection
    • Method Injection

Dependency injection bir sinifin içərisində başqa sinifə aid obyekti istifadə edən zaman bu sinifi new açar sözü ilə yaratmamağımızı deyir

Bu mövzu çox mürəkkəb olduğundan burada qeyd etmek yerine bu mövzunun çox gözəl izah edildiyi bir menbeni paylaşmaq istəyirəm, aşağıdakı linkdən keçid edərək Dependency injection haqqında daha ətraflı məlumat sahibi ola bilersiniz. https://www.youtube.com/watch?v=Bhj2XdcoT2Q

51) C#-da "==" və "Equals()" arasındakı fərq nədir?

Həm qarşılaşdırma operatoru olan "==" ,həm də Equals() metodu 2 fərqli dəyəri müqayisə etmək üçün istifadə edilir. 2 value tipi müqayisə edərkən heç bir fərq gözə dəymir, əsas fərq referans typeların müqayisəsində ortaya çıxır. Referans tiplərini (reference type) qarşılaşdırarkən == operatoru 2 obyektin referansını müqayisə edərkən, Equals() metodu isə obyektlərin dəyərlərini müqayisə edir.

Nümunədə referans tiplərin həm adresləri, həm də dəyərləri bərabər olduğundan hər ikisi true dönəcək.

string message = "some string value";
string operationMessage = message; //referanslar beraberleşdirildi

Console.WriteLine(message == operationMessage); //true
Console.WriteLine(message.Equals(operationMessage)); //true

Digər bir nümunədə isə dəyərləri eyni olmasına baxmayaraq tiplərin referansları fərqlidir, bu zaman nəticə aşağıdakı kimi olacaq

object motorbike = "kawasaki";
char[] values = { 'k', 'a', 'w', 'a', 's', 'a', 'k', 'i' };
object myBike = new string(values); //kawasaki

Console.WriteLine(myBike == motorbike); //false çünki referanslar ferqlidir
Console.WriteLine(myBike.Equals(motorbike)); //true esas deyerler eynidir

52) IQueryable və IEnumerable arasındakı fərqlər nələrdir?

IQueryable dataları uzaq mənbədən (database,web service) gətirmək üçün istifadə olunur, dataları database tərəfdə filter və s.-dan keçirir və ram-a gətirir ammaaa execute etmir, sadəcə gətirir.

IEnumerable isə dataları gətirir işə salır sonra filter və s. icra edir.

Aşağıdakı entity framework ilə bir IQueryable nümunəsidir

_context.Products.Where(x=>x.IsDeleted == false); //deyerler ram-a getirildi,bu hisseye qeder olan kod IQueryable dönür

Bu deyerleri execute etmek üçün ToList() və bənzəri metod istifadə etməliyik, bu zaman dəyərimiz IQueryable-dan IEnumerable-a keçir.

_context.Products.Where(x=>x.IsDeleted == false).ToList(); //ram-dakı datalar execute edildi, bu kod IEnumerable dönür

53) Singleton design patterni açıqla.

Bu patterndə istifadəçiyə özbaşına obyekt yaratmağa icazə verilmir, yəni istifadəçi new ilə obyekt yarada bilməz. İstifadəçi bizdən obyekt tələb edə bilər, bu tələbə cavab olaraq biz də istifadəçiyə yaddaşda var olan obyekti verəcəyik. Beləcə istifadəçi yalnızca bir obyekt ilə işləməli olacaq.Bir iş üçün 50 fərqli obyekt yaratmaq yerinə sadəcə 1 obyekt ilə işi görməli olduğumuz senaryolarda istifadə edə bilərik

İndi keçək dediklərimizi koda salmağa, ilk başda istifadəçinin yeni obyekt yaratmasına imkan verməməliyik, bunun üçün constructor metodu private etmək kifayətdir.

    class Singleton
    {
        private Singleton(){}
    }    

İstifadəçi constructoru istifadə edərək obyekt yarada bilməyəcəyi üçün ona obyekti biz verməliyik,bunun üçün də həmin class tipində obyekt dönən bir metod yaxud properti yaza bilərik.

    class Singleton
    {
        //Private edərək bu classdan obyekt yaradılmasına imkan vermirik
        private Singleton(){}
        
        //Obyekt tələb olunarkən bu property-ni göndərəcəyik
        private static Singleton Instance;

        //Yuxarıdakı propertinin null olub olmadığını yoxladıqdan sonra göndəririk
        public static Singleton GetInstance() => Instance ?? (Instance = new Singleton());
     }   

Bu şəkildə Singleton patterni tətbiq edə bilərik,və artıq hər yeni obyekt tələb olunan zaman yaddaşdakı obyekti verəcəyik, yoxlamaq üçün..

Singleton obj = Singleton.GetInstance();
Singleton obj2 = Singleton.GetInstance();

if(obj == obj2)
    Console.WriteLine("Same");

If blokunun çalışması obyektlərin referansının eyni olduğunu göstərir

Yuxarıdakı kod Singleton patterninin metod ilə tətbiqidir,Singleton patternini properti ilə tətbiq etmək üçün aşağıdakı koddan istifadə edə bilərik

    class Singleton
    {
         //Private edərək bu classdan obyekt yaradılmasına imkan vermirik
         private Singleton(){}
         
         //Obyekt tələb olunarkən bu property-ni göndərəcəyik
         private static Singleton Instance;
         
         //null kontroluna ehtiyac yoxdur
         public static Singleton GetInstanceProp
         {
            get { return Instance; }
         }

         //Statik constructor sadəcə 1 dəfə obyekt yaradılarkən çalışır,ən birinci işə düşəndir
         static Singleton()
         {
            Instance = new Singleton();
         }
     }