Nano Hash - криптовалюты, майнинг, программирование

Android NullPointerException CipherInputStream в растровое изображение

Я пытаюсь получить зашифрованные файлы из папки с ресурсами, расшифровать их в приложении и показать как содержимое в виде списка. Я использую хорошо известную реализацию Fedor Lazy List Loading. Как видите, я изменил его метод getBitmap на мой собственный метод:

package com.custom.lazylist;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.HashMap;
import java.util.Stack;
import javax.crypto.Cipher;
import javax.crypto.CipherInputStream;
import javax.crypto.spec.IvParameterSpec;
import javax.crypto.spec.SecretKeySpec;
import com.fedorvlasov.lazylist.R;
import android.app.Activity;
import android.content.Context;
import android.content.res.AssetManager;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.util.Log;
import android.widget.ImageView;

public class ImageLoader extends Activity {

    //the simplest in-memory cache implementation. This should be replaced with something like SoftReference or BitmapOptions.inPurgeable(since 1.6)
    private HashMap<String, Bitmap> cache=new HashMap<String, Bitmap>();

    private File cacheDir;

    public ImageLoader(Context context){
        //Make the background thead low priority. This way it will not affect the UI performance
        photoLoaderThread.setPriority(Thread.NORM_PRIORITY-1);

        //Find the dir to save cached images
        if (android.os.Environment.getExternalStorageState().equals(android.os.Environment.MEDIA_MOUNTED))
            cacheDir=new File(android.os.Environment.getExternalStorageDirectory(),"LazyList");
        else
            cacheDir=context.getCacheDir();
        if(!cacheDir.exists())
            cacheDir.mkdirs();
    }

    final int stub_id=R.drawable.stub;
    public void DisplayImage(String url, Activity activity, ImageView imageView)
    {
        if(cache.containsKey(url))
            imageView.setImageBitmap(cache.get(url));
        else
        {
            queuePhoto(url, activity, imageView);
            imageView.setImageResource(stub_id);
        }    
    }

    private void queuePhoto(String url, Activity activity, ImageView imageView)
    {
        //This ImageView may be used for other images before. So there may be some old tasks in the queue. We need to discard them. 
        photosQueue.Clean(imageView);
        PhotoToLoad p=new PhotoToLoad(url, imageView);
        synchronized(photosQueue.photosToLoad){
            photosQueue.photosToLoad.push(p);
            photosQueue.photosToLoad.notifyAll();
        }

        //start thread if it's not started yet
        if(photoLoaderThread.getState()==Thread.State.NEW)
            photoLoaderThread.start();
    }


    //MY METHOD
private Bitmap getBitmap(String src) {
    Bitmap myBitmap = null;
        //Decryption
        try {
        Cipher cipher = Cipher.getInstance("AES/CBC/NoPadding");
        SecretKeySpec keySpec = new SecretKeySpec("01234567890abcde".getBytes(), "AES");
        IvParameterSpec ivSpec = new IvParameterSpec("fedcba9876543210".getBytes());
        cipher.init(Cipher.DECRYPT_MODE, keySpec, ivSpec);

        AssetManager is = this.getAssets();        
        InputStream input = is.open(src); //open file in asset manager
        CipherInputStream cis = new CipherInputStream(input, cipher);

        myBitmap = BitmapFactory.decodeStream(cis);

        }
        catch(Exception e){
            e.printStackTrace();
            Log.v("ERROR","Error : "+e);
        }


        return myBitmap;
    }



    //Task for the queue
    private class PhotoToLoad
    {
        public String url;
        public ImageView imageView;
        public PhotoToLoad(String u, ImageView i){
            url=u; 
            imageView=i;
        }
    }

    PhotosQueue photosQueue=new PhotosQueue();

    public void stopThread()
    {
        photoLoaderThread.interrupt();
    }

    //stores list of photos to download
    class PhotosQueue
    {
        private Stack<PhotoToLoad> photosToLoad=new Stack<PhotoToLoad>();

        //removes all instances of this ImageView
        public void Clean(ImageView image)
        {
            for(int j=0 ;j<photosToLoad.size();){
                if(photosToLoad.get(j).imageView==image)
                    photosToLoad.remove(j);
                else
                    ++j;
            }
        }
    }

    class PhotosLoader extends Thread {
        public void run() {
            try {
                while(true)
                {
                    //thread waits until there are any images to load in the queue
                    if(photosQueue.photosToLoad.size()==0)
                        synchronized(photosQueue.photosToLoad){
                            photosQueue.photosToLoad.wait();
                        }
                    if(photosQueue.photosToLoad.size()!=0)
                    {
                        PhotoToLoad photoToLoad;
                        synchronized(photosQueue.photosToLoad){
                            photoToLoad=photosQueue.photosToLoad.pop();
                        }
                        Bitmap bmp=getBitmap(photoToLoad.url);
                        cache.put(photoToLoad.url, bmp);
                        Object tag=photoToLoad.imageView.getTag();
                        if(tag!=null && ((String)tag).equals(photoToLoad.url)){
                            BitmapDisplayer bd=new BitmapDisplayer(bmp, photoToLoad.imageView);
                            Activity a=(Activity)photoToLoad.imageView.getContext();
                            a.runOnUiThread(bd);
                        }
                    }
                    if(Thread.interrupted())
                        break;
                }
            } catch (InterruptedException e) {
                //allow thread to exit
            }
        }
    }

    PhotosLoader photoLoaderThread=new PhotosLoader();

    //Used to display bitmap in the UI thread
    class BitmapDisplayer implements Runnable
    {
        Bitmap bitmap;
        ImageView imageView;
        public BitmapDisplayer(Bitmap b, ImageView i){bitmap=b;imageView=i;}
        public void run()
        {
            if(bitmap!=null)
                imageView.setImageBitmap(bitmap);
            else
                imageView.setImageResource(stub_id);
        }
    }

    public void clearCache() {
        //clear memory cache
        cache.clear();

        //clear SD cache
        File[] files=cacheDir.listFiles();
        for(File f:files)
            f.delete();
    }  

}

..... но проблема в том, что я получаю это исключение:

08-11 11:11:45.293: WARN/System.err(1073): java.lang.NullPointerException
08-11 11:11:45.293: WARN/System.err(1073):     at android.content.ContextWrapper.getAssets(ContextWrapper.java:74)
08-11 11:11:45.293: WARN/System.err(1073):     at com.custom.lazylist.ImageLoader.getBitmap(ImageLoader.java:79)
08-11 11:11:45.293: WARN/System.err(1073):     at com.custom.lazylist.ImageLoader.access$0(ImageLoader.java:70)
08-11 11:11:45.293: WARN/System.err(1073):     at com.custom.lazylist.ImageLoader$PhotosLoader.run(ImageLoader.java:200)

вот как я использую LazyAdapter в своем основном классе:

 public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        list=(ListView)findViewById(R.id.list);
        adapter=new LazyAdapter(this, mStrings);
        list.setAdapter(adapter);

        list.setOnItemClickListener(new OnItemClickListener() {
            public void onItemClick(AdapterView<?> a, View v, int position, long id) 
            {
                Intent i = new Intent(v.getContext(), Cards.class);
                startActivity(i);

            }
        });

        Button btn = (Button) findViewById(R.id.btn);
        btn.setOnClickListener(listener);

/*          adapter.imageLoader.clearCache();
            adapter.notifyDataSetChanged();*/

        }

    @Override
    public void onDestroy()
    {
        adapter.imageLoader.stopThread();
        list.setAdapter(null);
        super.onDestroy();
    }



    public OnClickListener listener=new OnClickListener(){
        @Override
        public void onClick(View arg0) {
            adapter.imageLoader.clearCache();
            adapter.notifyDataSetChanged();
        }
    };

    private String[] mStrings={
            "card1_encrypted.jpg",
            "card1_encrypted.jpg",
            "card2_encrypted.jpg",
            "card1_encrypted.jpg",
            "card3_encrypted.jpg",
            "card1_encrypted.jpg",
            "card1_encrypted.jpg",
            "card3_encrypted.jpg",
            "card3_encrypted.jpg",
            "card2_encrypted.jpg",
            "card4_encrypted.jpg",
            "card5_encrypted.jpg",
            "card3_encrypted.jpg",
            "card2_encrypted.jpg",
            "card5_encrypted.jpg",
            "card1_encrypted.jpg"

    };

Итак, есть идеи, как это исправить? И последнее: в строке 79 у меня есть это:

AssetManager is = this.getAssets();   

Заранее спасибо!

11.08.2011

  • Где находится метод onCreate()? Ваш класс расширяет Activity, но ваш конструктор не превосходит Activity onCreate(). 11.08.2011
  • Нет основного класса. Вам нужно что-то вроде этого: @Override public void onCreate(Bundle saveInstanceState) { super.onCreate(savedInstanceState); 11.08.2011
  • Черт, я только что это понял... но когда я удаляю расширяет активность, я получаю сообщение об ошибке: метод getAssets() не определен для типа ImageLoader 11.08.2011
  • Тогда не удаляй. Вам нужно удалить свой конструктор. 11.08.2011
  • Если я удалю свой конструктор, у меня появятся ошибки в других файлах. Потому что в основном я использую этот класс ImageLoader только для загрузки изображения в классе Lazy Adapter, где я устанавливаю данные для просмотра списка. Мне просто нужно знать, как правильно получить файлы из активов и как вернуть расшифрованное изображение. 11.08.2011
  • Тогда не расширяйте Activity. В вашем конструкторе вам нужен аргумент AssetManager. Когда этот класс вызывается из действия, вам нужно использовать экземпляр AssetManager этого действия в качестве аргумента. 11.08.2011

Ответы:


1

Если первый класс на самом деле не является действием, не расширяйте действие только для использования this.getAssets(), передайте контекст (т.е. ваше действие) при создании экземпляра класса и используйте этот экземпляр для вызова getAssets()

11.08.2011
  • Я вообще не понял, что вы имеете в виду. Да, класс ImageLoader не является действием. Думаю, проблема заключается в вызове this.getAssets(); Я попытался изменить метод getBitmap, например getBitmap(Context contect,String src), и изменить AssetManager на context.getAssets(); но он все еще не работал 11.08.2011
  • Избавьтесь от расширения Activity. и поместите AssetManager = context.getAssets() 11.08.2011
  • Я сделал это, и у меня все еще есть исключение + еще 2 исключения о cipherinputstream и bufferedinputstream. 11.08.2011

  • 2

    Вам нужен метод onCreate().

      @Override
    
        public void onCreate(Bundle savedInstanceState) {
    
            super.onCreate(savedInstanceState);
    
            AssetManager am = getAssets();//this should not get a nullpointerexception
      }
    

    Или сделайте следующее:

    В вашей деятельности (в зависимости от того, что вызывает загрузчик изображений) напишите следующее:

    ImageLoader il = new ImageLoader(this);//this being the context.
    

    Затем в вашем классе ImageLoader вы можете использовать

    AssetManager am = context.getAssets();//context being the activity which called this class.
    
    11.08.2011
  • Продолжали ли вы расширять активность? Вы избавились от своего конструктора? Как вы называете этот класс? По умыслу? 11.08.2011
  • Я называю это так: imageLoader=new ImageLoader(activity.getApplicationContext()); 11.08.2011
  • В загрузчике изображений используйте AssetManager is = context.getAssets(); 11.08.2011
  • Новые материалы

    Кластеризация: более глубокий взгляд
    Кластеризация — это метод обучения без учителя, в котором мы пытаемся найти группы в наборе данных на основе некоторых известных или неизвестных свойств, которые могут существовать. Независимо от..

    Как написать эффективное резюме
    Предложения по дизайну и макету, чтобы представить себя профессионально Вам не позвонили на собеседование после того, как вы несколько раз подали заявку на работу своей мечты? У вас может..

    Частный метод Python: улучшение инкапсуляции и безопасности
    Введение Python — универсальный и мощный язык программирования, известный своей простотой и удобством использования. Одной из ключевых особенностей, отличающих Python от других языков, является..

    Как я автоматизирую тестирование с помощью Jest
    Шутка для победы, когда дело касается автоматизации тестирования Одной очень важной частью разработки программного обеспечения является автоматизация тестирования, поскольку она создает..

    Работа с векторными символическими архитектурами, часть 4 (искусственный интеллект)
    Hyperseed: неконтролируемое обучение с векторными символическими архитектурами (arXiv) Автор: Евгений Осипов , Сачин Кахавала , Диланта Хапутантри , Тимал Кемпития , Дасвин Де Сильва ,..

    Понимание расстояния Вассерштейна: мощная метрика в машинном обучении
    В обширной области машинного обучения часто возникает необходимость сравнивать и измерять различия между распределениями вероятностей. Традиционные метрики расстояния, такие как евклидово..

    Обеспечение масштабируемости LLM: облачный анализ с помощью AWS Fargate и Copilot
    В динамичной области искусственного интеллекта все большее распространение получают модели больших языков (LLM). Они жизненно важны для различных приложений, таких как интеллектуальные..