Android 学习笔记 Service服务与远程通信...(AIDL)

PS:这一章节看的我有几分迷茫,不是很容易理解...不过还好总算是明白了一大半了...基本的迷惑是解决了...

学习内容:

1.跨应用启动服务...

2.跨应用绑定服务...

3.跨应用实现通信...

由于5.0版本之前和5.0版本之后是有很大的区别的,因此我都会在这里进行具体的介绍...

1.跨应用启动服务...

  跨应用启动服务,其实就是多个应用程序之间产生一种沟通关系,应用程序之间可以进行通信或者是完成一些其他的互动,总而言之,就是在我本身的应用程序中去启动其他应用程序的某个服务,这就完成了跨应用启动服务...

2.跨应用绑定服务...

  跨应用绑定服务和跨应用启动服务基本类似,就是跨应用实现了绑定,调用的函数不一样而已...在这里我就不进行粘贴代码了,因为没什么必要,仅仅是跨应用启动和绑定服务是没有任何意义的,因为应用之间只有实现了通信,那么跨应用才有必要,否则我光绑定或者只是启动服务,而无法完成通信,那就一点意义都没有了...因此重点还是下面这点...

3.跨应用实现通信...

  跨应用实现通信,这个东西是非常重要的,应用程序之间,难免是需要通信的...这里完成通信一般都是使用绑定服务来完成,也就是说,都是使用bindService()来完成的..在这里呢Android机制为我们提供了一个AIDL服务,来方便我们完成跨进程通信,那么什么是AIDL呢,就是Android接口语言..它用来公开服务的接口,然后是其他的应用程序也可以访问这个应用程序内部的服务..是个非常猛的机制...那么如何去使用呢...

  在这里我建立两个应用程序,一个是服务端,一个是客户端...服务端用来提供数据,客户端用来获取数据...

i.首先我们先建立一个服务端...

  新建一个项目..exam_Cilent,包名为org.hnist.demo。。然后点击包文件,New->File,然后会弹出窗口,在最下方输入IPerson.aidl。。直接Finish就行了...然后编辑我们的aidl文件..

package org.hinst.demo;
interface IPerson{
    void setName(String name);
    void setAge(int age);
    void setEmail(String email);
    String display();
} 

在正常情况下在R.java文件下会自动生成一个IPerson.java文件...这个需要一点点时间...如果实在是没生成,那就Rebuild project...然后就搞定了...

  然后再建立一个IPersonImp.java文件,这个文件需要继承IPerson.Stub类,这里是核心部分...想必大家都明白,我们定义了一个接口,必然需要有一个类去实现这个接口..这个类才是真正能被客户端调用的东西..

package org.hinst.demo;

import android.os.RemoteException;

public class IPersonImpl extends IPerson.Stub{

    private String name="ClearLove";
    private int age=20;
    private String email="zxf@163.com";
    @Override
    public void setName(String name) throws RemoteException {
        // TODO Auto-generated method stub
        this.name=name;
    }

    @Override
    public void setAge(int age) throws RemoteException {
        // TODO Auto-generated method stub
        this.age=age;
    }

    @Override
    public void setEmail(String email) throws RemoteException {
        // TODO Auto-generated method stub
        this.email=email;
    }

    @Override
    public String display() throws RemoteException {
        // TODO Auto-generated method stub
        return "姓名:"+name+"年龄"+age+"邮箱"+email;
    }

}

服务端的MainActivity.java文件保持原样就行...什么东西都不加也可以...也可以自己定义一些代码调用服务..

  最后我们需要在AndroidManifest.xml文件中进行一些相应的配置...这个文件是必须要进行配置的,否则就会出现问题...只需要在</activity>和</application>之间加入下列代码就完成了...这个在5.0版本之前都是允许的,但是在5.0版本以后是不允许再这样做的...

<service android:name="MyRemoteService">
            <intent-filter>
                <action android:name="org.hinst.demo.MyRemoteService"/>
            </intent-filter>
</service>

 ii.这样服务端我们就配置完了...接下来就是客户端了...客户端相对就很简单....只是注意一点细节就可以了...我们再次建立另一个项目...包名为org.exam...然后再新建一个包..这个包用来存放...服务端的aidl文件...包名要和客户端保存的adil文件的包名相同,就是org.hnist.demo...然后再次新建一个aidl文件,输入的内容与服务端的一样...完成后,在客户端的R.java文件也会生成一个文件...这样客户端的aidl就配置好了...

然后就是客户端进行调用了...在MainActivity中进行调用...

package org.hnist.exam;

import org.hinst.demo.IPerson;

import com.example.exam6_7client.R;

import android.os.Bundle;
import android.os.IBinder;
import android.os.RemoteException;
import android.app.Activity;
import android.app.Service;
import android.content.ComponentName;
import android.content.Intent;
import android.content.ServiceConnection;
import android.view.Menu;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.TextView;

public class MainActivity extends Activity implements View.OnClickListener {

    private IPerson iPerson;
    private TextView textview;
    Intent intent=new Intent("org.hinst.demo.MyRemoteService");
    private ServiceConnection con=new ServiceConnection() {
        
        @Override
        public void onServiceDisconnected(ComponentName name) {
            // TODO Auto-generated method stub
            iPerson=null;
        }
        
        @Override
        synchronized public void onServiceConnected(ComponentName name, IBinder service) {
            // TODO Auto-generated method stub
            iPerson=IPerson.Stub.asInterface(service);
        }
    };
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        findViewById(R.id.start).setOnClickListener(this);
        bindService(intent, con, Service.BIND_AUTO_CREATE);
        textview=(TextView) findViewById(R.id.show);
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.main, menu);
        return true;
    }

    @Override
    public void onClick(View v) {
        // TODO Auto-generated method stub
        switch(v.getId()){
        case R.id.start:
            try {
                iPerson.setName("Clearlove");
                String msg=iPerson.display();
                MainActivity.this.textview.setText(msg);
            } catch (RemoteException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            
        }
    }
    protected void onDestroy(){
        unbindService(con);
        super.onDestroy();
    }

}

  布局文件我就不进行粘贴了....我们先启动服务端完成相应的aidl文件的配置,然后调用客户端,这样在触发按钮的时候,客户端就会输出服务端的信息了...这样也就完成了客户端与服务端的通信...

  在5.0版本以后呢,我们就不能使用配置AndroidManifest.xml文件,使用隐式的Intent已经无法完成多个应用程序之间的通信了...5.0版本之后..aidl文件的配置是一样的,没有任何的变化,唯一的变化就是需要使用显式的Intent,去进行传递...这个的原因我还真就不是很清楚...

  如何传递呢?就是这样...

serviceintent =new Intent();
serviceintent.setComponent(new ComponentName("com.example.myserviceapp","com.example.myserviceapp.AppMyService"));

  只要把Intent建立的对象制定一个主键值,这个主键值有两个参数,一个是我们要调用的服务的包名,一个是这个服务名....然后这样就完成了主键值的赋值....然后再次使用bindservice(serviceintent,this,Server.BIND_AUTO_CREATE)方法在进行传递...就完成了...最后放一个完整的代码,方便大家理解...这里使用了一个打包的机制..大家不用弄懂..关键是看服务端与客户端的实现...其实我上面的代码也是可以运行的...上面的更加的简单...

  这个跨进程通信不是很好理解,实际上并不难,只要理解了就会发现,其实没什么难的东西...关键还是看自己是否用心研究...

这个是服务端的代码...

http://files.cnblogs.com/files/RGogoing/AIDLService.rar

这个是客户端...

http://files.cnblogs.com/files/RGogoing/AIDLClient.rar