java - 当我需要通知监听器时,为什么监听器集是空的? (在 Android 应用程序中)

标签 java android listener

在我的 Android 应用程序中,我有三个类,一个是 MainActivity,另一个称为 CustomerTransaction,它只是一个 Java 类。还有另一个名为 DataActivity 的 Activity ,当单击 MainActivity 屏幕上的“输入数据”按钮时,该 Activity 就会启动。 Activity DataActivity 允许用户在 EditText 中输入数字,然后单击“发送”按钮以触发 CustomerTransaction 中的方法。我将在下面添加完整的代码。但首先让我解释一下我的问题。在 CustomerTransaction 中,我有一个方法为两个变量提供值:totalAmount 和 transactionType(均为 int)。当这些值可用时,MainActivity 中需要将这些值显示在主屏幕上。 MainActivity 实现一个名为 CustomListeners 的接口(interface),其中包含显示两个值(在 TextView 中)的函数的 header 。通过在值可用时通知监听器,将值从 CustomerTransaction 传送到 MainActivity 中的方法。问题是,当我有值时,监听器集是空的。我想了解为什么会发生这种情况,然后更正代码。

代码如下:

package com.example.customlistenertest;

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;

public class MainActivity extends Activity implements CustomListeners {
TextView tv_amount;
TextView tv_transactionType;
Button btn_enterData;
CustomerTransaction control= new CustomerTransaction();

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    control.registerListener(this);

    tv_amount = (TextView) findViewById(R.id.textView_amount);
    tv_transactionType = (TextView) findViewById(R.id.textView_transactionType);
    btn_enterData = (Button) findViewById(R.id.button_enterData);

    addListenerOnButtonEnterData();
}

public void addListenerOnButtonEnterData() {
    btn_enterData.setOnClickListener(new OnClickListener() {

@Override
public void onClick(View arg0) {
Intent intent = 
new Intent(MainActivity.this,com.example.customlistenertest.DataActivity.class);       
startActivity(intent); 
        }
    });

}

public void updateTotal(final int transactionType, final int totalAmount) {
   runOnUiThread(new Runnable() {
   public void run() {
    tv_amount.setText(Integer.toString(totalAmount));
    tv_transactionType.setText(Integer.toString(transactionType));
   }});
}

@Override    
public void onStart() {
    super.onStart();
}
@Override
 public void onResume(){
    super.onResume();
    control.registerListener(this);
 }
 @Override
 protected void onPause() {
     super.onPause();
 }

@Override
 protected void onDestroy() {
     super.onDestroy();
     control.unregisterListener(this);
 }

} // End MainActivity

以下是 DataActivity 类的代码:

package com.example.customlistenertest;

import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;

public class DataActivity extends Activity {

TextView tv_enterNumber;
EditText et_number;
Button btn_send;
CustomerTransaction transaction;

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.enter_data);

    btn_send = (Button) findViewById(R.id.button_send);
    et_number = (EditText) findViewById(R.id.editText_number);
}

public void send(View view) {
    transaction = new CustomerTransaction();
    transComplete();
}

private void transComplete() {
    new Thread("TransComplete") {
        public void run() {
            String s_number = et_number.getText().toString();
            int value = Integer.parseInt(s_number);
            transaction.setState(5, value);
            finish();
        }
    }.start();
}
} // end DataActivity

以下是 CustomerTransaction 类的代码:

package com.example.customlistenertest;

import java.util.HashSet;
import java.util.Set;

public class CustomerTransaction {
public static final int DATA_ENTERED = 1;
public static final int TRANSACTION_COMPLETE = 5;
private int state;
private int totalAmount;
private int transactionType;
public Set<CustomListeners> listeners = new HashSet<CustomListeners>();


public synchronized void registerListener(CustomListeners listener) {
    listeners.add(listener);
}

public synchronized void unregisterListener(CustomListeners listener) {
       listeners.remove(listener);
}

public synchronized void notifyListeners() {
   for(CustomListeners listener : listeners) {
 if (totalAmount != 0)
    listener.updateTotal(transactionType,totalAmount);
else
    listener.updateTotal(transactionType,0);
   }
}

public void performOperation() {

  new Thread("MyThread") {
   public void run() { 
    notifyListeners();
 }
 }.start();
}    
public void setState(final int state, final int value) {
   this.state = state;
   this.totalAmount = value;
   this.transactionType = 999;
   if (state == DATA_ENTERED) {

   }
   if (state == TRANSACTION_COMPLETE) {
        performOperation();
   }
 }

} // end CustomerTransactionjava

我还将添加两个布局文件。第一个是activity_main.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

<TextView
    android:id="@+id/textView_amount"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="TextView" />

<TextView
    android:id="@+id/textView_transactionType"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="TextView" />

<Button
    android:id="@+id/button_enterData"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="Enter Data" />

</LinearLayout>

下一个是文件enter_data.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

<TextView
    android:id="@+id/textView_enterNumber"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="Enter a number:" />

<EditText
    android:id="@+id/editText_number"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:ems="10"
    android:inputType="number" >
    <requestFocus />
 </EditText>

 <Button
    android:id="@+id/button_send"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:onClick="send"
    android:text="Send" />

 </LinearLayout>

最后,这是 list 文件。

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.customlistenertest"
android:versionCode="1"
android:versionName="1.0" >

<uses-sdk
    android:minSdkVersion="8"
    android:targetSdkVersion="19" />

<application 
   android:theme="@style/AppTheme" 
   android:allowBackup="true"
   android:icon="@drawable/ic_launcher"
   android:label="@string/app_name">
    <activity android:label="@string/app_name" android:name="MainActivity">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />
            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
    <activity android:name="com.example.customlistenertest.DataActivity" >    
    </activity>
</application>
</manifest>

最佳答案

MainActivity 中使用的“CustomerTransaction 事务”与 DataActivity 中定义的“CustomerTransaction 事务”不是同一个实例。这就解释了为什么 DataActivity 是空的,因为您只订阅了 MainActivity 中的 Activity 。该问题的两种可能的解决方案:

  1. 将“控件”设置为单例,这样您就可以确保一次只有一个它的实例。
  2. 将“listeners”变量和“register/unregister/notitifylisteners”方法设为静态。

我更喜欢选项 1,但您可以选择您认为最好的选项。

祝你好运!

关于java - 当我需要通知监听器时,为什么监听器集是空的? (在 Android 应用程序中),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24520573/

相关文章:

java - 是否可能出现这样的竞争条件等待线程完成任务?

java - 在 Controller 之前使用拦截器编辑对象的值

java - 关于servlet应用程序中连接池的一些问题

java - Android/Java 问题。这两个决策树有何不同?

java - Android:传递允许 recyclerview 由特定字符串填充的变量

android - 我怎么知道谁从市场上下载了我的应用程序?

java - 使用一种标识来处理多个监听器

Android numberpicker,永远不会从 OnValueChangedListener 断点传递

c# - 如何从java调用DLL?

java - 更新 JTable 数据——使用 AbstractTableModel