并发编程Java线程

时间:2017-12-11 00:23:56

标签: java concurrency java-threads

我被赋予了上课的任务,但是我们只是在一定程度上被教导,这留下了知识空白。我们被要求完成我们在课堂上所做的一些剩余的代码,导师已经描述了需要做什么而不是如何,以及自从我上次做任何独立Java工作以来的时间大约是7个月所以我是下面有点生锈的是我们用导师指导编写的代码,并且在代码的描述之下。

public class CardHolder implements Runnable {
private int id;
private Account account;
final static int numIterations = 20;

public CardHolder(int id, Account account) {
    this.id = id;
    this.account = account;
}

/*
 * run method is what is executed when you start a Thread that
 * is initialised with an instance of this class.
 * You will need to add code to keep track of local balance (cash
 * in hand) and report this when the thread completes.
 */
public void run() {
    for (int i = 0; i < numIterations; i++) {
        // Generate a random amount from 1-10
        int amount = (int)(Math.random()*10)+1;
        // Then with 50/50 chance, either deposit or withdraw it
        if (Math.random() > 0.5) {
            account.withdraw(id, amount); 
        } else {
            account.deposit(id, amount); 
        }
        try {
            Thread.sleep(200);
        } catch (InterruptedException e) {
            e.printStackTrace();
        } 
    }
    System.out.println("THREAD "+ id + " finished"); 

}
}

import java.util.ArrayList;

public class Account {
private String name;

/**
 * @param args
 */
public static void main(String[] args) {
    // Check to make sure program has been called with correct number of
    // command line arguments
    if (args.length != 3) {
        System.err.println("Error: program should take exactly three command line arguments:");
        System.err.println("\t<No. of accounts> <main acct starting bal.> <backup acct. starting bal.>");
        System.exit(0);
    }
    // And then make sure that those args are all integers
    try {
        int numCards = Integer.parseInt(args[0]);
        Account account = new Account("Main", Integer.parseInt(args[1]));
        Account backup = new Account("Backup", Integer.parseInt(args[2]));

        // Your code to create and manage the threads should go here.
    } catch (NumberFormatException e) {
        System.err.println("All three arguments should be integers");
        System.err.println("\t<No. of accounts> <main acct starting bal.> <backup acct. starting bal.>");
    }
}

// Create an account - initalisation goes in here
public Account(String name, int startingBalance) {
}

// Deposit <amount> into the account
public void deposit(int id, int amount) { 
} 

// Withdraw <amount> from the account
public void withdraw(int id, int amount) {
}  

// Print out the statement of transactions
public void printStatement() {
    System.out.println("Account \"" + name + "\":");
}
}

eclipse中run config中包含的参数是50 1000 1000

任务

许多银行系统允许多个持卡人访问单个帐户(例如商家) 帐户)。这样做的结果是不止一个人可以尝试 同时从单个账户中提取或存款。如果访问了 帐户是天真地实施(忽略并发编程的原则),这个 可能导致竞争条件,以及以下欺诈成功的可能性: 多个持卡人可能串通并试图进行定时攻击。在这样的中 一次攻击,多名持卡人同时从账户中提取资金, 目的是只扣除一笔账户余额。 你的任务是编写一个程序(下面的详细规范)来模拟 具有多个链接卡的启用ATM的链接银行账户的操作。换句话说,
将有几个人可以访问每个帐户,每个帐户都有他/她自己的卡()。 你的程序应该演示并发编程的原理,使其成为 上述欺诈行为不可能成功

非常感谢任何帮助

由于

1 个答案:

答案 0 :(得分:1)

好的,您需要担心的主要是存款和取款方式。例如:

public Paralelogramo(double x1, double y1, double x2, double y2, double x3, double y3, double x4, double y4) {
  super(P1, P2, P3, P4);
}

这个问题是增量/减量操作可能不是原子的。在引擎盖下可能会发生一些操作,而不仅仅是一个。例如:

public class Account {
    public void deposit(int id, int amount) {
        balance += amount;
    }

    public void withdraw(int id, int amount) {
        balance -= amount;
    }

    private int balance;
}

如果你有很多线程同时执行此操作,可能会导致你不想要的结果。试着想一想为什么。您可以使用 synchronized关键字来避免这种情况,这会强制整个方法以原子方式执行。我认为它的作用是将类的实例转换为互斥

int temp1 = balance;
int temp2 = amount;
int temp3 = temp1 - temp2;
balance = temp3;

为了测试这一点,我不会使用随机数,因为它不是确定性。我要做的是启动说,1000个线程同时执行特定数量的提款。如果您有1,000,000美元的余额,并且您启动1000个线程以超过1000次提取1美元,则在测试后您应该在该帐户中获得$ 0。在添加synchronized关键字之前尝试模拟错误的答案,并通过添加synchronized关键字来验证它是否已修复。