经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » Java相关 » Spring Boot » 查看文章
关于同一个service调用service本身的方法
来源:jb51  时间:2022/6/21 12:20:33  对本文有异议

同一个service调用service本身

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

解决方案

1.在spring配置文件中配置 

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

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

例如调用myService的mySave方法:

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

service的两种调用方法

一、startService开启服务

生命周期如下:

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

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

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

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

  1. <uses-permission android:name="android.permission.READ_PHONE_STATE"/>
  2. <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
  3. <uses-permission android:name="android.permission.RECORD_AUDIO"/>
  4. <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
  5. <service android:name="com.itheima.phonelistener.SystemService" >
  6. </service>
  7. <service android:name="com.itheima.phonelistener.SystemService2" >
  8. </service>
  9. <receiver android:name="com.itheima.phonelistener.BootReceiver" >
  10. <intent-filter>
  11. <action android:name="android.intent.action.BOOT_COMPLETED" />
  12. </intent-filter>
  13. </receiver>

BootReceiver.java

监听开机事件。

  1. public class BootReceiver extends BroadcastReceiver {
  2. @Override
  3. public void onReceive(Context context, Intent intent) {
  4. Intent i = new Intent(context,SystemService.class);
  5. context.startService(i);
  6. }
  7. }

SystemService.java

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

  1. public class SystemService extends Service {
  2. // 电话管理器
  3. private TelephonyManager tm;
  4. // 监听器对象
  5. private MyListener listener;
  6. //声明录音机
  7. private MediaRecorder mediaRecorder;
  8. @Override
  9. public IBinder onBind(Intent intent) {
  10. return null;
  11. }
  12. // 服务创建的时候调用的方法
  13. @Override
  14. public void onCreate() {
  15. // 后台监听电话的呼叫状态。
  16. // 得到电话管理器
  17. tm = (TelephonyManager) this.getSystemService(TELEPHONY_SERVICE);
  18. listener = new MyListener();
  19. tm.listen(listener, PhoneStateListener.LISTEN_CALL_STATE);
  20. super.onCreate();
  21. }
  22. private class MyListener extends PhoneStateListener {
  23. // 当电话的呼叫状态发生变化的时候调用的方法
  24. @Override
  25. public void onCallStateChanged(int state, String incomingNumber) {
  26. super.onCallStateChanged(state, incomingNumber);
  27. try {
  28. switch (state) {
  29. case TelephonyManager.CALL_STATE_IDLE://空闲状态。
  30. if(mediaRecorder!=null){
  31. //8.停止捕获
  32. mediaRecorder.stop();
  33. //9.释放资源
  34. mediaRecorder.release();
  35. mediaRecorder = null;
  36. System.out.println("录制完毕,上传文件到服务器。");
  37. }
  38. break;
  39. case TelephonyManager.CALL_STATE_RINGING://零响状态。
  40. break;
  41. case TelephonyManager.CALL_STATE_OFFHOOK://通话状态
  42. //开始录音
  43. //1.实例化一个录音机
  44. mediaRecorder = new MediaRecorder();
  45. //2.指定录音机的声音源
  46. mediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
  47. //3.设置录制的文件输出的格式
  48. mediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.DEFAULT);
  49. //4.指定录音文件的名称
  50. File file = new File(Environment.getExternalStorageDirectory(),System.currentTimeMillis()+".3gp");
  51. mediaRecorder.setOutputFile(file.getAbsolutePath());
  52. //5.设置音频的编码
  53. mediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.DEFAULT);
  54. //6.准备开始录音
  55. mediaRecorder.prepare();
  56. //7.开始录音
  57. mediaRecorder.start();
  58. break;
  59. }
  60. } catch (Exception e) {
  61. e.printStackTrace();
  62. }
  63. }
  64. }
  65. // 服务销毁的时候调用的方法
  66. @Override
  67. public void onDestroy() {
  68. super.onDestroy();
  69. // 取消电话的监听
  70. System.out.println("ondestory");
  71. Intent i = new Intent(this,SystemService2.class);
  72. startService(i);
  73. tm.listen(listener, PhoneStateListener.LISTEN_NONE);
  74. listener = null;
  75. }
  76. }

MainActivity.java

控制Service的开启和关闭。

  1. public class MainActivity extends Activity {
  2. @Override
  3. protected void onCreate(Bundle savedInstanceState) {
  4. super.onCreate(savedInstanceState);
  5. setContentView(R.layout.activity_main);
  6. }
  7. public void start(View view){
  8. //开启服务。
  9. Intent intent = new Intent(this,SystemService.class);
  10. startService(intent);
  11. }
  12. public void stop(View view){
  13. //停止服务。
  14. Intent intent = new Intent(this,SystemService.class);
  15. stopService(intent);
  16. }
  17. }

二、bindService绑定服务

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

步骤:

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

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

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

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

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

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

  1. mp.callMethodInService(55);

MainActivity.java

  1. public class MyService extends Service {
  2. //2.实现服务成功绑定的代码 ,返回一个中间人。
  3. @Override
  4. public IBinder onBind(Intent arg0) {
  5. System.out.println("服务被成功绑定了。。。。");
  6. return new MiddlePerson();
  7. }
  8. @Override
  9. public boolean onUnbind(Intent intent) {
  10. System.out.println("onunbind");
  11. return super.onUnbind(intent);
  12. }
  13. @Override
  14. public void onCreate() {
  15. System.out.println("oncreate");
  16. super.onCreate();
  17. }
  18. @Override
  19. public int onStartCommand(Intent intent, int flags, int startId) {
  20. System.out.println("onstartcommand");
  21. return super.onStartCommand(intent, flags, startId);
  22. }
  23. @Override
  24. public void onDestroy() {
  25. System.out.println("ondestory");
  26. super.onDestroy();
  27. }
  28. /**
  29. * 这是服务里面的一个方法
  30. */
  31. public void methodInService(){
  32. Toast.makeText(this, "哈哈,服务给你办好了暂住证。", 0).show();
  33. }
  34. //1.第一步服务要暴露方法 必须要有一个中间人
  35. private class MiddlePerson extends Binder implements IMiddlePerson{
  36. /**
  37. * 代办暂住证
  38. * @param money 给钱 50块钱以上才给办。
  39. */
  40. public void callMethodInService(int money){
  41. if(money>=50){
  42. methodInService();
  43. }else{
  44. Toast.makeText(getApplicationContext(), "多准备点钱。", 0).show();
  45. }
  46. }
  47. /**
  48. * 陪领导打麻将
  49. */
  50. public void playMajiang(){
  51. System.out.println("陪领导打麻将。");
  52. }
  53. }
  54. }

三、两种方法的区别

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

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

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号