Look into the code as below, the task whoAmI() is not declared using 'virtual', but when calling it from class B, it actually get B's task executed. Then it seems that the 'virtual' keyword is not necessary here?
class A;
task whoAmI();
$display("This is A");
endtask // whoAmI
endclass // Aclass B extends A;
task whoAmI();
$display("This is B");
endtask // whoAmI
endclass // B
B b = new();
b.whoAmI();