So the virtual and override keywords are new additions as of solidity v0.6.0, when they were introduced, they allow much more flexibility when inheriting from other smart contracts, especially when using multiple levels of inheritance when thing scan get tricky. So the purpose of these keywords is to be more explicit when overriding a function.
When should you put the virtual keyword on a function?
Base functions can be overridden by inheriting contratcs to change their behavouir if they are marked as virtual. So if you wish to give a future user of your contratc the ability to modifiy and change its implementation and functionailty then the virtual keyword should be included in your function definition
When should you put the keyword override on a function?
As i just said, base functions can be overridden by inheriting contratcs to change their behavouir if they are marked as virtual. The ovveriding function in this inherited contratc must be labelled with the override keyword in order to be able to actually do and override. This keyword allows developers to signal their intent for certain functions so that others have a better understanding of the purpose of the function. So not all inherited functions have to be override the option exists only for scenarios where the functionality is needed.
Why would a function have both virtual and override keywords on it?
If your like me then you noticed that many functions in the openzeppelin library contracts used both the override and virtual keywords in tandom in the same function. This is because a function may want to be virtual so that if some future user comes along and inherits from it then they may want to do their own ovveride. And if the function that they are overriding also is inheriting from some higher parent contratc (multiple levels of inheritance) then the function must also be labelled with the override keyword. We notice this a lot in the openzeppelin library since a many contratcs have multiple levels of inheritance. So to summarise itās a good idea to use virtual and override if you are inheriting from a parent contract that you wish to override and that you also wish to give future developers the ability to override your new function at some point in the future.
For example in the screenshot below of my transfer function we can see that the function declaration is underlined so we have an error. In this transfer function i make a few modifications for the specific needs of my contract such as the require statements etc and then i call
ERC20.transfer(recipient, amount)
which is a call to the transfer function in the ERC20.sol contratc. However since i dont have the override keyword in my function statement solidity throws an error because were not specifying that we want to ovveride this function. Thus we must include it to get rid of the error.

(TYPO-- recipient == msg.sender should be !=) Note alswell that i have also included the virtual keyword argument in my function statement. But notice how if we remove the virtual keyword we dont get an error.

Thats because we dont techically need the virtual keyword here. However in this case i wanted to give the option of future users of my contratct to have the ability to override my transfer function. (Not that anyone would ever be using my contratc i know lol) but i just wanted to include it anyway.
Perhaps though it could be good not to be over reliant on these keywords as they do seem to simplify the process of things a lot and saves us time having to actually sift through parent contracts to see whats actually going on in many functions. So perhaps that is something to consider??? just a thought