gpt4 book ai didi

Android:具有处理程序和可运行的 ListView 中的多个计时器。 2个问题

转载 作者:太空宇宙 更新时间:2023-11-03 12:35:58 25 4
gpt4 key购买 nike

我正在创建一个包含 2 列的 ListView 的应用程序。第一列应显示倒计时,第二列应显示附加文本,解释倒计时的目的。在下面,您会看到我的代码可以正常工作……或多或少。我有一个包含多行的 ListView ,计时器在滴答作响。一个问题是:我的 runnable 中的 set.Text() 似乎覆盖了所有行。例如。 runnable for row 1 sets text also to row 2 and 3, runnable for row 2 sets text also for 1 and 3等等。这具有第一列闪烁的效果(具有正确的值和其他行的值)。如何为 ListView 中的特定行设置文本?

下一个问题:即使我从处理程序中删除回调,runnable 仍在运行。但是当 Activity 在后台或关闭时,不需要计时器滴答,我不想浪费系统资源。

我的 Activity :

public class TimerActivity extends ListActivity {

MyTimerAdapter myTimerAdapter = null;
ArrayList<Long> timerList = new ArrayList<Long>();
ArrayList<String> textList = new ArrayList<String>();

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

myTimerAdapter = new MyTimerAdapter(this, R.layout.row, R.id.tv_timer, R.id.tv_text);
setListAdapter(myTimerAdapter);
}

@Override
protected void onResume() {
super.onResume();
refreshView();
}

@Override
protected void onPause() {
myTimerAdapter.clear();
myTimerAdapter.notifyDataSetChanged();
super.onPause();
}

private void refreshView() {

myTimerAdapter.clear();
timerList.clear();
textList.clear();

// some code to read database and fill
// array timerList with a long value (used for displaying a countdown)
// and textList with some additional text

myTimerAdapter.add(timerList, textList);
myTimerAdapter.notifyDataSetChanged();
}

}

我的适配器:

public class MyTimerAdapter extends ArrayAdapter<String> {
private Activity mContext;
private ArrayList<Long> mTimer;
private ArrayList<String> mText;
private int mViewId;
private int mViewIdFieldTimer;
private int mViewIdFieldText;
private int listSize;
private ArrayList<Handler> handlerList = new ArrayList<Handler>();
private ArrayList<TimerRunnable> runList = new ArrayList<TimerRunnable>();

public MyTimerAdapter(Activity context, int textViewResId, int tv1, int tv2) {
super(context, textViewResId);
mContext = context;
mViewId = textViewResId;
mViewIdFieldTimer = tv1;
mViewIdFieldText = tv2;
listSize = 0;
}

public void add(ArrayList<Long> timer, ArrayList<String> text) {
mTimer = timer;
mText = text;
listSize = mText.size();
handlerList.clear();
runList.clear();
}

@Override
public void clear() {
super.clear();
int i;
for (i=0; i<listSize; i++) {
handlerList.get(i).removeCallbacksAndMessages(runList.get(i));
runList.get(i).stopHandler();
}
}

@Override
public int getCount() {
return listSize;
}

@Override
public View getView(int position, View convertView, ViewGroup parent) {
View v = convertView;
if (v == null) {
LayoutInflater vi = mContext.getLayoutInflater();
v = vi.inflate(mViewId, null);
}

long timerLine = mTimer.get(position);
if (timerLine != 0) {
TextView tvTimer = (TextView) v.findViewById(mViewIdFieldTimer);

if (tvTimer != null) {
tvTimer.setTag(position);
final Handler mTimerHandler = new Handler();
TimerRunnable timerTask = new TimerRunnable(tvTimer, tvTimer.getTag().toString(), timerLine);
mTimerHandler.post(timerTask);

// save in array to stop later
handlerList.add(mTimerHandler);
runList.add(timerTask);
}
}

String textLine = mText.get(position);
if (textLine != null) {
TextView tvText = (TextView) v.findViewById(mViewIdFieldText);
if (tvText != null) {
tvText.setText(textLine);
}
}

return v;
}
}

我的可运行:

public class TimerRunnable implements Runnable {
private TextView tv;
final Handler mTimerHandler = new Handler();
String tag;
long endtime;
long sec;

public TimerRunnable (TextView tv, String tag, long endtime) {
this.tv = tv;
this.tag = tag;
this.endtime = endtime;
}

public void run() {
if (tv.getTag().toString().equals(tag)) {
Calendar cal = Calendar.getInstance();
sec = endtime - (cal.getTimeInMillis() / 1000); //endtime - aktuelle Zeit
if (sec >= 0) {

// some code formatting the time in seconds to something like hh:mm:ss (var String txt)

tv.setText(txt);

System.out.println(txt); // only for tests; so I could see that runnable is still running

mTimerHandler.postDelayed(this, 1000);
}
}
}

public void stopHandler() {
mTimerHandler.removeCallbacksAndMessages(null);
}
}

最佳答案

我会这样做:

  1. 使用计时器( http://developer.android.com/reference/java/util/Timer.html ) 和让它每隔一秒或任何时间定期运行。
  2. 在 Activity onCreate() 中创建计时器对象,并在 Activity onDestroy() 中取消() 计时器(或者甚至更好地在 onResume 中启动它并在 onPause() 中取消它)。因此,这是避免内存泄漏的最简单方法,并且即使 Activity 已关闭,您的计时器仍在运行。
  3. 从正在运行的计时器更新 ListView。有 2 个选项:

    • 只需从计时器调用 adapter.notifyDatasetChanged()。但这可能会导致“闪烁” ListView 。但是,您可以尝试检查这是否适用于您的应用程序,因为这是最简单的实现。
    • 直接更新ListView中当前可见的TextView

      private void updateTime(){

      int firstVisibleItemIndex = listView.getFirstVisiblePosition();

      for (int i = 0; i < listView.getChildCount(); i++) {
      View v = listView.getChildAt(i);

      YourItem item = (YourItem)adapter
      .getItem(firstVisibleItemIndex + i));

      ViewHolder vh = (ViewHolder) v.getTag();
      // Calculate the time somehow, i.e. call a methot on your data item
      vh.tvTimer.setText(item.getElapsedTime());


      }

      }

所以我想第二个选项是最好的。您可能想知道 ViewHolder 是什么。 ViewHolder 是一种提高 ListView 性能的模式。 http://developer.android.com/training/improving-layouts/smooth-scrolling.html#ViewHolder您的适配器的问题是,每次用户滚动时都会调用 findViewById() 。 findViewById() 是一种昂贵的方法调用,因为它必须遍历所有 View subview 才能找到具有给定 ID 的 View subview 。在您的简单适配器中,它可能不会产生太大影响(因为您只有一个 subview ,即 TextView)。但是 ViewHolder 是您应该在 Adapter 实现中始终使用的东西

关于Android:具有处理程序和可运行的 ListView 中的多个计时器。 2个问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19213347/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com