Fonte: Medium Hoje iremos desenvolver uma aplicação Java simples para um sistema bancário. Isso nos ajudará a entender melhor como os conceitos OOP são usados em programas Java. Para começar, precisaremos de um ambiente Java instalado no computador, de preferência Java 11. A seguir, começaremos com uma descrição detalhada das funções do aplicativo console. Funcional:
- Crie a sua conta aqui;
- Entrada saida;
- Exibição das últimas 5 transações;
- Depósito em dinheiro;
- Exibir informações do usuário atual.
- Herança;
- Polimorfismo;
- Encapsulamento.
Desenvolvimento de aplicações
Vamos criar um novo projeto Java no Eclipse ou IntelliJ IDEA. Vamos definir uma nova interface chamada SavingsAccount .public interface SavingsAccount {
void deposit(double amount, Date date);
}
Implementei uma interface que hospeda o método de depósito . Eu chamo esse método sempre que adiciono dinheiro à minha conta corrente. O conceito OOP usado aqui é polimorfismo (métodos em uma interface não possuem corpo). A implementação deste método pode ser encontrada na classe Customer substituindo um método com o mesmo nome e parâmetros. Dessa forma, você substitui um método da interface pai na classe filha. Então precisamos de um cliente para adicionar dinheiro na conta corrente. Mas primeiro, vamos definir nossa classe Cliente .
public class Customer extends Person implements SavingsAccount {
private String username;
private String password;
private double balance;
private ArrayList<String> transactions = new ArrayList<>(5);
public Customer(String firstName, String lastName, String address, String phone, String username, String password, double balance, ArrayList<String> transactions, Date date) {
super(firstName, lastName, address, phone);
this.username = username;
this.password = password;
this.balance = balance;
addTransaction(String.format("Initial deposit - " + NumberFormat.getCurrencyInstance().format(balance) + " as on " + "%1$tD" + " at " + "%1$tT.", date));
}
private void addTransaction(String message) {
transactions.add(0, message);
if (transactions.size() > 5) {
transactions.remove(5);
transactions.trimToSize();
}
}
//Getter Setter
public ArrayList<String> getTransactions() {
return transactions;
}
@Override
public void deposit(double amount, Date date) {
balance += amount;
addTransaction(String.format(NumberFormat.getCurrencyInstance().format(amount) + " credited to your account. Balance - " + NumberFormat.getCurrencyInstance().format(balance) + " as on " + "%1$tD" + " at " + "%1$tT.", date));
}
@Override
public String toString() {
return "Customer{" +
"username='" + username + '\'' +
", password='" + password + '\'' +
", balance=" + balance +
", transactions=" + transactions +
'}';
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Customer customer = (Customer) o;
if (Double.compare(customer.getBalance(), getBalance()) != 0) return false;
if (getUsername() != null ? !getUsername().equals(customer.getUsername()) : customer.getUsername() != null)
return false;
if (getPassword() != null ? !getPassword().equals(customer.getPassword()) : customer.getPassword() != null)
return false;
return getTransactions() != null ? getTransactions().equals(customer.getTransactions()) : customer.getTransactions() == null;
}
@Override
public int hashCode() {
int result;
long temp;
result = getUsername() != null ? getUsername().hashCode() : 0;
result = 31 * result + (getPassword() != null ? getPassword().hashCode() : 0);
temp = Double.doubleToLongBits(getBalance());
result = 31 * result + (int) (temp ^ (temp >>> 32));
result = 31 * result + (getTransactions() != null ? getTransactions().hashCode() : 0);
return result;
}
O conceito OOP usado aqui é herança , pois a classe Customer recebe propriedades da classe Person . Ou seja, quase todos os atributos da classe Person são herdados e os relacionamentos pai-filho são visíveis de Person para Customer . Agora precisamos de um construtor com todos os atributos das duas classes e adicionando uma palavra-chave superconstrutor para especificar os atributos herdados. Ao implementar a interface SavingsAccount , devemos substituir o método de depósito na classe Customer . Para fazer isso, escreveremos uma implementação do método nesta classe. Além disso, a lista de transações é inicializada para exibir as últimas cinco transações. O construtor chama o método addTransaction , que exibe a data das alterações e transações concluídas.
private void addTransaction(String message) {
transactions.add(0, message);
if (transactions.size() > 5) {
transactions.remove(5);
transactions.trimToSize();
}
}
Agora vamos falar sobre a classe pai Person .
public class Person {
private String firstName;
private String lastName;
private String address;
private String phone;
public Person() {}
public Person(String firstName, String lastName, String address, String phone) {
this.firstName = firstName;
this.lastName = lastName;
this.address = address;
this.phone = phone;
}
//Getters Setters
@Override
public String toString() {
return "Person{" +
"firstName='" + firstName + '\'' +
", lastName='" + lastName + '\'' +
", address='" + address + '\'' +
", phone='" + phone + '\'' +
'}';
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Person person = (Person) o;
if (getFirstName() != null ? !getFirstName().equals(person.getFirstName()) : person.getFirstName() != null)
return false;
if (getLastName() != null ? !getLastName().equals(person.getLastName()) : person.getLastName() != null)
return false;
if (getAddress() != null ? !getAddress().equals(person.getAddress()) : person.getAddress() != null)
return false;
return getPhone() != null ? getPhone().equals(person.getPhone()) : person.getPhone() == null;
}
@Override
public int hashCode() {
int result = getFirstName() != null ? getFirstName().hashCode() : 0;
result = 31 * result + (getLastName() != null ? getLastName().hashCode() : 0);
result = 31 * result + (getAddress() != null ? getAddress().hashCode() : 0);
result = 31 * result + (getPhone() != null ? getPhone().hashCode() : 0);
return result;
}
Na classe Person , utilizamos o conceito de encapsulamento aplicando o modificador de acesso privado a cada atributo. O encapsulamento em Java pode ser definido como o mecanismo pelo qual os métodos que operam nesses dados são combinados em uma única unidade. Essencialmente, os dados da classe Person estão disponíveis apenas nesta classe e não em outras classes ou pacotes. E por fim, nossa classe principal, chamada Bank . Esta é a classe principal de onde executamos a aplicação e interagimos com a funcionalidade de todas as classes.
public class Bank {
private static double amount = 0;
Map<String, Customer> customerMap;
Bank() {
customerMap = new HashMap<String, Customer>();
}
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
Customer customer;
Bank bank = new Bank();
int choice;
outer:
while (true) {
System.out.println("\n-------------------");
System.out.println("BANK OF JAVA");
System.out.println("-------------------\n");
System.out.println("1. Registrar cont.");
System.out.println("2. Login.");
System.out.println("3. Exit.");
System.out.print("\nEnter your choice : ");
choice = sc.nextInt();
sc.nextLine();
switch (choice) {
case 1:
System.out.print("Enter First Name : ");
String firstName = sc.nextLine();
System.out.print("Enter Last Name : ");
String lastName = sc.nextLine();
System.out.print("Enter Address : ");
String address = sc.nextLine();
System.out.print("Enter contact number : ");
String phone = sc.nextLine();
System.out.println("Set Username : ");
String username = sc.next();
while (bank.customerMap.containsKey(username)) {
System.out.println("Username already exists. Set again : ");
username = sc.next();
}
System.out.println("Set a password:");
String password = sc.next();
sc.nextLine();
customer = new Customer(firstName, lastName, address, phone, username, password, new Date());
bank.customerMap.put(username, customer);
break;
case 2:
System.out.println("Enter username : ");
username = sc.next();
sc.nextLine();
System.out.println("Enter password : ");
password = sc.next();
sc.nextLine();
if (bank.customerMap.containsKey(username)) {
customer = bank.customerMap.get(username);
if (customer.getPassword().equals(password)) {
while (true) {
System.out.println("\n-------------------");
System.out.println("W E L C O M E");
System.out.println("-------------------\n");
System.out.println("1. Deposit.");
System.out.println("2. Transfer.");
System.out.println("3. Last 5 transactions.");
System.out.println("4. User information.");
System.out.println("5. Log out.");
System.out.print("\nEnter your choice : ");
choice = sc.nextInt();
sc.nextLine();
switch (choice) {
case 1:
System.out.print("Enter amount : ");
while (!sc.hasNextDouble()) {
System.out.println("Invalid amount. Enter again :");
sc.nextLine();
}
amount = sc.nextDouble();
sc.nextLine();
customer.deposit(amount, new Date());
break;
case 2:
System.out.print("Enter beneficiary username : ");
username = sc.next();
sc.nextLine();
System.out.println("Enter amount : ");
while (!sc.hasNextDouble()) {
System.out.println("Invalid amount. Enter again :");
sc.nextLine();
}
amount = sc.nextDouble();
sc.nextLine();
if (amount > 300) {
System.out.println("Transfer limit exceeded. Contact bank manager.");
break;
}
if (bank.customerMap.containsKey(username)) {
Customer payee = bank.customerMap.get(username); //Todo: check
payee.deposit(amount, new Date());
customer.withdraw(amount, new Date());
} else {
System.out.println("Username doesn't exist.");
}
break;
case 3:
for (String transactions : customer.getTransactions()) {
System.out.println(transactions);
}
break;
case 4:
System.out.println("Titularul de cont cu numele: " + customer.getFirstName());
System.out.println("Titularul de cont cu prenumele : " + customer.getLastName());
System.out.println("Titularul de cont cu numele de utilizator : " + customer.getUsername());
System.out.println("Titularul de cont cu addresa : " + customer.getAddress());
System.out.println("Titularul de cont cu numarul de telefon : " + customer.getPhone());
break;
case 5:
continue outer;
default:
System.out.println("Wrong choice !");
}
}
} else {
System.out.println("Wrong username/password.");
}
} else {
System.out.println("Wrong username/password.");
}
break;
case 3:
System.out.println("\nThank you for choosing Bank Of Java.");
System.exit(1);
break;
default:
System.out.println("Wrong choice !");
}}}}
Usando a biblioteca java.util , chamamos o Scanner para ler o teclado. Ao lançar o objeto Cliente através de sua definição, conhecida como Banco , criamos um novo objeto do tipo Banco() . Depois de algum tempo, exibimos o menu iniciar. Ao usar nextLine , o número adicionado no teclado é lido. Abaixo temos um novo construtor que salva nosso mapa , os dados do cliente. Map.put é usado para salvar ou atualizar dados do cliente. cliente = novo Cliente(nome, sobrenome, endereço, telefone, nome de usuário, senha, nova data());
bank.customerMap.put(username, customer);
Se houver conexão, obtemos um novo menu com opções. A mesma abordagem funciona usando while e switch para chamar a funcionalidade do aplicativo. Etapa 1: Adicione dinheiro à sua conta corrente. Etapa 2: exiba as últimas 5 transações. Etapa 3: enviamos os dados do cliente do cartão para o console. Etapa 4: Feche o menu. O código fonte do programa pode ser encontrado aqui . Espero que este exemplo ajude você a se familiarizar mais com o uso dos conceitos OOP em Java.
GO TO FULL VERSION