安卓黑马教学(八)

本文深入解析Android中的服务机制,包括进程概念、服务启动方式、电话窃听器实现、特殊广播接收者的使用、bindService方法的应用及混合方式开启服务等。同时介绍了AIDL的基本原理和应用场景。

1 进程概念的介绍

【1】安卓下四大组件都是运行在主线程中的,运行一个应用系统会开一个应用开一个linux进程和一个main线程。
【2】服务是在后台运行的,没有界面

进程的优先级:
【1】Foreground process  前台进程   优先级最高,相当于activity执行了onResume方法,用户正在交互
【2】Visible process 可视进程     一直影响用户看得见,相当于activity执行了onPause方法
【3】service  process   服务进程    通过startServer开启了一个服务
【4】background process  后台进程   相当于activity执行了onStop方法,界面不可见,但是activity并没有被销毁
【5】Empty process  空进程   不会维持任何组件运行
一般只会杀死第四和第五个进程。

2 start方式启动服务的特点

【1】定义一个类继承
【2】第一次点击开启服务的时候,服务执行onCreate方法和onStart方法
【3】第二次点击按钮,再次开启服务,服务执行onStrat方法
【4】服务一旦开启,服务就会在后台长期运行,就算应用退出了也服务也存在,直到用户手工停止。

补充:子线程也是在后台运行的。但是当用户退出了应用之后,这个应用就变成了空进程,但是子线程还是在运行当中的,不过因为空进程的优先级较低
子线程很容易就会被杀死。


3 电话窃听器

【1】smsManager  短信管理者,可以获取与短信相关的状态信息
【2】TelephoneManager  电话管理者,可以获取与电话状态相关的信息

实现步骤:
【1】定义一个服务,开启服务,记得在清单文件中配置
【2】在服务的onCreate方法中获取TelephoneManager   
TelephonyManager tm = (TelephonyManager) getSystemService(TELEPHONY_SERVICE);

【3】注册电话监听
tm.listen(new MyPhoneStateListener(), PhoneStateListener.LISTEN_CALL_STATE);
【4】定义一个类,用来监听电话的状态
private class MyPhoneStateListener extends PhoneStateListener{

		@Override
		public void onCallStateChanged(int state, String incomingNumber) {
			// TODO Auto-generated method stub
			super.onCallStateChanged(state, incomingNumber);
			switch (state){
			case TelephonyManager.CALL_STATE_IDLE://空闲状态,没来电
				
				break;
			case TelephonyManager.CALL_STATE_OFFHOOK://接听
				
				break;
			case TelephonyManager.CALL_STATE_RINGING://电话响铃
				
				break;
			}
			
			
			
			
			
		}
		
	}
	
【5录音功能
private class MyPhoneStateListener extends PhoneStateListener{
		MediaRecorder recorder;
		@Override
		public void onCallStateChanged(int state, String incomingNumber) {
			// TODO Auto-generated method stub
			super.onCallStateChanged(state, incomingNumber);
			switch (state){
			case TelephonyManager.CALL_STATE_IDLE://空闲状态,没来电
				if(recorder != null){
					recorder.stop();
					recorder.reset();
					recorder.release();
				}
				break;
			case TelephonyManager.CALL_STATE_OFFHOOK://接听
				
				break;
			case TelephonyManager.CALL_STATE_RINGING://电话响铃
				
				recorder = new MediaRecorder();
				recorder.setAudioChannels(MediaRecorder.AudioSource.MIC);
				recorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
				recorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
				recorder.setOutputFile("/mnt/sdcard//luyin.3gp");
				try {
					recorder.prepare();
				} catch (IllegalStateException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				} catch (IOException e) {
					// TODO Auto-generated catch block
					e.printStackTrace();
				}
				
				
				
				break;
			}
			
			
			
			
			
		}
		
	}

【6】把注册服务方到广播之中

public class BootReceiver extends BroadcastReceiver {

	@Override
	public void onReceive(Context arg0, Intent arg1) {
		// TODO Auto-generated method stub
		Intent intent1 = new Intent(arg0,PhoneService.class);
		arg0.startService(intent1);
	}

}

4 使用服务注册特殊的广播接收者

操作特别频繁地广播事件,比如屏幕的解锁和锁屏。
【1】定义广播接受者

public class ScreenReceiver extends BroadcastReceiver {

	@Override
	public void onReceive(Context arg0, Intent arg1) {
		// TODO Auto-generated method stub
		String action = arg1.getAction();
		if("android.intent.action.SCREEN_OFF".equals(action)){
			System.out.println("锁屏了");
		}else if("android.intent.action.SCREEN_ON".equals(action)){
			System.out.println("解锁了");
		}
	}

}
【2写一个服务,用来注册广播接受者

public class ScreenService extends Service {
	ScreenReceiver receiver;
	@Override
	public IBinder onBind(Intent arg0) {
		// TODO Auto-generated method stub
		return null;
	}
	@Override
	public void onCreate() {
		// TODO Auto-generated method stub
		super.onCreate();
		receiver = new ScreenReceiver();
		IntentFilter filter = new IntentFilter();
		filter.addAction("android.intent.action.SCREEN_OFF");
		filter.addAction("android.intent.action.SCREEN_ON");
		registerReceiver(receiver, filter);
		
	}
	@Override
	public void onDestroy() {
		// TODO Auto-generated method stub
		super.onDestroy();
		unregisterReceiver(receiver);
	}
}
【3在mainActivity的oncreate方法里开启服务
public class MainActivity extends Activity {
	
	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		Intent intent = new Intent(this,ScreenService.class);
		startService(intent);
		
		/*
		 *    <receiver android:name=".ScreenReceiver">
            <intent-filter >
                <action android:name="android.intent.action.SCREEN_OFF"/>
                <action android:name="android.intent.action.SCREEN_ON"/>
            </intent-filter>
        </receiver>
        以上是静态注册,现在要动态注册
		 */
		
	}

	@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
	protected void onDestroy() {
		// TODO Auto-generated method stub
		super.onDestroy();
		
	}
	
}

【4】在清单文件配置服务


5 通过bindService开启服务

通过startService方法来启动服务:先调用服务的onCreate方法,然后再调用onStartCommand方法
通过bindService方法来启动服务:先调用服务的onCreate方法,然后再调用onBind方法。而且,如果onBind方法的返回值是null,则不执行监视服务状态那个类的onServiceConnected方法。而且bindService只能绑定一次,也只能解绑一次。在Activity销毁的时候,服务也会被销毁。在这个时候,logcat会出现一段报错,注意意思是说
Activity  leaked。这个是因为在Activity销毁的时候没有解绑服务,只需要在Activity的onDestory方法中解绑服务即可。还有,通过此方法启动服务,在设置页面是找不到此服务的。通过startService方法启动服务可以找到。




6 为什么引入bindService?

因为要使用服务里边定义的方法。想要使用服务的方法,并不能直接new一个对象,因为此对象不在谷歌提供的上下文环境中,会出现空指针异常。




7 通过bindService方式调用服务方法里边的过程

【1】在服务的内部定义一个方法,让activity去调用

	public void banZheng(int money){
		if(money > 1000){
			Toast.makeText(getApplicationContext(), "我是领导亲戚,我帮你办了", 1).show();
		}else{
			Toast.makeText(getApplicationContext(), "这点钱还想办事?", 1).show();
		}
	}


【2】在服务的内部定义一个中间人对象(IBinder)

	public class MyBinder extends Binder{
		public void callBanZheng(int m){
			//内部类可以调外部类的方法
			banZheng(m);
		}
	}

【3】把定义的中间人对象在onBind方法中返回
	public IBinder onBind(Intent arg0) {
		// TODO Auto-generated method stub
		return new MyBinder();
	}
【4】在Activity的onCreate方法里调用bindServicce方法
		Intent intent = new Intent(this,BanZhengService.class);
		
		bindService(intent, conn, BIND_AUTO_CREATE);


【5】获取中间人对象

		public void onServiceConnected(ComponentName arg0, IBinder arg1) {
			// TODO Auto-generated method stub
			myBinder = (MyBinder) arg1;
			myBinder.callBanZheng(1000);
		}

【6】拿到中间人对象就可以间接地调用服务里边的方法了

【7】Activity销毁的时候做一个解绑方法。



8  通过接口方式调用服务里边的方法

接口可以隐藏代码内的细节,让程序员暴露自己只想暴露的方法。

【1】定义一个接口,把想暴露的方法都定义在接口里面。
【2】把我们定义的中间人对象实现我们自己定义的接口
【3】获取中间人对象
myBinder = (Iservice) arg1;


混合方式开启服务:既想让服务在后台长期运行,又想调用服务里边的方法。
谷歌提供的固定流程:
①先调用startService服务方法开启服务,保证服务在后台长期运行。
②调用bindService方法获取中间人对象。
③调用unbindService方法解绑。这个时候服务不会销毁。
④调动stopService。

9 百度音乐盒  



10 aidl介绍

本地服务:运行在自己应用里的服务
远程服务:运行在其他应用的服务
实现进程之间的通讯    IPC
aidl:安卓接口定义语言。专门用来解决进程间的通信,也就是应用间的通信。
总结,使用步骤:
【1】把Iservice.java文件变成一个aidl文件
【2】aidl这个语言不认识public,把这个标记符删去
【3】自动生成一个Iservice.java文件,系统自动帮我们生成一个Stub类


【4】我们自定义的中间人对象直接继承Stub
【5】保证两个应用的aidl文件是同一个,保证aidl文件所在的报名相同,把一个文件的包复制到另外一个文件中去。
【6】拿到中间人对象。
	private class MyConn implements ServiceConnection{
		
		public void onServiceConnected(ComponentName arg0, IBinder arg1) {
			// TODO Auto-generated method stub
			iservice =Stub.asInterface(arg1);
		}

		public void onServiceDisconnected(ComponentName arg0) {
			// TODO Auto-generated method stub
			
		}
		
	}


11  aidl应用的场景

支付宝:另外游戏需要充值,使用支付宝,调用另一个应用。
支付宝把支付的方法暴露出来给其他应用。








|--Activity不允许横竖屏切换 |--Activity常用小技巧 |--Activity按返回直接回到桌面 |--aidl之结合反射获取应用缓存大小等空间占用 |--aidl调用系统service未公开的方法挂电话 |--aidl调用系统未公开的方法代码示例2 |--android dp和px之间转换 |--android INSTALL_PARSE_FAILED_MANIFEST_MALFORMED |--android root下禁用组件 |--android 判断网络状态 |--android 对话框样式 |--android 开机启动 |--android 挪动dialog的位置 |--android 控制对话框位置 |--android 根据uri获取路径 |--android 模拟器错误 |--android 横竖屏切换 |--android 获取mac地址 |--android 获取sd卡状态 |--android 设置apn |--android 调节屏幕亮度 |--android 资源uri |--android 还原短信 |--android 重启 |--android中anim文件特效 |--app信息menifest获取(如版本号) |--AsyncQueryHandler之异步查询Cursor处理 |--AutoCompleteTextView自动提示的用法 |--BitMap、Drawable、inputStream及byte[] 互转 |--ContentProvider内容提供者定义 |--DatePicker日期控件 |--desktop |--Dialog之位置的挪动与控制 |--Dialog实现无标提栏及自定义风格 |--Dialog风格Activity的作法 |--ExpandableListView(下拉伸缩ListView) |--GridView表格布局的用法 |--httpclient超时 |--info体系 |--Intent启动应用apk安装 |--Intent常用功能 |--IO将输入流转成字节 |--Json读js资源文件 |--layout布局样式之style配置 |--listview 页面 图片加文字 |--ListView之CursorAdapter异步查询框架之短信 |--ListView之动态添加子view |--ListView优化之分页加载 |--ListView优化之动态加载 |--ListView优化之控制getView实现复杂显示 |--ListView优化之标准写法 |--listview老虎机 界面设计 水果机 |--listview页面跳转 数据库交互 事务 dao biz 层 |--Log的收集 |--Manager下的info |--Manager之ActivityManager进程管理 |--Manager之LocationManager |--Manager之PackageManager |--Menu之不同模式下显示不同菜单 |--openGL-ES上绘制文字 |--openGL-ES纹理贴图 |--openGL-ES获取帧率 |--openGL-ES雾化 |--PopupWindow的使用 |--PopupWindow的返回健关闭 |--RadioGroup的用法(里面的成员可以是任何view) |--SD卡之计算剩余空间 |--Spinner下拉菜单组件 |--SplashActivity |--StringUtils工具类的常用方法 |--TabHost一个界面显示多Activity |--TextView单行跑马灯效果 |--TextView虚拟获得焦点 |--uploadServlet |--uri之表示资源resource |--ViewPage的使用 |--view中的tag用法之存储对象 |--view常用属性 |--xml常用属性 |--xml文件的pull解析与序列化写入 |--xml的封装序列化 |--任务循环之只在Activity显示时执行 |--修改文件的最后修改时间 |--偏好设置(回显) |--内存优化之各种方法 |--内容提供者之短信的序列化对象读写 |--内容提供者之短信的获取与写入 |--内容提供者之联系人读写与批量操作 |--内容提供者之获取通话记录 |--内容提供者的定义 |--写入联系人信息 |--利用FinalHttp实现多线程断点续传 |--加密之MD5 |--动画Animation详解 |--动画之view左右抖动 |--动画之移动动画 |--动画之组合动画 |--动画之缩放动画ScaleAnimation |--反序列化对象 |--发送短信 读天气 调音量 |--回调函数的定义 |--图片之BitMap、Drawable、inputStream及byte[] 互转 |--图片之保存图片至SD卡 |--图片之删除40%最近没有被使用的 |--图片之的本地缓存至SD卡 |--图片之网络异步下载图片 |--图片之获取SD卡所有及边界可调及压缩和软引用和内存回收 |--图片的LRU算法内存保存和读取 |--图片的缩放处理(防内存溢出) |--多媒体应用设计图 |--多线程下载 |--多线程下载及断点续传 |--多线程之AsyncTask的用法 |--多线程之线程池ExecutorService |--字体为粗体 |--安卓下的多线程断点上传 |--对话框与进度条结合用法 |--屏幕之Activity全屏 |--屏幕之横竖屏切换 |--屏幕之调节屏幕亮度 |--屏幕相关之Display类获取屏幕尺寸和分辨率 |--屏幕适配之ScrollView |--屏幕适配之像素dp和px之间转换 |--工具类之Log的封装类 |--工具类之不同log打印的封装 |--工具类之开启新的Activity |--布局加载器的获取 |--广播接收者之开机启动 |--广播接收者之获取管理员权限 |--广播接收者代码注册与卸载 |--广播接收者常用广播的获取 |--应用之分享(隐式意图) |--应用之卸载(隐式意图) |--应用之安装(隐式意图) |--应用之完美退出 |--应用之获取名称和图标 |--应用之获得占用内存大小 |--应用之通过包名开启一个应用 |--应用之隐式意图开启设置界面 |--应用启动之检查版本更新及初始化 |--延时任务的工具类 |--异常之UncaughtExceptionHandler全局捕获处理 |--异步任务AsyncTask的用法 |--异步任务的自定义 |--快捷方式增删查 |--手势识别器GestureDetector的用法 |--拍照之调用系统相机并显示及保存 |--拨打电话 |--按健之长按menu事件屏蔽 |--按健监听按返回健回桌面 |--搜索之调用系统Searchable的用法 |--数据库CURD通过execSQL与rawQuery |--数据库SQLiteOpenHelper标准写法 |--数据库复杂多表查询 |--数据库查询之归属地 |--数据库直接CURD |--数据提交无需权限 |--文件之从服务器下载 |--文件之拷贝文件至某个目录 |--文件之指定编码读写文件 |--文件之释放Assets下的文件到应用的File目录 |--文件之随机存储RandomAccessFile |--文件使用时间排序 |--文件复制粘贴 |--文件路径之通过uri获取 |--时间java常用应用 |--时间之handle记时器 |--时间之time的用法得到特定时间的long值 |--时间之之定时任务TimerTask |--时间之倒计时CountDownTimer |--时间之当前时间动态显示 |--时间之自动任务ScheduledExecutorService |--时间之记时器 |--时间日期格式化 |--服务之判断是否处于运行状态 |--服务之定义录音机 |--服务之应用内绑定服务调用方法 |--服务之电话录音 |--服务之看门狗代码示例 |--格式化之DecimalFormat数字格式化 |--桌面快捷方式的添加代码示例 |--桌面控件widget的创建方法 |--模拟发短信 指定号码 短信窃听提示 |--消息机制 mesage looper |--滑动之左右滑动的两种定义 |--电话簿读取联系人信息 |--监听 |--监听之CheckBox是否选中监听 |--监听之EditText内容变化监听 |--监听之GridView条目点击监听 |--监听之ListView条目点击事件监听 |--监听之ListView滑动监听 |--监听之单击监听的两种定义 |--监听之双击监听 |--监听之电话状态监听 |--监听之触摸监听 |--短信之根据id删除及查询短信 |--短信发送小demo |--短信的截取 |--系统之SD卡清理 |--系统之获取所有开机启动应用 |--系统之重启实现 |--系统信息之获取SD卡内存信息 |--系统信息之获取动态内存RAM信息 |--系统信息之获取可用内存 |--系统信息之获得mac地址 |--系统信息之获得手机sim卡序列号 |--缓存优化之几种方案lastModified |--缓存优化之本地缓存优化(超过规定值或SD卡容量不够时) |--网络post提交查询请求 |--网络之HttpClient的get和post用法 |--网络之判断网络状态是否可用 |--网络之设置apn |--网络图片查看器 |--网络图片的下载与缓存 |--网络文件的下载与关联进度条 |--联系人之各种查找 |--联系人读写与批量操作 |--联系人读取 |--自定义shape形状颜色渐变资源 |--自定义Toast |--自定义view的属性 |--自定义下拉刷新ListView |--自定义动画资源 |--自定义常用小控件 |--自定义控件之小技巧 |--自定义控件风格和全局背景 |--自定义组合控件 |--自定义配置文件资源 |--自定义颜色状态选择资源 |--自定义颜色资源 |--获取网页的源码 |--读取外部资源的文件3种方式 |--读取手机内的短信 |--通信之Application实现应用全局通信 |--重启adb |--隐式转换 页面 |--震动效果的实现
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值