欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

使用Android服务,实现报警管理器和广播接收器

程序员文章站 2022-06-24 20:53:37
介绍 几乎在所有平台上都有很多进程运行背景,它们被称为服务。可能在Android平台中有一些服务可以执行长时间运行的操作,这些操作在处理时不需要用户交互。 在本文中,借助预定义的Android警报服务,我们将创建一个应用程序,在所需的时间间隔内将电话模式更改为振动模式。除此之外,我们将编写自己的Se ......

介绍

几乎在所有平台上都有很多进程运行背景,它们被称为服务。可能在android平台中有一些服务可以执行长时间运行的操作,这些操作在处理时不需要用户交互。  

在本文中,借助预定义的android警报服务,我们将创建一个应用程序,在所需的时间间隔内将电话模式更改为振动模式。除此之外,我们将编写自己的service类并在特定时间调用它。此外,此演示应用程序将回答以下问题:

  • 如何使用alarm manager?
  • 如何通过alarm manager启动intent?
  • 如何使用broadcastreceiver?
  • 如何使用服务?
  • 如何在androidmanifest.xml中注册服务和接收器?
  • 如何更改手机铃声模式?

背景

要理解本文,读者应该了解java和android平台。

使用代码

在开始编码之前,应用程序的结构应该在编码器的脑海中清楚。对于此演示应用程序,我们可以按照以下简单步骤操作:

  1. 获取用户的时间间隔 mainactivity
  2. 根据时间间隔,设置闹钟以广播它
  3. 写入broadcastreceivers以接收警报并执行操作或呼叫服务。

在这个演示中,有4个类:

mainactivity             // main calss

fromhouralarmreceiver    //broadcastreceiver

tohouralarmreceiver      //broadcastreceiver

myservice                //service class

1-在mainactivity中获取用户的时间间隔

a)mainactivity.class

public class mainactivity extends activity {

 private edittext edittext1;    //create the objects
 private edittext edittext2;
 private button btn1;
 private int hourfrom;
 private int hourto;
             
       @override
       protected void oncreate(bundle savedinstancestate) {
          super.oncreate(savedinstancestate);
          setcontentview(r.layout.activity_main);
         
          edittext1 = (edittext) findviewbyid(r.id.edittext1); //bind the object
          edittext2 = (edittext) findviewbyid(r.id.edittext2);
          btn1 = (button) findviewbyid(r.id.btn1);
                      
          btn1.setonclicklistener(new onclicklistener() { //click listener for btn
                
                 @override
                 public void onclick(view v) {
          });
       }
 
       @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;
       }
 
}

b)main_activity.xml

<linearlayout xmlns:android="http://schemas.android.com/apk/res/android"

    xmlns:tools="http://schemas.android.com/tools"

    android:layout_width="match_parent"

    android:layout_height="match_parent"

    android:paddingbottom="@dimen/activity_vertical_margin"

    android:paddingleft="@dimen/activity_horizontal_margin"

    android:paddingright="@dimen/activity_horizontal_margin"

    android:paddingtop="@dimen/activity_vertical_margin"

    android:orientation="vertical"

    tools:context=".mainactivity" >
    <textview

        android:layout_width="wrap_content"

        android:layout_height="wrap_content"

        android:text="enter the desired time interval for to changed in vibrate mode" />
    <textview

        android:layout_width="wrap_content"

        android:layout_height="wrap_content"

        android:text="from (24 hour format)" />
    <edittext

        android:id="@+id/edittext1"

        android:layout_width="fill_parent"

        android:layout_height="wrap_content"

        android:numeric="integer"

        />
    <textview

        android:layout_width="wrap_content"

        android:layout_height="wrap_content"

        android:text="to (24 hour format)" />
    <edittext

        android:id="@+id/edittext2"

        android:layout_width="fill_parent"

        android:layout_height="wrap_content"

        android:numeric="integer"

        />
    <button

        android:id="@+id/btn1"

        android:layout_width="wrap_content"

        android:layout_height="wrap_content"

        android:text="set the service"/>
 
</linearlayout>

2 - 根据间隔设置警报管理器

在创建alarm manager之前,我们需要创建我们的意图来通过alarmmanager以下方式调用它们

intent intent1 = new intent(getbasecontext(), fromhouralarmreceiver.class);
final pendingintent sender1 = pendingintent.getbroadcast(this, 192837, intent1, pendingintent.flag_update_current);

intent intent2 = new intent(getbasecontext(), tohouralarmreceiver.class);
final pendingintent sender2 = pendingintent.getbroadcast(this, 192837, intent2, pendingintent.flag_update_current);

在这里,intent是要执行的操作的名称。由于我们将调用一个类,fromhouralarmreceiver该类需要知道发生了什么,为什么调用它,谁是调用者等等。因此我们需要通过intent对象[1]发送上下文。

这里有另一个术语pendingintent,这有两点重要。第一个表明我们写的意图将在稍后开始。第二个是通过使用pendingintent我们告诉我们正与3通信android平台研发 android平台的第三方应用程序或服务。在这个演示中,它是alarmmanager服务。

我们创建了两个意图,因为我们将设置2个警报,第一个将手机状态更改为振动模式,另一个将其更改为正常模式。所以我们需要2个日历对象来设置时间。

calendar cal1 = calendar.getinstance();
cal1.set(calendar.hour,hourfrom);
                                  
calendar cal2 = calendar.getinstance();
cal2.set(calendar.hour,hourto);

在我们拥有所有设置警报的对象和信息之后:

alarmmanager am = (alarmmanager) getsystemservice(alarm_service);am.set(alarmmanager.rtc_wakeup,
cal1.gettimeinmillis(), sender1);am.set(alarmmanager.rtc_wakeup,
cal2.gettimeinmillis(), sender2);

最后我们mainactivity将看起来像这样:

public class mainactivity extends activity {

 private edittext edittext1;
 private edittext edittext2;
 private button btn1;
 private int hourfrom;
 private int hourto;
             
   @override
   protected void oncreate(bundle savedinstancestate) {
      super.oncreate(savedinstancestate);
      setcontentview(r.layout.activity_main);
     
      edittext1 = (edittext) findviewbyid(r.id.edittext1);
      edittext2 = (edittext) findviewbyid(r.id.edittext2);
      btn1 = (button) findviewbyid(r.id.btn1);
     
       intent intent1 = new intent(getbasecontext(), fromhouralarmreceiver.class);
       final pendingintent sender1 = pendingintent.getbroadcast(
         this, 192837, intent1, pendingintent.flag_update_current);
       
       intent intent2 = new intent(getbasecontext(), tohouralarmreceiver.class);
       final pendingintent sender2 = pendingintent.getbroadcast(
         this, 192837, intent2, pendingintent.flag_update_current);
     
      btn1.setonclicklistener(new onclicklistener() {
            
             @override
             public void onclick(view v) {
                  
             try{
                   hourfrom = integer.parseint(edittext1.gettext().tostring());
                   hourto = integer.parseint(edittext2.gettext().tostring());
             } catch(exception e){}
                   if((0<hourfrom&&hourfrom<24)&&
                             (0<hourto&&hourto<24)){
                         
                       calendar cal1 = calendar.getinstance();
                       cal1.set(calendar.hour,hourfrom);
                       
                       calendar cal2 = calendar.getinstance();
                       cal2.set(calendar.hour,hourto);
                       
                       alarmmanager am = (alarmmanager) getsystemservice(alarm_service);
                       am.set(alarmmanager.rtc_wakeup, cal1.gettimeinmillis(), sender1);
                        am.set(alarmmanager.rtc_wakeup, cal2.gettimeinmillis(), sender2);

                       toast.maketext(getbasecontext(), 
                         "phone mode will be changed automatically !",toast.length_long).show();
                   }
                   else{
                      toast.maketext(getbasecontext(), 
                        "please enter hour in between 1-23 !",toast.length_long).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;
   }
}

3实现broadcastreceivers

平台中,几乎所有在设备中执行的动作都被广播。它可以想象成一个游泳池。无论操作是什么,都会将信息发送到该池。这样,您可以检查设备中发生的情况并根据它们执行操作。

在我们的演示中,alarmmanager将广播已发生警报并捕获此警报,我们需要编写broadcastreceiver如下所示:

public class fromhouralarmreceiver extends broadcastreceiver{
   @override
   public void onreceive(context context, intent intent) {
      }
}

可以用tie onreceive()方法执行期望的操作

注意:当a broadcastreceiver添加到项目时,需要将其注册到androidmanifest.xml中以下代码适用于此:

<receiverandroid:process=":remote" android:name="fromhouralarmreceiver"></receiver>
<receiverandroid:process=":remote" android:name="tohouralarmreceiver"></receiver>

所以我们有2个broadcastreceivers:

  1. fromhouralarmreceiver 负责将手机状态更改为振动模式。
  2. tohouralarmreceiver 负责将手机状态更改为正常模式。

fromhouralarmreceiver.class

public class fromhouralarmreceiver extends broadcastreceiver{

   @override
   public void onreceive(context context, intent intent) {
      audiomanager am= (audiomanager) context.getsystemservice(context.audio_service);
      am.setringermode(audiomanager.ringer_mode_vibrate);
      toast.maketext(context, "phone mode is changed to vibrate mode", toast.length_long).show();
   }
}

tohouralarmmanager.class

public class tohouralarmreceiver extends broadcastreceiver{

   @override
   public void onreceive(context context, intent intent) {
      audiomanager am= (audiomanager) context.getsystemservice(context.audio_service);
      am.setringermode(audiomanager.ringer_mode_normal);
      toast.maketext(context, "phone mode is changed to normal mode", toast.length_long).show();
      log.d("warnning", "something is happend...");
   }
}

注意2:当a收到警报提醒时broadcastreceiver,您可以调用自己的服务类,如下所示:

intent myserviceintent = new
intent(context,myservice.class);
context.startservice(myserviceintent);

注3:不要忘记将service类注册到androidmanifest.xml

<service class=".myservice" android:name="myservice">
 <intent-filter>
   <action android:value="com.javaorigin.android.sample.service.my_service"

           android:name=".myservice" />       
 </intent-filter>
</service>

myservice.class

public class myservice extends service{

    @override
      public void oncreate() {
          super.oncreate();    

    }

    @override
       public int onstartcommand(intent intent, int flags, int startid) {
           toast.maketext(getapplicationcontext(), 
            "*** i am called by broadcastreceiver ***", toast.length_long).show();
                 return startid;
    }
     @override
      public void ondestroy() {
          super.ondestroy();  
      }
   @override
   public ibinder onbind(intent arg0) {
          // todo auto-generated method stub
          return null;
   }
}

2.服务类的注册

众所周知,androidmanifest.xml负责所有权限,服务,意图等。所以我们编写的服务必须由androidmanifest.xml知道为此,应将以下代码添加到androidmanifes.xml

<service class=".myservice" android:name="myservice">
 <intent-filter>
   <action android:value="com.javaorigin.android.sample.service.my_service"

           android:name=".myservice" />       
   </intent-filter>
</service>

注意4:如果服务在运行时被调用,可能会崩溃。要防止出现此类错误,请检查方法的标志类型onstartcommand

节点5:大多数系统应用程序服务对用户不可见。如果你想编写一个不可见的服务,你应该使你的“apk”好像它是一个系统应用程序。