简体   繁体   English

为什么我的viewPager崩溃了?

[英]Why is my viewPager crashing?

The view pager works fine when moving forward but if I move it back then forward or back, the app crashes. 向前移动时,视图分页器工作正常,但是如果我将其移回然后向前或向后移动,则应用程序将崩溃。

Here is the java file that deals ViewPager Adapter 这是处理ViewPager适配器的Java文件

    public class HalfScreenImageAdapter extends PagerAdapter {


    MemoryCache memoryCache = new MemoryCache();
    FileCache fileCache;
    private Map<ImageView, String> imageViews = Collections.synchronizedMap(new WeakHashMap<ImageView, String>());
    ExecutorService executorService;
    private Activity _activity;
    private String[] _imagePaths;
    private LayoutInflater inflater;

    // constructor
    public HalfScreenImageAdapter(Activity activity,
            String [] imagePaths, Context context) {
        fileCache = new FileCache(context);
        executorService = Executors.newFixedThreadPool(5);
        this._activity = activity;
        this._imagePaths = imagePaths;
    }

    int stub_id = R.drawable.ic_launcher;


    @Override
    public int getCount() {
        // TODO Auto-generated method stub
        return this._imagePaths.length;
    }

    @Override
    public boolean isViewFromObject(View view, Object object) {
        // TODO Auto-generated method stub
        return view == ((RelativeLayout) object);
    }

    @Override
    public Object instantiateItem(ViewGroup container, int position) {

        Context context = _activity;
        ImageView imgDisplay = new ImageView(context);








        inflater = (LayoutInflater) _activity
                .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        View viewLayout = inflater.inflate(R.layout.layout_full_screen_image, container,
                false);

        stub_id = R.drawable.hotel_overlay;
        imgDisplay = (ImageView) viewLayout.findViewById(R.id.imgDisplay);
        imageViews.put(imgDisplay, _imagePaths[position]);



        Bitmap bitmap = memoryCache.get(_imagePaths[position]);
        if(bitmap != null) {

            imgDisplay.setImageBitmap(bitmap);
            ((ViewPager) container).addView(viewLayout);
            //return imgDisplay;




        } else {

        queuePhoto(_imagePaths[position], imgDisplay);
        imgDisplay.setImageResource(R.drawable.hotel_overlay);

        }



        ((ViewPager) container).addView(viewLayout);

        return viewLayout;
    }


     @Override
        public void destroyItem(ViewGroup container, int position, Object object) {
            ((ViewPager) container).removeView((RelativeLayout) object);

     }


    private void queuePhoto(String url, ImageView imageView){

            PhotoToLoad p = new PhotoToLoad(url, imageView);
            executorService.submit(new PhotosLoader(p));

    } 


    private Bitmap getBitmap(String url){

            File f = fileCache.getFile(url);

            //from SD cache
            Bitmap b = decodeFile(f);
            if(b != null)
                return b;

            //from web
            try {

                Bitmap bitmap = null;
                URL imageUrl = new URL(url);
                HttpURLConnection conn = (HttpURLConnection)imageUrl.openConnection();
                conn.setConnectTimeout(30000);
                conn.setReadTimeout(30000);
                conn.setInstanceFollowRedirects(true);
                InputStream is = conn.getInputStream();
                OutputStream os = new FileOutputStream(f);
                Utils.CopyStream(is, os);
                os.close();
                bitmap = decodeFile(f);
                return bitmap;
            }catch (Exception ex){

                ex.printStackTrace();
                return null;

            }

    }

    //decodes image and scales it to reduce memory consumption
        private Bitmap decodeFile(File f){

            try{

                //decode image size
                BitmapFactory.Options o = new BitmapFactory.Options();
                o.inJustDecodeBounds = true;
                BitmapFactory.decodeStream(new FileInputStream(f), null, o);

                //Find the correct scale value. It should be the power of 2.
                final int REQUIRED_SIZE = 100;
                int width_tmp = o.outWidth, height_tmp = o.outHeight;
                int scale = 1;
                while(true){

                    if(width_tmp/2<REQUIRED_SIZE || height_tmp/2<REQUIRED_SIZE)
                        break;
                    width_tmp /= 2;
                    height_tmp /= 2;
                    scale *= 2;

                }

                //decode with inSampleSize
                BitmapFactory.Options o2 = new BitmapFactory.Options();
                o2.inSampleSize = scale;
                return BitmapFactory.decodeStream(new FileInputStream(f), null, o2);

            } catch (FileNotFoundException e) {}
            return null;
    }

    //Task for the queue
    private class PhotoToLoad   {

        public String url;
        public ImageView imageView;
        public PhotoToLoad(String u, ImageView i) {

            url = u;
            imageView = i;

        }

    }
    class PhotosLoader implements Runnable {

        PhotoToLoad photoToLoad;
        PhotosLoader(PhotoToLoad photoToLoad){

            this.photoToLoad = photoToLoad;

        }
        @Override
        public void run() {

            if(imageViewReused(photoToLoad))
                return;
            Bitmap bmp = getBitmap(photoToLoad.url);
            memoryCache.put(photoToLoad.url, bmp);
            if(imageViewReused(photoToLoad))
                return;
            BitmapDisplayer bd =  new BitmapDisplayer(bmp, photoToLoad);
            Activity a = (Activity)photoToLoad.imageView.getContext();
            a.runOnUiThread(bd);

        }

    }

    boolean imageViewReused(PhotoToLoad photoToLoad){

        String tag = imageViews.get(photoToLoad.imageView);
        if(tag==null || !tag.equals(photoToLoad.url))
            return true;
        return false;

    }

    //Used to display bitmap in the UI thread
    class BitmapDisplayer implements Runnable {

        Bitmap bitmap;
        PhotoToLoad photoToLoad;
        public BitmapDisplayer(Bitmap b, PhotoToLoad p){

            bitmap = b;
            photoToLoad = p;

        }

        public void run() {

            if(imageViewReused(photoToLoad))
                return;
            if(bitmap != null)
                photoToLoad.imageView.setImageBitmap(bitmap);
            else
                photoToLoad.imageView.setImageResource(stub_id);

        }

    }

    public void clearCache(){

        memoryCache.clear();
        fileCache.clear();

    }






}

The app crashes at this line. 应用程序在此行崩溃。

((ViewPager) container).addView(viewLayout);

return viewLayout;

here is the logcat 这是logcat

10-14 21:52:54.600: E/AndroidRuntime(1945): FATAL EXCEPTION: main 10-14 21:52:54.600: E/AndroidRuntime(1945): java.lang.IllegalStateException: The specified child already has a parent. 10-14 21:52:54.600:E / AndroidRuntime(1945):致命例外:主要10-14 21:52:54.600:E / AndroidRuntime(1945):java.lang.IllegalStateException:指定的孩子已经有一个父母。 You must call removeView() on the child's parent first. 您必须先在孩子的父母上调用removeView()。 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.view.ViewGroup.addViewInner(ViewGroup.java:3509) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.view.ViewGroup.addView(ViewGroup.java:3380) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.support.v4.view.ViewPager.addView(ViewPager.java:1304) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.view.ViewGroup.addView(ViewGroup.java:3325) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.view.ViewGroup.addView(ViewGroup.java:3301) 10-14 21:52:54.600: E/AndroidRuntime(1945): at com.example.bertin.HalfScreenImageAdapter.instantiateItem(HalfScreenImageAdapter.java:107) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.support.v4.view.ViewPager.addNewItem(ViewPager.java:832) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.support.v4.view.ViewPager.populate(ViewPager.java:1016) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.support.v4.view.ViewPager.populate(ViewPager.java:914) 10-14 21:52:54.600 10-14 21:52:54.600:E / AndroidRuntime(1945):在android.view.ViewGroup.addViewInner(ViewGroup.java:3509)10-14 21:52:54.600:E / AndroidRuntime(1945):在Android。 view.ViewGroup.addView(ViewGroup.java:3380)10-14 21:52:54.600:E / AndroidRuntime(1945):在android.support.v4.view.ViewPager.addView(ViewPager.java:1304)10-14 21:52:54.600:E / AndroidRuntime(1945):在android.view.ViewGroup.addView(ViewGroup.java:3325)10-14 21:52:54.600:E / AndroidRuntime(1945):在android.view.ViewGroup .addView(ViewGroup.java:3301)10-14 21:52:54.600:E / AndroidRuntime(1945):at com.example.bertin.HalfScreenImageAdapter.instantiateItem(HalfScreenImageAdapter.java:107)10-14 21:52:54.600 :E / AndroidRuntime(1945):位于android.support.v4.view.ViewPager.addNewItem(ViewPager.java:832)10-14 21:52:54.600:E / AndroidRuntime(1945):位于android.support.v4。 view.ViewPager.populate(ViewPager.java:1016)10-14 21:52:54.600:E / AndroidRuntime(1945):at android.support.v4.view.ViewPager.populate(ViewPager.java:914)10-14 21:52:54.600 : E/AndroidRuntime(1945): at android.support.v4.view.ViewPager$3.run(ViewPager.java:244) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.view.Choreographer$CallbackRecord.run(Choreographer.java:749) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.view.Choreographer.doCallbacks(Choreographer.java:562) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.view.Choreographer.doFrame(Choreographer.java:531) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:735) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.os.Handler.handleCallback(Handler.java:730) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.os.Handler.dispatchMessage(Handler.java:92) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.os.Looper.loop(Looper.java:137) 10-14 21:52:54.600: E/AndroidRuntime(1945): at android.app.ActivityThread.main(ActivityThread.java:5103) 10-14 21:52:54.600: E/AndroidRuntime(1945): at java.lang.ref :E / AndroidRuntime(1945):在android.support.v4.view.ViewPager $ 3.run(ViewPager.java:244)10-14 21:52:54.600:E / AndroidRuntime(1945):在android.view.Choreographer $ CallbackRecord.run(Choreographer.java:749)10-14 21:52:54.600:E / AndroidRuntime(1945):在android.view.Choreographer.doCallbacks(Choreographer.java:562)10-14 21:52:54.600 :E / AndroidRuntime(1945):位于android.view.Choreographer.doFrame(Choreographer.java:531)10-14 21:52:54.600:E / AndroidRuntime(1945):位于android.view.Choreographer $ FrameDisplayEventReceiver.run( Choreographer.java:735)10-14 21:52:54.600:E / AndroidRuntime(1945):在android.os.Handler.handleCallback(Handler.java:730)10-14 21:52:54.600:E / AndroidRuntime( 1945):位于android.os.Handler.dispatchMessage(Handler.java:92)10-14 21:52:54.600:E / AndroidRuntime(1945):位于android.os.Looper.loop(Looper.java:137)10 -14 21:52:54.600:E / AndroidRuntime(1945):在android.app.ActivityThread.main(ActivityThread.java:5103)10-14 21:52:54.600:E / AndroidRuntime(1945):在java.lang .REF lect.Method.invokeNative(Native Method) 10-14 21:52:54.600: E/AndroidRuntime(1945): at java.lang.reflect.Method.invoke(Method.java:525) 10-14 21:52:54.600: E/AndroidRuntime(1945): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:737) 10-14 21:52:54.600: E/AndroidRuntime(1945): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553) 10-14 21:52:54.600: E/AndroidRuntime(1945): at dalvik.system.NativeStart.main(Native Method) 10-14 21:52:56.370: I/SQLiteAssetHelper(2005): successfully opened database product_database 10-14 21:52:56.430: D/dalvikvm(2005): GC_FOR_ALLOC freed 59K, 9% free 2751K/3008K, paused 2ms, total 3ms 10-14 21:52:56.430: I/dalvikvm-heap(2005): Grow heap (frag case) to 3.950MB for 1127532-byte allocation 10-14 21:52:56.460: D/dalvikvm(2005): GC_FOR_ALLOC freed 1K, 7% free 3851K/4112K, paused 29ms, total 29ms 10-14 21:52:56.480: W/dalvikvm(2005): threadid=11: thread exiting with uncaught exception (group=0xb0f36648) 10-14 21:52:56.480: lect.Method.invokeNative(本机方法)10-14 21:52:54.600:E / AndroidRuntime(1945):at java.lang.reflect.Method.invoke(Method.java:525)10-14 21:52:54.600 :E / AndroidRuntime(1945):在com.android.internal.os.ZygoteInit $ MethodAndArgsCaller.run(ZygoteInit.java:737)10-14 21:52:54.600:E / AndroidRuntime(1945):在com.android。 internal.os.ZygoteInit.main(ZygoteInit.java:553)10-14 21:52:54.600:E / AndroidRuntime(1945):at dalvik.system.NativeStart.main(Native Method)10-14 21:52:56.370 :I / SQLiteAssetHelper(2005):成功打开数据库product_database 10-14 21:52:56.430:D / dalvikvm(2005):GC_FOR_ALLOC释放59K,释放9%的2751K / 3008K,暂停2ms,总计3ms 10-14 21:52 :56.430:I / dalvikvm-heap(2005):将堆(frag情况)增加到3.950MB以分配1127532字节分配10-14 21:52:56.460:D / dalvikvm(2005):GC_FOR_ALLOC释放了1K,7%释放了3851K / 4112K,暂停29ms,总计29ms 10-14 21:52:56.480:W / dalvikvm(2005):threadid = 11:线程退出且未捕获到异常(group = 0xb0f36648)10-14 21:52:56.480: E/AndroidRuntime(2005): FATAL EXCEPTION: ModernAsyncTask #1 10-14 21:52:56.480: E/AndroidRuntime(2005): java.lang.RuntimeException: An error occured while executing doInBackground() 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.support.v4.content.ModernAsyncTask$3.done(ModernAsyncTask.java:137) 10-14 21:52:56.480: E/AndroidRuntime(2005): at java.util.concurrent.FutureTask.finishCompletion(FutureTask.java:352) 10-14 21:52:56.480: E/AndroidRuntime(2005): at java.util.concurrent.FutureTask.setException(FutureTask.java:219) 10-14 21:52:56.480: E/AndroidRuntime(2005): at java.util.concurrent.FutureTask.run(FutureTask.java:239) 10-14 21:52:56.480: E/AndroidRuntime(2005): at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1080) 10-14 21:52:56.480: E/AndroidRuntime(2005): at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:573) 10-14 21:52:56.480: E/AndroidRuntime(2005): at java.lang.Thread.run(Thread.java:841) 10-14 21:52:56.480: E E / AndroidRuntime(2005):致命例外:ModernAsyncTask#1 10-14 21:52:56.480:E / AndroidRuntime(2005):java.lang.RuntimeException:执行doInBackground()10-14 21:52时发生错误: 56.480:E / AndroidRuntime(2005):在android.support.v4.content.ModernAsyncTask $ 3.done(ModernAsyncTask.java:137)10-14 21:52:56.480:E / AndroidRuntime(2005):在java.util。 parallel.FutureTask.finishCompletion(FutureTask.java:352)10-14 21:52:56.480:E / AndroidRuntime(2005):at java.util.concurrent.FutureTask.setException(FutureTask.java:219)10-14 21: 52:56.480:E / AndroidRuntime(2005):在java.util.concurrent.FutureTask.run(FutureTask.java:239)10-14 21:52:56.480:E / AndroidRuntime(2005):在java.util.concurrent .ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1080)10-14 21:52:56.480:E / AndroidRuntime(2005):at java.util.concurrent.ThreadPoolExecutor $ Worker.run(ThreadPoolExecutor.java:573)10-14 21 :52:56.480:E / AndroidRuntime(2005):at java.lang.Thread.run(Thread.java:841)10-14 21:52:56.480:E /AndroidRuntime(2005): Caused by: java.lang.IllegalArgumentException: the bind value at index 2 is null 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.database.sqlite.SQLiteProgram.bindString(SQLiteProgram.java:164) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.database.sqlite.SQLiteProgram.bindAllArgsAsStrings(SQLiteProgram.java:200) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.database.sqlite.SQLiteDirectCursorDriver.query(SQLiteDirectCursorDriver.java:47) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.database.sqlite.SQLiteDatabase.rawQueryWithFactory(SQLiteDatabase.java:1314) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.database.sqlite.SQLiteDatabase.queryWithFactory(SQLiteDatabase.java:1161) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.database.sqlite.SQLiteDatabase.query(SQLiteDatabase.java:1032) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.database.sqlite.SQLiteDatabase.query(SQLiteDatabase.java:1200) 10-14 21:52:5 / AndroidRuntime(2005):原因:java.lang.IllegalArgumentException:索引2处的绑定值为null 10-14 21:52:56.480:E / AndroidRuntime(2005):位于android.database.sqlite.SQLiteProgram.bindString( SQLiteProgram.java:164)10-14 21:52:56.480:E / AndroidRuntime(2005):位于android.database.sqlite.SQLiteProgram.bindAllArgsAsStrings(SQLiteProgram.java:200)10-14 21:52:56.480:E / AndroidRuntime(2005):位于android.database.sqlite.SQLiteDirectCursorDriver.query(SQLiteDirectCursorDriver.java:47)10-14 21:52:56.480:E / AndroidRuntime(2005):位于android.database.sqlite.SQLiteDatabase.rawQueryWithFactory(SQLiteDatabase .java:1314)10-14 21:52:56.480:E / AndroidRuntime(2005):位于android.database.sqlite.SQLiteDatabase.queryWithFactory(SQLiteDatabase.java:1161)10-14 21:52:56.480:E / AndroidRuntime (2005):位于android.database.sqlite.SQLiteDatabase.query(SQLiteDatabase.java:1032)10-14 21:52:56.480:E / AndroidRuntime(2005):位于android.database.sqlite.SQLiteDatabase.query(SQLiteDatabase。 java:1200)10-14 21:52:5 6.480: E/AndroidRuntime(2005): at com.example.bertin.database.ProductDB.getHotelLocations(ProductDB.java:46) 10-14 21:52:56.480: E/AndroidRuntime(2005): at com.example.bertin.Products.query(Products.java:112) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.content.ContentProvider.query(ContentProvider.java:744) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.content.ContentProvider$Transport.query(ContentProvider.java:199) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.content.ContentResolver.query(ContentResolver.java:414) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.content.ContentResolver.query(ContentResolver.java:357) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.support.v4.content.CursorLoader.loadInBackground(CursorLoader.java:49) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.support.v4.content.CursorLoader.loadInBackground(CursorLoader.java:35) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.support.v4.content.AsyncTaskLo 6.480:E / AndroidRuntime(2005):com.example.bertin.database.ProductDB.getHotelLocations(ProductDB.java:46)10-14 21:52:56.480:E / AndroidRuntime(2005):com.example.bertin .Products.query(Products.java:112)10-14 21:52:56.480:E / AndroidRuntime(2005):at android.content.ContentProvider.query(ContentProvider.java:744)10-14 21:52:56.480 :E / AndroidRuntime(2005):位于android.content.ContentProvider $ Transport.query(ContentProvider.java:199)10-14 21:52:56.480:E / AndroidRuntime(2005):位于android.content.ContentResolver.query( ContentResolver.java:414)10-14 21:52:56.480:E / AndroidRuntime(2005):在android.content.ContentResolver.query(ContentResolver.java:357)10-14 21:52:56.480:E / AndroidRuntime( 2005):位于android.support.v4.content.CursorLoader.loadInBackground(CursorLoader.java:49)10-14 21:52:56.480:E / AndroidRuntime(2005):位于android.support.v4.content.CursorLoader.loadInBackground (CursorLoader.java:35)10-14 21:52:56.480:E / AndroidRuntime(2005):位于android.support.v4.content.AsyncTaskLo ader.onLoadInBackground(AsyncTaskLoader.java:242) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.support.v4.content.AsyncTaskLoader$LoadTask.doInBackground(AsyncTaskLoader.java:51) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.support.v4.content.AsyncTaskLoader$LoadTask.doInBackground(AsyncTaskLoader.java:40) 10-14 21:52:56.480: E/AndroidRuntime(2005): at android.support.v4.content.ModernAsyncTask$2.call(ModernAsyncTask.java:123) 10-14 21:52:56.480: E/AndroidRuntime(2005): at java.util.concurrent.FutureTask.run(FutureTask.java:234) 10-14 21:52:56.480: E/AndroidRuntime(2005): ... 3 more ader.onLoadInBackground(AsyncTaskLoader.java:242)10-14 21:52:56.480:E / AndroidRuntime(2005):at android.support.v4.content.AsyncTaskLoader $ LoadTask.doInBackground(AsyncTaskLoader.java:51)10-14 21:52:56.480:E / AndroidRuntime(2005):位于android.support.v4.content.AsyncTaskLoader $ LoadTask.doInBackground(AsyncTaskLoader.java:40)10-14 21:52:56.480:E / AndroidRuntime(2005):在android.support.v4.content.ModernAsyncTask $ 2.call(ModernAsyncTask.java:123)10-14 21:52:56.480:E / AndroidRuntime(2005):在java.util.concurrent.FutureTask.run(FutureTask.java :234)10-14 21:52:56.480:E / AndroidRuntime(2005):... 3另外

10-14 21:52:54.600: E/AndroidRuntime(1945): FATAL EXCEPTION: main 10-14 21:52:54.600: E/AndroidRuntime(1945): java.lang.IllegalStateException: **The specified child already has a parent. You must call removeView() on the child's parent first.**

You have already added the root view in inflate 您已经在inflate添加了根视图

View viewLayout = inflater.inflate(R.layout.layout_full_screen_image, container, false); function. 功能。

You need not add the view again to the container . 您无需再次将视图添加到container

According to your crash log, you have to remove view first if it has a parent. 根据崩溃日志,如果视图具有父视图,则必须先将其删除。 so 所以

        View parent = viewLayout.getRootView();
        if (parent != null && parent instanceof ViewPager) {
            ((ViewPager) parent).removeView(viewLayout);
            ((ViewPager) container).addView(viewLayout);
        }

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM