在 ListView 的自定义适配器中从 URL 加载图像 (Android Studio)
Loading image from URL in custom adapter for ListView (Android Studio)
虽然位图似乎获取正确,但变量 'userBitmap' 将保持为空。但是,当在我的平板电脑上向上或向下滚动时,新的列表行将包含图片,但它们都是一样的而且是错误的。真的,真的很迷茫。我尝试了多种从网络获取图像的不同方法。非常感谢任何帮助。
我的自定义适配器:
public class MessagesArrayAdapter extends ArrayAdapter<ChatData>
{
Bitmap userBitmap;
public MessageArrayAdapter(Context context, List<ChatData> objects)
{
super(context, 0, objects);
}
public View getView(int position, View convertView, ViewGroup parent)
{
ChatCell chatCell = new ChatCell();
LayoutInflater inflater = LayoutInflater.from(getContext());
convertView = inflater.inflate(R.layout.cell_chat, parent, false);
chatCell.usernameTextView = (TextView) convertView.findViewById(R.id.usernameTextView);
chatCell.messageTextView = (TextView) convertView.findViewById(R.id.messageTextView);
chatCell.userImageView = (ImageView) convertView.findViewById(R.id.userImageView);
ChatData chatData = getItem(position);
// Get user image from web
new loadImageAsync().execute(chatData.avatarURL);
chatCell.userImageView.setImageBitmap(userBitmap);
chatCell.usernameTextView.setText(chatData.username);
chatCell.messageTextView.setText(chatData.message);
return convertView;
}
private static class ChatCell
{
TextView usernameTextView;
TextView messageTextView;
ImageView userImageView;
}
private class loadImageAsync extends AsyncTask<String, Void, Double>{
@Override
protected Double doInBackground(String... params) {
userBitmap = loadImage(params[0]);
return null;
}
}
public Bitmap loadImage(String str) {
InputStream instream = null;
try {
HttpGet httpRequest = new HttpGet(URI.create(str));
HttpClient httpclient = new DefaultHttpClient();
HttpResponse response = (HttpResponse) httpclient.execute(httpRequest);
HttpEntity entity = response.getEntity();
BufferedHttpEntity bufHttpEntity = new BufferedHttpEntity(entity);
instream = bufHttpEntity.getContent();
Bitmap myBitmap = BitmapFactory.decodeStream(instream);
return myBitmap;
} catch (Exception e) {
e.printStackTrace();
return null;
} finally {
//close input
if (instream != null) {
try {
instream.close();
} catch (IOException ioex) {
// Handle error
}
}
}
}
}
我认为您 运行 遇到了两个问题:
1) 视图回收 - 我自己是 Android 的新手,这在使用列表视图时经常出现,当部分列表视图移出屏幕时,向后滚动时它们可能是错误的。我个人遇到了一个问题,即在一行中上下滚动时,按钮状态会因错误的行而改变。 This tutorial 更详细地介绍了回收是什么,并在 View Holder private class 中提供了一个对我有用的解决方案。我相信 SO 的其他部分也谈到了这一点。
2) Image Null - 这肯定是空的,还是只是没有出现在列表视图中?如果是前者,则 LoadImage() 中的某些步骤一定无法正常工作。如果是后者,您可能想在设置图像后尝试调用 notifyDataSetChanged(),以告知主线程图像已准备好绘制。如下所示(在您的 AsyncTask 中):
@Override
protected void onPostExecute() {
notifyDataSetChanged();
}
}
解决问题的最简单方法是使用一些库,例如 Picasso。它不仅更易于使用并使您的代码更易于阅读,而且还可以防止您在图像太大时出现 OutOfMemory 异常。加载图像是一行的事情:
Picasso.with(context)
.load(urlOfYourImage)
.resize(50, 50) // here you resize your image to whatever width and height you like
.into(imageView)
虽然位图似乎获取正确,但变量 'userBitmap' 将保持为空。但是,当在我的平板电脑上向上或向下滚动时,新的列表行将包含图片,但它们都是一样的而且是错误的。真的,真的很迷茫。我尝试了多种从网络获取图像的不同方法。非常感谢任何帮助。
我的自定义适配器:
public class MessagesArrayAdapter extends ArrayAdapter<ChatData>
{
Bitmap userBitmap;
public MessageArrayAdapter(Context context, List<ChatData> objects)
{
super(context, 0, objects);
}
public View getView(int position, View convertView, ViewGroup parent)
{
ChatCell chatCell = new ChatCell();
LayoutInflater inflater = LayoutInflater.from(getContext());
convertView = inflater.inflate(R.layout.cell_chat, parent, false);
chatCell.usernameTextView = (TextView) convertView.findViewById(R.id.usernameTextView);
chatCell.messageTextView = (TextView) convertView.findViewById(R.id.messageTextView);
chatCell.userImageView = (ImageView) convertView.findViewById(R.id.userImageView);
ChatData chatData = getItem(position);
// Get user image from web
new loadImageAsync().execute(chatData.avatarURL);
chatCell.userImageView.setImageBitmap(userBitmap);
chatCell.usernameTextView.setText(chatData.username);
chatCell.messageTextView.setText(chatData.message);
return convertView;
}
private static class ChatCell
{
TextView usernameTextView;
TextView messageTextView;
ImageView userImageView;
}
private class loadImageAsync extends AsyncTask<String, Void, Double>{
@Override
protected Double doInBackground(String... params) {
userBitmap = loadImage(params[0]);
return null;
}
}
public Bitmap loadImage(String str) {
InputStream instream = null;
try {
HttpGet httpRequest = new HttpGet(URI.create(str));
HttpClient httpclient = new DefaultHttpClient();
HttpResponse response = (HttpResponse) httpclient.execute(httpRequest);
HttpEntity entity = response.getEntity();
BufferedHttpEntity bufHttpEntity = new BufferedHttpEntity(entity);
instream = bufHttpEntity.getContent();
Bitmap myBitmap = BitmapFactory.decodeStream(instream);
return myBitmap;
} catch (Exception e) {
e.printStackTrace();
return null;
} finally {
//close input
if (instream != null) {
try {
instream.close();
} catch (IOException ioex) {
// Handle error
}
}
}
}
}
我认为您 运行 遇到了两个问题:
1) 视图回收 - 我自己是 Android 的新手,这在使用列表视图时经常出现,当部分列表视图移出屏幕时,向后滚动时它们可能是错误的。我个人遇到了一个问题,即在一行中上下滚动时,按钮状态会因错误的行而改变。 This tutorial 更详细地介绍了回收是什么,并在 View Holder private class 中提供了一个对我有用的解决方案。我相信 SO 的其他部分也谈到了这一点。
2) Image Null - 这肯定是空的,还是只是没有出现在列表视图中?如果是前者,则 LoadImage() 中的某些步骤一定无法正常工作。如果是后者,您可能想在设置图像后尝试调用 notifyDataSetChanged(),以告知主线程图像已准备好绘制。如下所示(在您的 AsyncTask 中):
@Override
protected void onPostExecute() {
notifyDataSetChanged();
}
}
解决问题的最简单方法是使用一些库,例如 Picasso。它不仅更易于使用并使您的代码更易于阅读,而且还可以防止您在图像太大时出现 OutOfMemory 异常。加载图像是一行的事情:
Picasso.with(context)
.load(urlOfYourImage)
.resize(50, 50) // here you resize your image to whatever width and height you like
.into(imageView)