High-Quality Methods Design and Implement High-Quality Methods. Cohesion and Coupling SoftUni Team Technical Trainers Software University

Slides:



Advertisements
Similar presentations
Revealing the Secrets of Self-Documenting Code Svetlin Nakov Telerik Corporation For C# Developers.
Advertisements

Software Engineering and Design Principles Chapter 1.
© Copyright 1992–2004 by Deitel & Associates, Inc. and Pearson Education Inc. All Rights Reserved. Chapter 5 - Functions Outline 5.1Introduction 5.2Program.
Computer Programming and Basic Software Engineering 4. Basic Software Engineering 1 Writing a Good Program 4. Basic Software Engineering 3 October 2007.
© 2006 Pearson Addison-Wesley. All rights reserved2-1 Chapter 2 Principles of Programming & Software Engineering.
Chapter 9: Coupling & Cohesion Omar Meqdadi SE 273 Lecture 9 Department of Computer Science and Software Engineering University of Wisconsin-Platteville.
Software Quality Assurance QA Engineering, Testing, Bug Tracking, Test Automation Software University Technical Trainers SoftUni Team.
C# Advanced Topics Methods, Classes and Objects SoftUni Team Technical Trainers Software University
AngularJS Services Built-in and Custom Services SoftUni Team Technical Trainers Software University
Methods Writing and using methods, overloads, ref, out SoftUni Team Technical Trainers Software University
Advanced JavaScript Course Introduction SoftUni Team Technical Trainers Software University
Designing Methods Correctly, Cohesion and Coupling Svetlin Nakov Telerik Corporation
Software Testing Lifecycle Exit Criteria Evaluation, Continuous Integration Ivan Yonkov Technical Trainer Software University.
© Copyright 1992–2004 by Deitel & Associates, Inc. and Pearson Education Inc. All Rights Reserved. C How To Program - 4th edition Deitels Class 05 University.
Design Patterns: Structural Design Patterns
High-Quality Programming Code Code Correctness, Readability, Maintainability, Testability, Etc. SoftUni Team Technical Trainers Software University
Conditional Statements Implementing Control-Flow Logic in C# SoftUni Team Technical Trainers Software University
Loops Repeating Code Multiple Times SoftUni Team Technical Trainers Software University
Database APIs and Wrappers
Entity Framework Performance SoftUni Team Technical Trainers Software University
Improving the Quality of Existing Code Svetlin Nakov Telerik Corporation
Svetlin Nakov Technical Trainer Software University
Build Processes and Continuous Integration Automating Build Processes Software University Technical Trainers SoftUni Team.
Cohesion and Coupling CS 4311
Introduction CS 3358 Data Structures. What is Computer Science? Computer Science is the study of algorithms, including their  Formal and mathematical.
Defensive Programming, Assertions and Exceptions Designing Fault-Resistant Code SoftUni Team Technical Trainers Software University
Test-Driven Development Learn the "Test First" Approach to Coding SoftUni Team Technical Trainers Software University
Defining Classes Classes, Fields, Constructors, Methods, Properties SoftUni Team Technical Trainers Software University
Functions Reusable Parts of Code SoftUni Team Technical Trainers Software University
Static Members and Namespaces Static Members, Indexers, Operators, Namespaces SoftUni Team Technical Trainers Software University
Graphs and Graph Algorithms Fundamentals, Terminology, Traversal, Algorithms SoftUni Team Technical Trainers Software University
1 CSCD 326 Data Structures I Software Design. 2 The Software Life Cycle 1. Specification 2. Design 3. Risk Analysis 4. Verification 5. Coding 6. Testing.
Using SQL Connecting, Retrieving Data, Executing SQL Commands, … Svetlin Nakov Technical Trainer Software University
Defensive Programming, Assertions and Exceptions Designing Error Steady Code SoftUni Team Technical Trainers Software University
C# Basics Course Introduction Svetlin Nakov Technical Trainer Software University
Defining Classes Classes, Fields, Constructors, Methods, Properties Svetlin Nakov Technical Trainer Software University
Forms Overview, Query string, Submitting arrays, PHP & HTML, Input types, Redirecting the user Mario Peshev Technical Trainer Software.
High-Quality Methods Design and Implement High-Quality Methods. Cohesion and Coupling SoftUni Team Technical Trainers Software University
Exam Preparation Algorithms Course: Sample Exam SoftUni Team Technical Trainers Software University
High-Quality Programming Code Code Correctness, Readability, Maintainability Svetlin Nakov Technical Trainer Software University
High-Quality Code: Course Introduction Course Introduction SoftUni Team Technical Trainers Software University
Design Patterns: Structural Design Patterns General and reusable solutions to common problems in software design Software University
Advanced C# Course Introduction SoftUni Team Technical Trainers Software University
Events Event Handling in JavaScript SoftUni Team Technical Trainers Software University
Object-Oriented Programming Course Introduction Svetlin Nakov Technical Trainer Software University
Reflection Programming under the hood SoftUni Team Technical Trainers Software University
Mocking with Moq Tools for Easier Unit Testing SoftUni Team Technical Trainers Software University
Operators and Expressions
Design Patterns: Behavioral Design Patterns General and reusable solutions to common problems in software design Software University
Mocking Unit Testing Methods with External Dependencies SoftUni Team Technical Trainers Software University
Mocking with Moq Mocking tools for easier unit testing Svetlin Nakov Technical Trainer Software University
Chapter 9: Coupling & Cohesion Omar Meqdadi SE 273 Lecture 9 Department of Computer Science and Software Engineering University of Wisconsin-Platteville.
ORM Basics Repository Pattern, Models, Entity Manager Ivan Yonkov Technical Trainer Software University
High-Quality Code: Course Introduction Course Introduction SoftUni Team Technical Trainers Software University
Functional Programming Data Aggregation and Nested Queries Ivan Yonkov Technical Trainer Software University
Coupling and Cohesion Schach, S, R. Object-Oriented and Classical Software Engineering. McGraw-Hill, 2002.
Coupling and Cohesion Pfleeger, S., Software Engineering Theory and Practice. Prentice Hall, 2001.
Inheritance Class Hierarchies SoftUni Team Technical Trainers Software University
Stacks and Queues Processing Sequences of Elements SoftUni Team Technical Trainers Software University
Generics SoftUni Team Technical Trainers Software University
High-Quality Programming Code Code Correctness, Readability, Maintainability, Testability, Etc. SoftUni Team Technical Trainers Software University
High-Quality Methods How to Design and Implement High-Quality Methods? Understanding Cohesion and Coupling High-Quality Code Telerik Software Academy
Interface Segregation / Dependency Inversion
Repeating Code Multiple Times
Coupling and Cohesion 1.
Iterators and Comparators
Chapter 5 - Functions Outline 5.1 Introduction
Improving the Design “Can the design be better?”
Programming Logic and Design Fourth Edition, Comprehensive
Software Design Lecture : 9.
Presentation transcript:

High-Quality Methods Design and Implement High-Quality Methods. Cohesion and Coupling SoftUni Team Technical Trainers Software University

2  Why Do We Need Methods?  Method-Level Cohesion and Coupling  Strong Cohesion  Loose Coupling  Method Parameters  Pseudocode Table of Contents

4  Methods (functions, routines) are important in software development  Reduce complexity  Divide and conquer  Complex problems are split into composition of smaller problems  Improve code readability  Small methods with good method names make the code self- documenting  Avoid duplicating code  Duplicating code is hard to maintain Why Do We Need Methods?

5  Methods simplify software development  Hide implementation details  Complex logic is encapsulated and hidden behind a simple interface  Algorithms and data structures are hidden and can be transparently replaced later  Increase the level of abstraction  Methods address the business problem, not the technical implementation: Why We Need Methods? (2) Bank.Accounts[customer].Deposit(500);

6  The fundamental principle of correct method usage:  A method should do exactly what its name says  Nothing less (work correctly in all possible scenarios)  Nothing more (no side effects)  In case of incorrect input or incorrect preconditions  An error should be returned (e.g. throw exception) Using Methods: Fundamentals A method should do what its name says or should indicate an error (throw an exception). Any other behavior is incorrect!

7 Bad Methods – Examples int Sum(int[] elements) { int sum = 0; int sum = 0; foreach (int element in elements) foreach (int element in elements) { sum = sum + element; sum = sum + element; } return sum; return sum;} double CalcTriangleArea(double a, double b, double c) { double s = (a + b + c) / 2; double s = (a + b + c) / 2; double area = Math.Sqrt(s * (s - a) * (s - b) * (s - c)); double area = Math.Sqrt(s * (s - a) * (s - b) * (s - c)); return area; return area;} What will happen if we sum 2,000,000, ,000,000,000? What will happen if a = b = c = -1 ? Result: The same result as when a = b = c = 1  both triangles have the same size.

8 Good Methods – Examples long Sum(int[] elements) { long sum = 0; long sum = 0; foreach (int element in elements) foreach (int element in elements) { sum = sum + element; sum = sum + element; } return sum; return sum;} double CalcTriangleArea(double a, double b, double c) { if (a <= 0 || b <= 0 || c <= 0) if (a <= 0 || b <= 0 || c <= 0) { throw new ArgumentException("Sides should be positive."); throw new ArgumentException("Sides should be positive."); } double s = (a + b + c) / 2; double s = (a + b + c) / 2; double area = Math.Sqrt(s * (s - a) * (s - b) * (s - c)); double area = Math.Sqrt(s * (s - a) * (s - b) * (s - c)); return area; return area;}

9  Some methods do not indicate errors correctly  If the property name does not exist  A null reference exception will be thrown (implicitly)  it is not meaningful Indicating an Error internal object GetValue(string propertyName) { PropertyDescriptor descriptor = PropertyDescriptor descriptor = this.propertyDescriptors[propertyName]; this.propertyDescriptors[propertyName]; return descriptor.GetDataBoundValue(); return descriptor.GetDataBoundValue();}

10  Use the correct exception handling instead: Indicating an Error (2) internal object GetValue(string propertyName) { PropertyDescriptor descriptor = PropertyDescriptor descriptor = this.propertyDescriptors[propertyName]; this.propertyDescriptors[propertyName]; if (descriptor == null) if (descriptor == null) { throw new ArgumentException("Property name: " throw new ArgumentException("Property name: " + propertyName + " does not exists!"); + propertyName + " does not exists!"); } return descriptor.GetDataBoundValue(); return descriptor.GetDataBoundValue();}

11  Method that does something different than its name is wrong for at least one of these reasons:  The method sometimes returns incorrect result  bug  The method returns incorrect result when its input is incorrect  low quality  Could be acceptable for private methods only  The method does too many things  bad cohesion  The method has side effects  spaghetti code  Method returns strange value when an error condition happens  it should indicate the error Symptoms of Wrong Methods

12 Wrong Methods – Examples long Sum(int[] elements) { long sum = 0; long sum = 0; for (int i = 0; i < elements.Length; i++) for (int i = 0; i < elements.Length; i++) { sum = sum + elements[i]; sum = sum + elements[i]; elements[i] = 0; // Hidden side effect elements[i] = 0; // Hidden side effect } return sum; return sum;} double CalcTriangleArea(double a, double b, double c) { if (a <= 0 || b <= 0 || c <= 0) if (a <= 0 || b <= 0 || c <= 0) { return 0; // Incorrect result return 0; // Incorrect result } double s = (a + b + c) / 2; double s = (a + b + c) / 2; double area = Math.Sqrt(s * (s - a) * (s - b) * (s - c)); double area = Math.Sqrt(s * (s - a) * (s - b) * (s - c)); return area; return area;}

13  Methods should have strong cohesion  Should address single task and address it well  Should have clear intent  Methods that address several tasks in the same time are hard to be named  Strong cohesion is used in engineering  In computer hardware any PC component solves a single task  E.g. hard disk performs a single task – storage Strong Cohesion

14  Functional cohesion (independent function)  Method performs certain well-defined calculation and returns a single result  The entire input is passed through parameters and the entire output is returned as result  No external dependencies or side effects Acceptable Types of Cohesion Math.Sqrt(value)  square root string.Substring(str, startIndex, length) char.IsLetterOrDigit(ch)

15  Sequential cohesion (algorithm)  Method performs certain sequence of operations to perform a single task and achieve certain result  It encapsulates an algorithm  Example: 1. Connect to mail server 2. Send message headers 3. Send message body 4. Disconnect from the server Acceptable Types of Cohesion (2) Send (recipient, subject, body)

16  Communicational cohesion (common data)  A set of operations used to process certain data and produce a result  Example: 1. Retrieve input data from database 2. Perform internal calculations over retrieved data 3. Build the report 4. Format the report as Excel worksheet 5. Display the Excel worksheet on the screen Acceptable Types of Cohesion (3) DisplayAnnualExpensesReport(int employeeId)

17  Temporal cohesion (time related activities)  Operations that are generally not related but need to happen in a certain moment  Examples: 1. Load user settings 2. Check for updates 3. Load all invoices from the database  Sequence of actions to handle the event Acceptable Types of Cohesion (4) InitializeApplication() ButtonConfirmClick()

18  Logical cohesion  Performs a different operation depending on an input parameter  Incorrect example:  Can be acceptable in event handlers  E.g. the KeyDown event in Windows Forms) Unacceptable Cohesion object ReadAll(int operationCode) { if (operationCode == 1) … // Read person name if (operationCode == 1) … // Read person name else if (operationCode == 2) … // Read address else if (operationCode == 2) … // Read address else if (operationCode == 3) … // Read date else if (operationCode == 3) … // Read date …}

19  Encapsulates an algorithm inside a class  Making each algorithm replaceable by others  All the algorithms can work with the same data transparently  The client can work with each algorithm transparently  Strategy Pattern

20 class QuickSort : SortStrategy { public override void Sort(IList list) {... } public override void Sort(IList list) {... }} class SortedList { private IList list = new List (); private IList list = new List (); public void Sort(SortStrategy strategy) { public void Sort(SortStrategy strategy) { // sortStrategy can be passed in constructor // sortStrategy can be passed in constructor sortStrategy.Sort(list); sortStrategy.Sort(list); }} class MergeSort : SortStrategy { public override void Sort(IList list) {... } public override void Sort(IList list) {... }} abstract class SortStrategy { public abstract void Sort(IList list); }

21  Coincidental cohesion (spaghetti)  Not related (random) operations grouped in a method for unclear reason  Incorrect example: 1. Prepares annual incomes report for given customer 2. Sorts an array of integers in increasing order 3. Calculates the square root of given number 4. Converts given MP3 file into WMA format 5. Sends to given customer Unacceptable Cohesion HandleStuff(customerId, int[], ref sqrtValue, mp3FileName, Address)

22  What is loose coupling?  Minimal dependences of the method on the other parts of the source code  Minimal dependences on the class members or external classes and their members  No side effects  If the coupling is loose, we can easily reuse a method or group of methods in a new project  Tight coupling  spaghetti code Loose Coupling

Loose Coupling (2)  The ideal coupling  A methods depends only on its parameters  Does not have any other input or output  Example: Math.Sqrt()  Real world  Complex software cannot avoid coupling but could make it as loose as possible  Example: complex encryption algorithm performs initialization, encryption, finalization 23

24  Intentionally increased coupling for more flexibility (.NET cryptography API): Coupling – Example byte[] EncryptAES(byte[] inputData, byte[] secretKey) { Rijndael cryptoAlg = new RijndaelManaged(); Rijndael cryptoAlg = new RijndaelManaged(); cryptoAlg.Key = secretKey; cryptoAlg.Key = secretKey; cryptoAlg.GenerateIV(); cryptoAlg.GenerateIV(); MemoryStream destStream = new MemoryStream(); MemoryStream destStream = new MemoryStream(); CryptoStream csEncryptor = new CryptoStream( CryptoStream csEncryptor = new CryptoStream( destStream, cryptoAlg.CreateEncryptor(), destStream, cryptoAlg.CreateEncryptor(), CryptoStreamMode.Write); CryptoStreamMode.Write); csEncryptor.Write(inputData, 0, inputData.Length); csEncryptor.Write(inputData, 0, inputData.Length); csEncryptor.FlushFinalBlock(); csEncryptor.FlushFinalBlock(); byte[] encryptedData = destStream.ToArray(); byte[] encryptedData = destStream.ToArray(); return encryptedData; return encryptedData;}

25  To reduce coupling we can make utility classes  Hide the complex logic and provide simple straightforward interface (a.k.a. façade): Loose Coupling – Example byte[] EncryptAES(byte[] inputData, byte[] secretKey) { MemoryStream inputStream = new MemoryStream(inputData); MemoryStream inputStream = new MemoryStream(inputData); MemoryStream outputStream = new MemoryStream(); MemoryStream outputStream = new MemoryStream(); EncryptionUtils.EncryptAES( EncryptionUtils.EncryptAES( inputStream, outputStream, secretKey); inputStream, outputStream, secretKey); byte[] encryptedData = outputStream.ToArray(); byte[] encryptedData = outputStream.ToArray(); return encryptedData; return encryptedData;}

26  Passing parameters through class fields  Typical example of tight coupling  Don't do this unless you have a good reason! Tight Coupling – Example class Sumator { public int a, b; public int a, b; int Sum() int Sum() { return a + b; return a + b; } static void Main() static void Main() { Sumator sumator = new Sumator() { a = 3, b = 5 }; Sumator sumator = new Sumator() { a = 3, b = 5 }; Console.WriteLine(sumator.Sum()); Console.WriteLine(sumator.Sum()); }}

27  Say, we have a large piece of software  We need to update subsystems and the subsystems are not really independent  E.g. a change in filtering affects sorting, etc: Tight Coupling in Real World Code class GlobalManager { public void UpdateSorting() {…} public void UpdateSorting() {…} public void UpdateFiltering() {…} public void UpdateFiltering() {…} public void UpdateData() {…} public void UpdateData() {…} public void UpdateAll () {…} public void UpdateAll () {…}}

28  Say, we have an application consisting of two layers:  Do not update top-down and bottom-up from a single method!  E.g. RemoveCustomer() method in the DataLayer changes also the presentation layer  Better use a notification (observer pattern / event) Cohesion Problems in Real-World Code Data Layer Presentation Layer

29  Reducing coupling with OOP techniques  Abstraction  Define a public interface and hide the implementation details  Encapsulation  Make methods and fields private unless they are definitely needed  Define new members as private  Increase visibility as soon as this is needed Loose Coupling and OOP

30  Method is coupled to its parameters  This is the best type of coupling  Method in a class is coupled to some class fields  This coupling is usual, do not worry too much  Method in a class is coupled to static methods, properties or constants in an external class Acceptable Coupling static int Sum(int[] elements) { … } static int CalcArea() { return this.Width * this.Height; } static double CalcCircleArea(double radius) { return Math.PI * radius * radius; }

31  Method in a class is coupled to static fields in external class  Use private fields and public properties  Methods take as input data some fields that could be passed as parameters  Check the intent of the method  Is it designed to process internal class data or is utility method?  Method is defined public without being part of the public class' interface  possible coupling Non-Acceptable Coupling

32  Put most important parameters first  Put the main input parameters first  Put non-important optional parameters last  Example:  Incorrect example: Method Parameters void RegisterUser(string username, string password, Date accountExpirationDate, Role[] roles) Date accountExpirationDate, Role[] roles) void RegisterUser(Role[] roles, string password, string username, Date accountExpirationDate) Date accountExpirationDate) void RegisterUser(string password, Date accountExpirationDate, Role[] roles, string username)

33  Do not modify the input parameters  Use new variables instead  Incorrect example:  Correct example: Methods Parameters (2) bool CheckLogin(string username, string password) { username = username.ToLower(); username = username.ToLower(); // Check the username / password here … // Check the username / password here …} bool CheckLogin(string username, string password) { string usernameLowercase = username.ToLower(); string usernameLowercase = username.ToLower(); // Check the username / password here … // Check the username / password here …}

34  Use parameters consistently  Use the same names and the same order in all methods  Incorrect example:  Output parameters should be put last Method Parameters (3) void EncryptFile(Stream input, Stream output, string key); void DecryptFile(string key, Stream output, Stream input); FindCustomersAndIncomes(Region region, out Customer[] customers, out decimal[] incomes) out Customer[] customers, out decimal[] incomes)

35  When should we pass an object containing few values and when these values separately?  Sometime we pass an object and use only a single field of it  Is this a good practice?  Examples:  Look at the method's level of abstraction  Is it intended to operate with employees of with rates and months?  the first is incorrect Pass Entire Object or Its Fields? CalculateSalary(Employee employee, int months); CalculateSalary(double rate, int months);

36  Limit the number of parameters to 7 (+/-2)  7 is a "magic" number in psychology  Human brain cannot process more than 7 (+/-2) things in the same time  If the parameters need to be too many, reconsider the method's intent  Does it have a clear intent?  Consider extracting few of the parameters in a new class How Many Parameters a Method Should Have?

37  How long should a method be?  There is no specific restriction  Avoid methods longer than one screen (30 lines)  Long methods are not always bad  Be sure you have a good reason for their length  Cohesion and coupling are more important than the method length!  Long methods often contain portions that could be extracted as separate methods with good names and clear intent Method Length

38  Pseudocode can be helpful in:  Routines design  Routines coding  Code verification  Cleaning up unreachable branches in a routine Pseudocode

39  What the routine will abstract i.e. the information a routine will hide?  Routine input parameters  Routine output  Preconditions  Conditions that have to be true before a routine is called  Postconditions  Conditions that have to be true after routine execution Designing in Pseudocode

40  Why it is better to spend time on design before you start coding?  The functionality may be already available in a library, so you do not need to code at all!  You need to think of the best way to implement the task considering your project requirements  If you fail on writing the code right the first time, you need to know that programmers get emotional to their code Design Before Coding

41 Pseudocode Example Routine that evaluates an aggregate expression for a database column (e.g. Sum, Avg, Min) Parameters: Column Name, Expression Preconditions: (1) Check whether the column exists and throw an argument exception if not (2) If the expression parser cannot parse the expression throw an ExpressionParsingException Routine code: Call the evaluate method on the DataView class and return the resulting value as string

42  Public routines in libraries and system software are hard to change  Because customers want no breaking changes  Two reasons why you need to change a public routine:  New functionality has to be added conflicting with the old features  The name is confusing and makes library usage unintuitive or inconvenient  Design better upfront, or refactor carefully Public Routines in Libraries

43  Deprecated method  About to be removed in future versions  When deprecating an old method  Include that in the documentation  Specify the new method that has to be used  Use the [Obsolete] attribute in.NET Method Deprecation [Obsolete("CreateXml() method is deprecated. Use CreateXmlReader instead.")] public void CreateXml (…) { … }

44  Inline routines (in C / C++) provide two benefits:  Performance benefit of not creating a new routine on the stack  Abstraction – use a well named routine, instead of inlined code  Some applications (e.g. games) need that optimization  Used for the most frequently used routines  Example: a short routine called 100,000 times  Not all languages support inline routines  The C# compiler inlines routines at compile time when necessary Inline Routines

Recursion To Understand Recursion, One Must First Understand Recursion

46  Useful when you want to walk (traverse) trees or graph-like structures  Be aware of infinite recursion or indirect recursion  Recursion example: Recursion void PrintWindowsRecursive(Window w) { w.Print() w.Print() foreach(childWindow in w.ChildWindows) foreach(childWindow in w.ChildWindows) { PrintWindowsRecursive(childWindow); PrintWindowsRecursive(childWindow); } } } }

47  Ensure that recursion has an end (bottom)  Verify that recursion is not very high-cost  Check the occupied system resources  You can always use stack classes and iteration  Don't use recursion when there is better linear (iteration based) solution, e.g.  Factorials  Fibonacci numbers  Some languages optimize tail-call recursions Recursion Tips

High-Quality Methods Exercises in Class

49 1.Designing methods  No single solution  There are various tradeoffs 2.Choosing the best approach  Evaluate the requirements  Choose the most appropriate solution  Try to be flexible if the requirements change  Ensure strong cohesion and loose coupling Summary

? ? ? ? ? ? ? ? ? High-Quality Methods httpshttps://softuni.bg/courses/oop/

License  This course (slides, examples, demos, videos, homework, etc.) is licensed under the "Creative Commons Attribution- NonCommercial-ShareAlike 4.0 International" licenseCreative Commons Attribution- NonCommercial-ShareAlike 4.0 International 51  Attribution: this work may contain portions from  "Fundamentals of Computer Programming with C#" book by Svetlin Nakov & Co. under CC-BY-SA licenseFundamentals of Computer Programming with C#CC-BY-SA  "High Quality Code" course by Telerik Academy under CC-BY-NC-SA licenseHigh Quality CodeCC-BY-NC-SA

Free Software University  Software University Foundation – softuni.orgsoftuni.org  Software University – High-Quality Education, Profession and Job for Software Developers  softuni.bg softuni.bg  Software Facebook  facebook.com/SoftwareUniversity facebook.com/SoftwareUniversity  Software YouTube  youtube.com/SoftwareUniversity youtube.com/SoftwareUniversity  Software University Forums – forum.softuni.bgforum.softuni.bg