Android: возникают трудности при передаче объекта из одного действия в другое с использованием Parcel - PullRequest
1 голос
/ 25 октября 2011

Я прочитал МНОГО статей о том, как использовать Parcel, и я использовал его в своем приложении, но он не работает. Я только что создал действие, где вы можете подключиться к компьютеру, используя Sockets. Я открыл java.io.PrintWriter на сокете OutputStream и хочу, чтобы другое действие (которое открывается при нажатии кнопки в первом упражнении) использовало то же самое PrintWriter. Я думаю, что ошибка, которую я здесь делаю, заключается в том, что я не клонирую PrintWriter object. Я не знаю, как его клонировать.

Вот первое упражнение (я удалил несколько ненужных функций):

public class TestActivity extends Activity implements OnClickListener
{
    String ip;
    int port;
    Socket soc;
    PrintWriter writer;   // this works fine here

    @Override
    public void onCreate(Bundle savedInstanceState)
    {
        ip = "";
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        // Connect button click handler (you can skip this part)
        Button connect = (Button)findViewById(R.id.connectBTN);
        connect.setOnClickListener(new View.OnClickListener()
        {
            public void onClick(View v)
            {
                EditText TB = (EditText)findViewById(R.id.ipTB);
                ip = TB.getText().toString();
                TB = (EditText)findViewById(R.id.portTB);
                port = Integer.parseInt(TB.getText().toString());
                try {
                soc = new Socket(ip,port);
                showDialog("Connected successfully!");
                writer = new PrintWriter(soc.getOutputStream(), true);
                }
                catch(Exception er)
                {
                    showDialog("Couldn't connect.\n"+er.getMessage());
                    soc = null;
                }
            }
        } );

        findViewById(R.id.controlmouseBTN).setOnClickListener(this);

        // disconnect click handler  (skip this part too)
        Button disconnectBTN = (Button)findViewById(R.id.disconnectBTN);
        disconnectBTN.setOnClickListener(new View.OnClickListener()
        {
            public void onClick(View v)
            {
                try
                {
                    if(soc.isConnected()) soc.close();
                    writer.close();
                }
                catch(Exception er) { showDialog("Couldn't close connection.\n"+er.getMessage()); }
            }
        });

    }

    @Override
    public void onClick(View v)   // here is the erroneous part 
    {
        //if(soc == null || !soc.isConnected()) { showDialog("Not connected to any computer."); return; } 
        //if(writer == null) return;
        Intent i = new Intent(getApplicationContext(), PCController.class);
        Intermediate inter = new Intermediate();   // this is the Parcelable class I've created
        //inter.set(new String("hello"));  (I tried sending a string, which worked fine, just to confirm that my Parcelable class is working)
        inter.set(writer);
        i.putExtra("printer", inter);
        startActivity(i);
    }

}

Вот код класса Parcelable Intermediate

import android.os.*;

public class Intermediate implements Parcelable
{
    Object ob;

    public Intermediate()
    {}

    public Intermediate(Parcel parcel)
    {
        this.ob = parcel.readValue(Intermediate.class.getClassLoader());
    }

    public void set(Object o)
    { this.ob = o; }

    public Object get()
    {
        return ob;
    }

    @Override
    public int describeContents()
    { return 0; }

    @Override
    public void writeToParcel(Parcel dest, int flags)
    {
        dest.writeValue(this.ob);
    }

   // I don't understand this part though! Just copy pasted and modified the ClassName

    public static final Intermediate.Creator<Intermediate> CREATOR = new Intermediate.Creator<Intermediate>() {
        public Intermediate createFromParcel(Parcel in) {
            return new Intermediate(in); }

        public Intermediate[] newArray(int size) {
            return new Intermediate[size]; }
    };  
}

Вот код запускаемого следующего действия. (Я выложил только основную часть)

PrintWriter writer;
    String s;   // for testing purpose
    Swipe swipe;   // a SurfaceView

    public void onCreate(Bundle bundle)
    {
        super.onCreate(bundle);
        swipe = new Swipe(this);
        swipe.setOnTouchListener(this);
        setContentView(swipe);
        Intent i = getIntent();
        Intermediate ob = (Intermediate)i.getParcelableExtra("printer");
        writer = (PrintWriter)ob.get();
        //s = (String)ob.get();
        //showDialog(s);   // this shows up the string I had sent, which means the string is getting passed.
    }

Теперь, когда я не подключаюсь (т. Е. writer имеет значение NULL) и запускаю следующее действие, ничего не происходит (без ошибок). Но когда я подключаюсь (writer не равно нулю) и запускаю следующее действие, основное действие просто падает. Я не удосужился записать сообщение об ошибке, так как знаю, что здесь что-то действительно не так.

Любая помощь будет оценена! Спасибо!

Редактировать : Вот трассировка стека исключений

10-26 00:37:55.445: WARN/System.err(251): java.lang.RuntimeException: Parcel: unable to marshal value java.io.PrintWriter@44eb0b40
10-26 00:37:55.474: WARN/System.err(251):     at android.os.Parcel.writeValue(Parcel.java:1087)
10-26 00:37:55.484: WARN/System.err(251):     at android.test.Intermediate.writeToParcel(Intermediate.java:33)
10-26 00:37:55.484: WARN/System.err(251):     at android.os.Parcel.writeParcelable(Parcel.java:1106)
10-26 00:37:55.494: WARN/System.err(251):     at android.os.Parcel.writeValue(Parcel.java:1029)
10-26 00:37:55.504: WARN/System.err(251):     at android.os.Parcel.writeMapInternal(Parcel.java:469)
10-26 00:37:55.514: WARN/System.err(251):     at android.os.Bundle.writeToParcel(Bundle.java:1445)
10-26 00:37:55.524: WARN/System.err(251):     at android.os.Parcel.writeBundle(Parcel.java:483)
10-26 00:37:55.534: WARN/System.err(251):     at android.content.Intent.writeToParcel(Intent.java:5237)
10-26 00:37:55.544: WARN/System.err(251):     at android.app.ActivityManagerProxy.startActivity(ActivityManagerNative.java:1204)
10-26 00:37:55.554: WARN/System.err(251):     at android.app.Instrumentation.execStartActivity(Instrumentation.java:1373)
10-26 00:37:55.564: WARN/System.err(251):     at android.app.Activity.startActivityForResult(Activity.java:2749)
10-26 00:37:55.574: WARN/System.err(251):     at android.app.Activity.startActivity(Activity.java:2855)
10-26 00:37:55.584: WARN/System.err(251):     at android.test.TestActivity.onClick(TestActivity.java:80)
10-26 00:37:55.594: WARN/System.err(251):     at android.view.View.performClick(View.java:2364)
10-26 00:37:55.606: WARN/System.err(251):     at android.view.View.onTouchEvent(View.java:4179)
10-26 00:37:55.615: WARN/System.err(251):     at android.widget.TextView.onTouchEvent(TextView.java:6541)
10-26 00:37:55.624: WARN/System.err(251):     at android.view.View.dispatchTouchEvent(View.java:3709)
10-26 00:37:55.634: WARN/System.err(251):     at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:884)
10-26 00:37:55.634: WARN/System.err(251):     at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:884)
10-26 00:37:55.644: WARN/System.err(251):     at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:884)
10-26 00:37:55.656: WARN/System.err(251):     at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:884)
10-26 00:37:55.664: WARN/System.err(251):     at com.android.internal.policy.impl.PhoneWindow$DecorView.superDispatchTouchEvent(PhoneWindow.java:1659)
10-26 00:37:55.674: WARN/System.err(251):     at com.android.internal.policy.impl.PhoneWindow.superDispatchTouchEvent(PhoneWindow.java:1107)
10-26 00:37:55.684: WARN/System.err(251):     at android.app.Activity.dispatchTouchEvent(Activity.java:2061)
10-26 00:37:55.694: WARN/System.err(251):     at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchTouchEvent(PhoneWindow.java:1643)
10-26 00:37:55.705: WARN/System.err(251):     at android.view.ViewRoot.handleMessage(ViewRoot.java:1691)
10-26 00:37:55.714: WARN/System.err(251):     at android.os.Handler.dispatchMessage(Handler.java:99)
10-26 00:37:55.724: WARN/System.err(251):     at android.os.Looper.loop(Looper.java:123)
10-26 00:37:55.734: WARN/System.err(251):     at android.app.ActivityThread.main(ActivityThread.java:4363)
10-26 00:37:55.744: WARN/System.err(251):     at java.lang.reflect.Method.invokeNative(Native Method)
10-26 00:37:55.754: WARN/System.err(251):     at java.lang.reflect.Method.invoke(Method.java:521)
10-26 00:37:55.764: WARN/System.err(251):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:860)
10-26 00:37:55.774: WARN/System.err(251):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:618)
10-26 00:37:55.785: WARN/System.err(251):     at dalvik.system.NativeStart.main(Native Method)
10-26 00:38:02.424: WARN/System.err(251): java.lang.RuntimeException: Parcel: unable to marshal value java.io.PrintWriter@44eb0b40
10-26 00:38:02.454: WARN/System.err(251):     at android.os.Parcel.writeValue(Parcel.java:1087)
10-26 00:38:02.454: WARN/System.err(251):     at android.test.Intermediate.writeToParcel(Intermediate.java:33)
10-26 00:38:02.464: WARN/System.err(251):     at android.os.Parcel.writeParcelable(Parcel.java:1106)
10-26 00:38:02.474: WARN/System.err(251):     at android.os.Parcel.writeValue(Parcel.java:1029)
10-26 00:38:02.486: WARN/System.err(251):     at android.os.Parcel.writeMapInternal(Parcel.java:469)
10-26 00:38:02.494: WARN/System.err(251):     at android.os.Bundle.writeToParcel(Bundle.java:1445)
10-26 00:38:02.505: WARN/System.err(251):     at android.os.Parcel.writeBundle(Parcel.java:483)
10-26 00:38:02.514: WARN/System.err(251):     at android.content.Intent.writeToParcel(Intent.java:5237)
10-26 00:38:02.524: WARN/System.err(251):     at android.app.ActivityManagerProxy.startActivity(ActivityManagerNative.java:1204)
10-26 00:38:02.534: WARN/System.err(251):     at android.app.Instrumentation.execStartActivity(Instrumentation.java:1373)
10-26 00:38:02.544: WARN/System.err(251):     at android.app.Activity.startActivityForResult(Activity.java:2749)
10-26 00:38:02.554: WARN/System.err(251):     at android.app.Activity.startActivity(Activity.java:2855)
10-26 00:38:02.554: WARN/System.err(251):     at android.test.TestActivity.onClick(TestActivity.java:80)
10-26 00:38:02.564: WARN/System.err(251):     at android.view.View.performClick(View.java:2364)
10-26 00:38:02.574: WARN/System.err(251):     at android.view.View.onTouchEvent(View.java:4179)
10-26 00:38:02.584: WARN/System.err(251):     at android.widget.TextView.onTouchEvent(TextView.java:6541)
10-26 00:38:02.594: WARN/System.err(251):     at android.view.View.dispatchTouchEvent(View.java:3709)
10-26 00:38:02.604: WARN/System.err(251):     at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:884)
10-26 00:38:02.614: WARN/System.err(251):     at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:884)
10-26 00:38:02.625: WARN/System.err(251):     at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:884)
10-26 00:38:02.634: WARN/System.err(251):     at android.view.ViewGroup.dispatchTouchEvent(ViewGroup.java:884)
10-26 00:38:02.644: WARN/System.err(251):     at com.android.internal.policy.impl.PhoneWindow$DecorView.superDispatchTouchEvent(PhoneWindow.java:1659)
10-26 00:38:02.654: WARN/System.err(251):     at com.android.internal.policy.impl.PhoneWindow.superDispatchTouchEvent(PhoneWindow.java:1107)
10-26 00:38:02.664: WARN/System.err(251):     at android.app.Activity.dispatchTouchEvent(Activity.java:2061)
10-26 00:38:02.674: WARN/System.err(251):     at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchTouchEvent(PhoneWindow.java:1643)
10-26 00:38:02.684: WARN/System.err(251):     at android.view.ViewRoot.handleMessage(ViewRoot.java:1691)
10-26 00:38:02.694: WARN/System.err(251):     at android.os.Handler.dispatchMessage(Handler.java:99)
10-26 00:38:02.704: WARN/System.err(251):     at android.os.Looper.loop(Looper.java:123)
10-26 00:38:02.714: WARN/System.err(251):     at android.app.ActivityThread.main(ActivityThread.java:4363)
10-26 00:38:02.714: WARN/System.err(251):     at java.lang.reflect.Method.invokeNative(Native Method)
10-26 00:38:02.724: WARN/System.err(251):     at java.lang.reflect.Method.invoke(Method.java:521)
10-26 00:38:02.734: WARN/System.err(251):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:860)
10-26 00:38:02.744: WARN/System.err(251):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:618)
10-26 00:38:02.754: WARN/System.err(251):     at dalvik.system.NativeStart.main(Native Method)

Ответы [ 2 ]

3 голосов
/ 25 октября 2011

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

Посылка не является механизмом сериализации общего назначения. Этот класс (и соответствующий API Parcelable для размещения произвольных объектов в Parcel) разработан как высокопроизводительный транспорт IPC. В связи с этим нецелесообразно помещать какие-либо данные об участках в постоянное хранилище: изменения в базовой реализации каких-либо данных из участка могут сделать старые данные нечитаемыми.

Вместо этого вы должны использовать связки. Обычно я создаю два статических метода в классе, который я пытаюсь передать между действиями toBundle и fromBundle. ToBundle берет экземпляр класса, который я хочу транспортировать, и возвращает пакет. FromBunle принимает пакет и возвращает экземпляр класса.

Мне неизвестен способ передачи сложного объекта, такого как PrintWriter, между Activity. Но есть и другие проблемы с вашим кодом. Вы делаете сетевые вещи в основном потоке пользовательского интерфейса. Это заставит ваше приложение заморозить много. Вы должны переместить все свои операции ввода-вывода в другой поток. Это хорошо, потому что тогда вы можете просто переместить все свои вещи ввода-вывода в отдельный класс, который доступен из обоих видов деятельности. Например, используйте IntentService . Это сделает ваше приложение оооочень лучше. Вы также можете прочитать статью Безболезненная нить . Это очень важный документ, который я рекомендую всем разработчикам Android.

0 голосов
/ 25 октября 2011

Нельзя передавать PrintWriter между двумя действиями Android, используя любой из обычных методов передачи данных Android (Пакеты / Пакеты). Если вам нужно совместно использовать PrintWriter между двумя операциями, вы можете использовать статическую переменную.

При этом вы также можете написать сервис, к которому вы обращаетесь и используете для каждого занятия. Возможно, вы захотите взглянуть на пример LocalService в проекте ApiDemos для примера того, как это сделать. Есть также бесчисленные учебники онлайн по этому поводу.

...