Entity Framework Core and LINQ to Entities in Depth (2) Modeling Database: Object-Relational Mapping

[LINQ via C# series]

[Entity Framework Core (EF Core) series]

[Entity Framework (EF) series]

In LINQ to Entities, the queries are based on Object-relational mapping. .NET and SQL database and  have 2 different data type systems. For example, .NET has System.Int64 and System.String, while SQL database has bigint and nvarchar; .NET has sequences and objects, while SQL database has tables and rows;, etc. Object-relational mapping is a popular technology to map and convert between application data objects and database relational data.

Data types

EF Core can map most SQL data types to .NET types:

SQL type category

SQL type

.NET type

C# primitive

Exact numeric

bit

System.Boolean

bool


tinyint

System.Byte

byte


smallint

System.Int16

short


int

System.Int32

int


bigint

System.Int64

long


smallmoney, money, decimal, numeric

System.Decimal

decimal

Approximate numeric

real

System.Single

float


float

System.Double

double

Character string

char, varchar, text

System.String

string


nchar, nvarchar, ntext

System.String

string

Binary string

binary, varbinary

System.Byte[]

byte[]


image

System.Byte[]

byte[]


rowversion (timestamp)

System.Byte[]

byte[]

Date time

date

System.DateTime



time

System.TimeSpan



smalldatetime, datetime, datetime2

System.DateTime



datetimeoffset

System.DateTimeOffset


Other

hierarchyid

No built-in support



xml

System.String

string


uniqueidentifier

System.Guid



sql_variant

System.Object

object

The mapping of spatial types are supported through NetTopologySuite, a free and open source library. For SQL database, just install the Microsoft.EntityFrameworkCore.SqlServer.NetTopologySuite NuGet package.

Database

A SQL database is mapped to a type derived from Microsoft.EntityFrameworkCore.DbContext:

public partial class AdventureWorks : DbContext { }

The following is the definition of DbContext:

namespace Microsoft.EntityFrameworkCore

{

    public class DbContext : IDisposable, IInfrastructure<IServiceProvider>

    {

        public DbContext(DbContextOptions options);


        public virtual ChangeTracker ChangeTracker { get; }


        public virtual DatabaseFacade Database { get; }


        public virtual void Dispose();


        public virtual int SaveChanges();


        public virtual DbSet<TEntity> Set<TEntity>() where TEntity : class;


        protected internal virtual void OnModelCreating(ModelBuilder modelBuilder);


        // Other members.

    }

}

DbContext implements IDisposable. Generally, a database instance should be constructed and disposed for each unit of work - a collection of data operations that should succeed or fail as a unit:

internal static void Dispose()

{

    using (AdventureWorks adventureWorks = new AdventureWorks())

    {

        // Unit of work.

    }

}

EF Core also support DbContext pooling to improve performance. In the application or service, If DbContext is used through dependency injection, and it is no custom state (just like the above AdventureWorks type with no fields), then DbContext pooling can be enabled to reuse DbContext without disposing.

In EF Core, most of the object-relational mapping can be implemented declaratively, and the rest of the mapping can be implemented imperatively by overriding DbContext.OnModelCreating, which is automatically called by EF Core during initialization:

public partial class AdventureWorks

{

    protected override void OnModelCreating(ModelBuilder modelBuilder)

    {

        base.OnModelCreating(modelBuilder);

 

        MapCompositePrimaryKey(modelBuilder);

        MapManyToMany(modelBuilder);

        MapDiscriminator(modelBuilder);

    }

}

The above MapCompositePrimaryKey, MapManyToMany, MapDiscriminator functions are implemented later in this chapter.

Connection resiliency and execution retry strategy

The connection to the mapped database can be specified from the constructor of DbContext:

public partial class AdventureWorks

{

    public AdventureWorks(DbConnection connection = null)

        : base(GetDbContextOptions(connection))

    {

    }


    private static DbContextOptions GetDbContextOptions(

        DbConnection connection = null) =>

            new DbContextOptionsBuilder<AdventureWorks>()

                .UseSqlServer(

                    connection: connection ??

                        new SqlConnection(ConnectionStrings.AdventureWorks),

                    sqlServerOptionsAction: options => options.EnableRetryOnFailure(

                        maxRetryCount: 5, maxRetryDelay: TimeSpan.FromSeconds(30),

                        errorNumbersToAdd: null))

                .Options;

}

Here when database connection is not provided to the constructor, a new database connection is created with the previously defined connection string. Also, regarding the connection between application and SQL database may be interrupted over the network, EF Core supports connection resiliency for SQL database. This is very helpful for Azure SQL database deployed in the cloud instead of local network. In the above example, EF Core is specified to automatically retries up to 5 times with the retry interval of 30 seconds.

Tables

There are tens of tables in the AdventureWorks database, but don’t panic, this book only involves a few tables, and a few columns of these tables. In EF Core, a table definition can be mapped to an entity type definition, where each column is mapped to a entity property. For example, the AdventureWorks database has a Production.ProductCategory table. Its definition can be virtually viewed as:

CREATE SCHEMA [Production];

GO

 

CREATE TYPE [dbo].[Name] FROM nvarchar(50) NULL;

GO

 

CREATE TABLE [Production].[ProductCategory](

    [ProductCategoryID] int IDENTITY(1,1) NOT NULL

        CONSTRAINT [PK_ProductCategory_ProductCategoryID] PRIMARY KEY CLUSTERED,

 

    [Name] [dbo].[Name] NOT NULL, -- nvarchar(50).

 

    [rowguid] uniqueidentifier ROWGUIDCOL NOT NULL -- Ignored in mapping.

        CONSTRAINT [DF_ProductCategory_rowguid] DEFAULT (NEWID()),

   

    [ModifiedDate] datetime NOT NULL -- Ignored in mapping.

        CONSTRAINT [DF_ProductCategory_ModifiedDate] DEFAULT (GETDATE()));

GO

This table definition can be mapped to a ProductCategory entity definition:

public partial class AdventureWorks

{

    public const string Production = nameof(Production); // Production schema.

}

 

[Table(nameof(ProductCategory), Schema = AdventureWorks.Production)]

public partial class ProductCategory

{

    [Key]

    [DatabaseGenerated(DatabaseGeneratedOption.Identity)]

    public int ProductCategoryID { get; set; }

 

    [MaxLength(50)]

    [Required]

    public string Name { get; set; }

 

    // Other columns are ignored.

}

The [Table] attribute specifies the table name and schema. [Table] can be omitted when the table name is the same as the entity name, and the table is under the default dbo schema. In the table-entity mapping:

·        The ProductCategoryID column of int type is mapped to a System.Int32 property with the same name. The [Key] attribute indicates it is a primary key. EF Core requires a table to have primary key to be mapped. [DatabaseGenerated] indicates it is an identity column, with value generated by database.

·        The Name column is of dbo.Name type. which is actually nvarchar(50), so it is mapped to Name property of type System.String. The [MaxLength] attribute indicates the max length of the string value is 50. [Required] indicates it should not be null or empty string or whitespace string.

·        The other columns rowguid and ModifiedDate are not mapped. They are not used in this book to keep the code examples simple.

At runtime, each row of Production.ProductCategory table is mapped to a ProductCategory entity instance. The entire table can be mapped to an IQueryable<T> data source, exposed as a property of the database mapping. EF Core provides DbSet<T>, which implements IQueryable<T>, to represent a table data source:

public partial class AdventureWorks

{

    public DbSet<ProductCategory>ProductCategories { get; set; }

}

EF Core also supports immutable entity definition:

[Table(nameof(ProductCategory), Schema = AdventureWorks.Production)]

public partial class ProductCategory

{

    public ProductCategory(int productCategoryID, string name) =>

        (this.ProductCategoryID, this.Name) = (productCategoryID, name);


    [Key]

    [DatabaseGenerated(DatabaseGeneratedOption.Identity)]

    public int ProductCategoryID { get; private set; }


   [MaxLength(50)]

    [Required]

    public string Name { get; private set; }

}

This book defines all table mapping as mutable, since it is easier to update the entities and save back to database.

Relationships

In SQL database, tables can have foreign key relationships, including one-to-one, one-to-many, and many-to-many relationships.

One-to-one

The following Person.Person table and HumanResources.Employee table has a one-to-one relationship:

HumanResources.Employee table’s BusinessEntityID column is a foreign key that refers to Person.Person table’s primary key. Their definition can be virtually viewed as:

CREATE TABLE [Person].[Person](

    [BusinessEntityID] int NOT NULL

        CONSTRAINT [PK_Person_BusinessEntityID] PRIMARY KEY CLUSTERED,


    [FirstName] [dbo].[Name] NOT NULL,


    [LastName] [dbo].[Name] NOT NULL


    /* Other columns. */);

GO

 

CREATE TABLE [HumanResources].[Employee](

    [BusinessEntityID] int NOT NULL

        CONSTRAINT [PK_Employee_BusinessEntityID] PRIMARY KEY CLUSTERED

        CONSTRAINT [FK_Employee_Person_BusinessEntityID] FOREIGN KEY

        REFERENCES [Person].[Person] ([BusinessEntityID]),

   

    [JobTitle] nvarchar(50) NOT NULL,


    [HireDate] date NOT NULL


    /* Other columns. */);

GO

So each row in HumanResources.Employee table refers to one row in Person.Person table (an employee must be a person). On the other hand, each row in Person.Person table can be referred by 0 or 1 row in HumanResources.Employee table (a person can be an employee, or not). This relationship can be represented by navigation property of entity type:

public partial class AdventureWorks

{

    public const string Person = nameof(Person);

 

    public const string HumanResources = nameof(HumanResources);

 

    public DbSet<Person> People { get; set; }

 

    public DbSet<Employee> Employees { get; set; }

}

 

[Table(nameof(Person), Schema = AdventureWorks.Person)]

public partial class Person

{

    [Key]

    public int BusinessEntityID { get; set; }

 

    [Required]

    [MaxLength(50)]

    public string FirstName { get; set; }

 

    [Required]

    [MaxLength(50)]

    public string LastName { get; set; }

 

    public virtual Employee Employee { get; set; } // Reference navigation property.

}

 

[Table(nameof(Employee), Schema = AdventureWorks.HumanResources)]

public partial class Employee

{

    [Key]

    [ForeignKey(nameof(Person))]

    public int BusinessEntityID { get; set; }

       

    [Required]

    [MaxLength(50)]

    public string JobTitle { get; set; }

 

    public DateTime HireDate { get; set; }

 

    public virtual Person Person { get; set; } // Reference navigation property.

}

The [ForeignKey] attribute indicates Employee entity’s BusinessEntityID property is the foreign key for the relationship represented by navigation property. Here Person is called the primary entity, and Employee is called the dependent entity. Their navigation properties are called reference navigation properties, because each navigation property can refer to a single entity. The navigation property is designed to be virtual to enable proxy entity to implement lazy loading. Proxy entity and lazy loading is discussed in the query translation and data loading chapter.

One-to-many

The Production.ProductCategory and Production.ProductSubcategory tables have a one-to-many relationship, so are Production.ProductSubcategory and Production.Product:

Each row in Production.ProductCategory table can refer to many rows in Production.ProductSubcategory table (category can have many subcategories), and each row in Production.ProductSubcategory table can refer to many rows in Production.Product table (subcategory can have many products). Their definitions can be virtually viewed as:

CREATE TABLE [Production].[ProductSubcategory](

    [ProductSubcategoryID] int IDENTITY(1,1) NOT NULL

        CONSTRAINT [PK_ProductSubcategory_ProductSubcategoryID] PRIMARY KEY CLUSTERED,


    [Name] [dbo].[Name] NOT NULL, -- nvarchar(50).

 

    [ProductCategoryID] int NOT NULL

        CONSTRAINT [FK_ProductSubcategory_ProductCategory_ProductCategoryID] FOREIGN KEY

        REFERENCES [Production].[ProductCategory] ([ProductCategoryID]),


    /* Other columns. */)

GO

 

CREATE TABLE [Production].[Product](

    [ProductID] int IDENTITY(1,1) NOT NULL

        CONSTRAINT [PK_Product_ProductID] PRIMARY KEY CLUSTERED,


    [Name] [dbo].[Name] NOT NULL, -- nvarchar(50).

 

    [ListPrice] money NOT NULL,


    [ProductSubcategoryID] int NULL

        CONSTRAINT [FK_Product_ProductSubcategory_ProductSubcategoryID] FOREIGN KEY

        REFERENCES [Production].[ProductSubcategory] ([ProductSubcategoryID])

   

    /* Other columns. */)

GO

These one-to-many relationships can be represented by navigation property of type ICollection<T>:

public partial class ProductCategory

{

    public virtual ICollection<ProductSubcategory> ProductSubcategories { get; set; } // Collection navigation property.

}

 

[Table(nameof(ProductSubcategory), Schema = AdventureWorks.Production)]

public partial class ProductSubcategory

{

    [Key]

    [DatabaseGenerated(DatabaseGeneratedOption.Identity)]

    public int ProductSubcategoryID { get; set; }

 

    [MaxLength(50)]

    [Required]

    public string Name { get; set; }

 

    public int ProductCategoryID { get; set; }

 

    public virtual ProductCategory ProductCategory { get; set; } // Reference navigation property.

 

    public virtual ICollection<Product> Products { get; set; } // Collection navigation property.

}

 

[Table(nameof(Product), Schema = AdventureWorks.Production)]

public partial class Product

{

    [Key]

    [DatabaseGenerated(DatabaseGeneratedOption.Identity)]

    public int ProductID { get; set; }

 

    [MaxLength(50)]

    [Required]

    public string Name { get; set; }

 

    public decimal ListPrice { get; set; }

 

    public int? ProductSubcategoryID { get; set; }

 

    public virtual ProductSubcategory ProductSubcategory { get; set; } // Reference navigation property.

}

Notice Production.Product table’s ProductSubcategoryID column is nullable, so it is mapped to a int? property. Here [ForeignKey] attribute is omitted, because each dependent entity’ foreign key is separated from its primary key, so the foreign key can be automatically discovered by EF Core.

Many-to-many

Production.Product and Production.ProductPhoto tables has many-to-many relationship.

This is implemented by 2 one-to-many relationships with another Production.ProductProductPhoto junction table. These tables’ definitions can be virtually viewed as:

CREATE TABLE [Production].[ProductPhoto](

    [ProductPhotoID] int IDENTITY(1,1) NOT NULL

        CONSTRAINT [PK_ProductPhoto_ProductPhotoID] PRIMARY KEY CLUSTERED,


    [LargePhotoFileName] nvarchar(50) NULL,

   

    [ModifiedDate] datetime NOT NULL

        CONSTRAINT [DF_ProductPhoto_ModifiedDate] DEFAULT (GETDATE())


    /* Other columns. */)

GO

 

CREATE TABLE [Production].[ProductProductPhoto](

    [ProductID] int NOT NULL

        CONSTRAINT [FK_ProductProductPhoto_Product_ProductID] FOREIGN KEY

        REFERENCES [Production].[Product] ([ProductID]),


    [ProductPhotoID] int NOT NULL

        CONSTRAINT [FK_ProductProductPhoto_ProductPhoto_ProductPhotoID] FOREIGN KEY

        REFERENCES [Production].[ProductPhoto] ([ProductPhotoID]),


    CONSTRAINT [PK_ProductProductPhoto_ProductID_ProductPhotoID] PRIMARY KEY NONCLUSTERED ([ProductID], [ProductPhotoID])

   

    /* Other columns. */)

GO

So the many-to-many relationship can be mapped to 2 one-to-many relationships with the junction:

public partial class Product

{

    public virtual ICollection<ProductProductPhoto> ProductProductPhotos { get; set; } // Collection navigation property.

}

 

[Table(nameof(ProductPhoto), Schema = AdventureWorks.Production)]

public partial class ProductPhoto

{

    [Key]

    [DatabaseGenerated(DatabaseGeneratedOption.Identity)]

    public int ProductPhotoID { get; set; }

 

    [MaxLength(50)]

    public string LargePhotoFileName { get; set; }

 

    [ConcurrencyCheck]

    public DateTime ModifiedDate { get; set; }

 

    public virtual ICollection<ProductProductPhoto> ProductProductPhotos { get; set; } // Collection navigation property.

}

 

[Table(nameof(ProductProductPhoto), Schema = AdventureWorks.Production)]

public partial class ProductProductPhoto

{

    [Key]

    [Column(Order = 0)]

    public int ProductID { get; set; }

 

    [Key]

    [Column(Order = 1)]

    public int ProductPhotoID { get; set; }

 

    public virtual Product Product { get; set; } // Reference navigation property.

 

    public virtual ProductPhoto ProductPhoto { get; set; } // Reference navigation property.

}

ProductPhoto.ModifiedDate has a [ConcurrencyCheck] attribute for concurrency conflict check, which is discussed in the data manipulation chapter. Production.ProductProductPhoto table has a composite primary key. As a junction table, each row in the table has a unique combination of ProductID and ProductPhotoID. EF Core requires additional initialization for composite primary key, which should be executed in DbContext’s OnModelCreating method:

public partial class AdventureWorks

{

    private static void MapCompositePrimaryKey(ModelBuilder modelBuilder) // Called by OnModelCreating.

    {

        modelBuilder.Entity<ProductProductPhoto>()

            .HasKey(productProductPhoto => new

            {

                ProductID = productProductPhoto.ProductID,

                ProductPhotoID = productProductPhoto.ProductPhotoID

            });

    }

}

EF Core also requires additional initialization for many-to-many relationship represented by 2 one-to-many relationships, which should be executed in OnModelCreating as well:

public partial class AdventureWorks

{

    private static void MapManyToMany(ModelBuilder modelBuilder) // Called by OnModelCreating.

    {

        modelBuilder.Entity<ProductProductPhoto>()

            .HasOne(productProductPhoto => productProductPhoto.Product)

            .WithMany(product => product.ProductProductPhotos)

            .HasForeignKey(productProductPhoto => productProductPhoto.ProductID);

 

        modelBuilder.Entity<ProductProductPhoto>()

            .HasOne(productProductPhoto => productProductPhoto.ProductPhoto)

            .WithMany(photo => photo.ProductProductPhotos)

            .HasForeignKey(productProductPhoto => productProductPhoto.ProductPhotoID);

    }

}

Finally, all the above tables can be exposed as properties of AdventureWorks:

public partial class AdventureWorks

{

    public DbSet<Person> People { get; set; }

 

    public DbSet<Employee> Employees { get; set; }

 

    public DbSet<ProductSubcategory>ProductSubcategories { get; set; }

 

    public DbSet<Product> Products { get; set; }

 

    public DbSet<ProductPhoto> ProductPhotos { get; set; }

}

Inheritance

EF Core supports table per hierarchy (TPH) inheritance for entity types. With TPH, one table is mapped to many entity types in the inheritance hierarchy, so a discriminator column is needed to identify each specific row’s mapping entity. Take the Production.TransactionHistory table as example, its definition can be virtually viewed as:

CREATE TABLE [Production].[TransactionHistory](

    [TransactionID] int IDENTITY(100000,1) NOT NULL

        CONSTRAINT [PK_TransactionHistory_TransactionID] PRIMARY KEY CLUSTERED,


    [ProductID] int NOT NULL

        CONSTRAINT [FK_TransactionHistory_Product_ProductID] FOREIGN KEY

        REFERENCES [Production].[Product] ([ProductID]),


    [TransactionDate] datetime NOT NULL,


    [TransactionType] nchar(1) NOT NULL

        CONSTRAINT [CK_Product_Style]

        CHECK (UPPER([TransactionType]) = N'P' OR UPPER([TransactionType]) = N'S' OR UPPER([TransactionType]) = N'W'),


    [Quantity] int NOT NULL,


    [ActualCost] money NOT NULL


    /* Other columns. */);

GO

Its TransactionType column allows value “P”, “S”, or “W” to indicate each row representing a purchase transaction, sales transaction, or work transaction. So the mapping entities’ hierarchy can be:

[Table(nameof(TransactionHistory), Schema = AdventureWorks.Production)]

public abstract class TransactionHistory

{

    [Key]

    public int TransactionID { get; set; }


    public int ProductID { get; set; }


    public DateTime TransactionDate { get; set; }


    public int Quantity { get; set; }


    public decimal ActualCost { get; set; }

}


public class PurchaseTransactionHistory : TransactionHistory { }


public class SalesTransactionHistory : TransactionHistory { }


public class WorkTransactionHistory : TransactionHistory { }

Then the discriminator must be specified when OnModelCreating is executed:

public enum TransactionType { P, S, W }


public partial class AdventureWorks

{

    private static void MapDiscriminator(ModelBuilder modelBuilder) // Called by OnModelCreating.

    {

        modelBuilder.Entity<TransactionHistory>()

            .HasDiscriminator<string>(nameof(TransactionType))

            .HasValue<PurchaseTransactionHistory>(nameof(TransactionType.P))

            .HasValue<SalesTransactionHistory>(nameof(TransactionType.S))

            .HasValue<WorkTransactionHistory>(nameof(TransactionType.W));

    }

}

Now these entities can all be exposed as data sources:

public partial class AdventureWorks

{

    public DbSet<TransactionHistory>Transactions { get; set; }

 

    public DbSet<PurchaseTransactionHistory> PurchaseTransactions { get; set; }

 

    public DbSet<SalesTransactionHistory>SalesTransactions { get; set; }

 

    public DbSet<WorkTransactionHistory>WorkTransactions { get; set; }

}

Views

A view can also be mapped as if it is a table, if the view has one or more columns which can be virtually viewed as primary key. Take the Production.vEmployee view as example, its definition can be virtually viewed as:

CREATE VIEW [HumanResources].[vEmployee]

AS

SELECT

    e.[BusinessEntityID],

    p.[FirstName],

    p.[LastName],

    e.[JobTitle] 

    -- Other columns.

FROM [HumanResources].[Employee] e

    INNER JOIN [Person].[Person] p

    ON p.[BusinessEntityID] = e.[BusinessEntityID]

    /* Other tables. */;

GO

The BusinessEntityID is unique and can be the virtual primary key. So it can be mapped as the following entity:

[Table(nameof(vEmployee), Schema = AdventureWorks.HumanResources)]

public class vEmployee

{

    [Key]

    public int BusinessEntityID { get; set; }


    public string FirstName { get; set; }


    public string LastName { get; set; }


    public string JobTitle { get; set; }

}

And then expose as data source:

public partial class AdventureWorks

{

    public DbSet<vEmployee> vEmployees { get; set; }

}

Summary

Text:

46 Comments

  • Similar i am searching for but my database is Azure database for MySQL 5.7 ExecutionStrategy, DBConfiguration. Where to find them please?

  • Very nice, amazing how far EF has come in only a few years. Thank you for sharing.

  • thanks a lot for a great post.

  • https://ma-study.blogspot.com/

  • The next step of the proposal is to outline the plan of action. Here, you should outline the steps that you will take to complete your proposal. Then, you should provide a schedule to the client so that they can approve your proposal. Moreover, make sure to proofread your proposal before submitting it. The reader should be able to understand the ideas in your proposals. The audience should be able to visualize the problem in the process.

  • The next part of how to write a proposal is the plan of action. The purpose of the plan of action is to outline the steps that you will take to complete your proposal. Typically, this section takes the form of a timeline or a verbal description, but the formality of the proposal will determine which method you choose. The audience will decide whether the project is worth the money. When the audience is satisfied with your work, the proposal will be accepted.

  • Your plan of action section of your proposal should outline the steps that you will take to complete your proposal. This section should be brief and concise, and should be presented in a graphical format if the project is more complicated. The plan of action should be clear and obvious. When possible, the conclusion of your proposal should be stated in the body of the document. After the conclusion, you should thank your audience for their time and clarify the next steps.

  • Are you searching for a UK Dissertation Help Near Me service? Look no encourage. We provide top quality dissertation help at a low price. We have a dedicated team of experts who work hard to guarantee that your dissertation is completed before the deadline. Reach out to us any time of the day or night and enjoy the best online dissertation help.

  • The conclusion should summarize the main points of the essay and state the author's position on the topic. The conclusion should not be too long, and it should not repeat the main ideas. The conclusion should reaffirm the main ideas developed in the body section. If there is a question that is raised in the introduction, the answer to that question should be mentioned there. Once the reader understands this, the essay will have more credibility. This article outlines the basic structure of an essay and explains its different sections.

  • Costs
    The costs of banking website development depend on several factors. First, the developer needs to be experienced in creating such applications. Then, they need to be familiar with the technology stack required for banking software. The developer should also have the necessary experience in building mobile and cross-platform applications. After the development is complete, the developer must also take care of marketing and customer support.

    Another factor that affects the cost of banking website development is the type of development required for the site. For example, a banking website may be designed to look like an official banking site on a desktop. However, the mobile version may not be as functional as the desktop version. A banking website with mobile functionality requires additional development, updating, and servicing.

    The complexity of banking website development can vary widely. A typical project requires approximately 430 hours. However, an experienced team of developers can complete the project in around 210 hours. The development process typically involves several phases, including the design. The initial design stage can take anywhere from a few days to several months.

    The final cost of banking website development is dependent on a variety of factors, including the platform used, complexity of the project, and the development team. Chatbots, customer support, and API integration are critical components to user experience. Customer support should be available round the clock and should answer most frequently asked questions.

    In addition, banks are increasingly creating mobile applications for smartphones and tablets. Mobile applications can help customers manage their accounts and receive notifications about card activities. Mobile banking applications also give customers the option to contact technical support directly through chat.

  • Welcome to FYAB Aesthetic Boutique Wholesale, a reliable and convenient way for Aesthetic Practitioners, Medical Professionals, specialist Training Academy's and Wholesalers to buy the most popular beauty and aesthetics products on the market.

  • Daily entertaining word guessing game. A fascinating brain game

  • It's too bad to check your article late. I wonder what it would be if we met a little faster. I want to exchange a little more, but please visit my site and leave a message!!

  • If you're considering replacing your furnace or heating system or air conditioning, we can assist in that. Our experienced technicians will walk you through the selection process of a new unit along with the installation of your furnace Saskatoon it in a proper manner. This includes ensuring that all permits required are in place and submitting all relevant papers with the city or county authorities prior to starting construction on your house.

  • Explore our four most loved blends in the Sampler Bundle. ROOT Medium Roast, C-4 High Caffeine Blend CODE BLK dark Roast and the FULL FORCE Espresso Blend. This is the perfect way to discover the <a href="https://rampagecoffee.com/">special coffee</a> you've ever had!

  • Explore our four most loved blends in the Sampler Bundle. ROOT Medium Roast, C-4 High Caffeine Blend CODE BLK dark Roast and the FULL FORCE Espresso Blend. This is the perfect way to discover the special coffee you've ever had!

  • Good post, very well written. If you need help writing an equally good post, article, or research paper, we will help you <a href="https://paperwritingservices.reviews/buyessay-org-review/" rel="nofollow">buyessay.org reviewі</a> the result will surprise you!

  • Good post, very well written. If you need help writing an equally good post, article, or research paper, we will help you https://paperwritingservices.reviews/buyessay-org-review/ the result will surprise you!

  • I was looking for another article by chance and found your article I am writing on this topic, so I think it will help a lot. I leave my blog address below. Please visit once.

  • Beef ribs can be divided into two categories which are spare ribs as well as short ribs. The spare ribs originate from the belly part of a cow's rib cage, over the breastbone. Compared to the more well-known pork spare ribs, beef spares are typically longer, larger and are curvier. The tips at the top can be cut off to make short ribs. Short ribs are also sourced from the brisket, plate and flanks of the cow. Both short and spare ribs have connective tissue that, like brisket, requires slow and low cooking in order to tenderize and soften.

  • I've been looking for photos and articles on this topic over the past few days due to a school assignment, <a href="http://clients1.google.pt/url?q=https%3A%2F%2Fmajorcasino.org">baccaratsite</a> and I'm really happy to find a post with the material I was looking for! I bookmark and will come often! Thanks :D

  • Through this article of yours I have improved my experience in software programming, this is quite an important knowledge and I thank you very much!

  • Your article is one of my favorites to read. you write very well I understand what you mean.

  • I was ecstatic to see this website. I wanted to thank you for the excellent read! It is surely entertaining throughout, and I have bookmarked your website to check for fresh content.

  • I really appreciate the developer's investment in this game. Basketball Stars is both attractive with the stressful situation of pitching, as well as tactical and talented. This makes the game a great choice for basketball fans.

  • I want to express my sincere gratitude for your wonderful blog piece. I appreciate your steadfast dedication to disseminating rare knowledge and find it admirable. I look forward to reading more of your content because your website has undoubtedly become one of my most trusted sources of information. Please keep sending us your fantastic content.

  • Your writing is perfect and complete. <a href="https://images.google.dz/url?sa=t&url=https%3A%2F%2Fwww.mtclean.blog/">totosite</a> However, I think it will be more wonderful if your post includes additional topics that I am thinking of. I have a lot of posts on my site similar to your topic. Would you like to visit once?

  • As I am looking at your writing, <a href="https://images.google.dm/url?sa=t&url=https%3A%2F%2Fwww.mtclean.blog/">baccarat online</a> I regret being unable to do outdoor activities due to Corona 19, and I miss my old daily life. If you also miss the daily life of those days, would you please visit my site once? My site is a site where I post about photos and daily life when I was free.

  • Looking at this article, I miss the time when I didn't wear a mask. <a href="https://images.google.dk/url?sa=t&url=https%3A%2F%2Fwww.mtclean.blog/">casinosite</a> Hopefully this corona will end soon. My blog is a blog that mainly posts pictures of daily life before Corona and landscapes at that time. If you want to remember that time again, please visit us.

  • I've been searching for hours on this topic and finally found your post. <a href="https://images.google.dj/url?sa=t&url=https%3A%2F%2Fwww.mtclean.blog/">slotsite</a>, I have read your post and I am very impressed. We prefer your opinion and will visit this site frequently to refer to your opinion. When would you like to visit my site?

  • You are an amazing person to me. You have something new that I keep looking for. Because your article is so good that I can't stop reading.

  • Understanding the intricate mapping between .NET types and SQL database data types is crucial in mastering LINQ to Entities. The way Entity Framework Core efficiently bridges these distinct data systems is truly fascinating. It's impressive how EF Core aptly aligns SQL types to their corresponding .NET counterparts, offering seamless compatibility and smooth data manipulation. Exploring these mappings through LINQ provides a deeper insight into the world of Object-Relational Mapping, making database interaction a more cohesive and understandable process.

  • Understanding the intricate mapping between .NET types and SQL database data types is crucial in mastering LINQ to Entities. The way Entity Framework Core efficiently bridges these distinct data systems is truly fascinating. It's impressive how EF Core aptly aligns SQL types to their corresponding .NET counterparts, offering seamless compatibility and smooth data manipulation. Exploring these mappings through LINQ provides a deeper insight into the world of Object-Relational Mapping, making database interaction a more cohesive and understandable process.

  • I am also visiting this web site daily, this website is in fact fastidious and the people are genuinely sharing nice thoughts.

  • Thanks for sharing this informative and useful article.

  • Great things you’ve always shared with us. Just keep writing this kind of posts.

  • Dixin's blog, a fusion of coding mastery and photography artistry, reflects a dynamic skill set. From Microsoft MVP in coding to a photographer, the content showcases versatility. The deep dive into Entity Framework Core and LINQ to Entities adds substantial value to the coding community. Bravo!

  • Good music, good beats, top rap battle on Friday Night Funkin, anyone who plays it will love it.

  • Pretty nice post. I just stumbled upon your weblog and wanted to say that I have really enjoyed browsing your blog posts

  • I look forward to reading more of your work in the future.

  • I really like your article. You wrote it very well.

  • You are obviously very knowledgeable.The article has truly peaked my interest. Appreciate it!

  • Thank you for sharing your knowledge and expertise with the world.

  • Great things you’ve always shared with us.


  • Your article has opened my eyes to new possibilities and perspectives.

  • Local professionals can serve as mentors, providing students with guidance, career advice, and personal development support. This personalized attention can motivate students and help them set and achieve their goals.

Add a Comment

As it will appear on the website

Not displayed

Your website