Android基础-Service和IntentService知识点详细总结

2023-11-10

Service 对于广大的Android开发者来说算是耳熟能详了,作为Android的四大组件之一,在我们的开发中也起着重要的作用,在Android面试中,Service相关的问题也是面试官问得比较多的,当别人问你,Service 到底是什么的时候?你可能随口就能答得上来,Service是一个在后台执行长时间运行操作而不用提供用户界面的应用组件,可由其他组件启动,即使用户切换到其他应用程序,Service 仍然在后台继续运行。没错,这是Service的概念,作为Android开发,或多或少都知道一些,但是不是每个人把所有知识点都了解得透测。因此,本篇文章对Service的用法做一个总结。

目录图

Service

Service 和Activity 一样同为Android 的四大组件之一,并且他们都有各自的生命周期,要想掌握Service 的用法,那就要了解Service 的生命周期有哪些方法,并且生命周期中各个方法回调的时机和作用。有一点比较重要,Service 有两种启动方式,并且它的两种启动方式的生命周期是不一样的。接下来分别看一下两种启动方式各自的生命周期方法。

startService方式启动Service

当应用组件通过startService方法来启动Service 时,Service 则会处于启动状态,一旦服务启动,它就会在后台无限期的运行,生命周期独立于启动它的组件,即使启动它的组件已经销毁了也不受任何影响,由于启动的服务长期运行在后台,这会大量消耗手机的电量,因此,我们应该在任务执行完成之后调用stopSelf()来停止服务,或者通过其他应用组件调用stopService 来停止服务。

startService 启动服务后,会执行如下生命周期:onCreate() -> onStartCommand() -> onStart()(现在已经废弃) -> onDestroy() 。具体看一下它的几个生命周期方法:

  • onCreate() :首次启动服务的时候,系统会调用这个方法,在onStartCommand 和 onBind 方法之前,如果服务已经启动起来了,再次启动时,则不会调用此方法,因此可以在onCreate 方法中做一些初始化的操作,比如要执行耗时的操作,可以在这里创建线程,要播放音乐,可以在这里初始化音乐播放器。

  • onStartCommand(): 当通过startService 方法来启动服务的时候,在onCreate 方法之后就会回调这个方法,此方法调用后,服务就启动起来了,将会在后台无限期的运行,直到通过stopService 或者 stopSelf 方法来停止服务。

  • onDestroy():当服务不再使用且将被销毁时,系统将调用此方法。服务应该实现此方法来清理所有资源,如线程、注册的侦听器、接收器等。 这是服务接收的最后一个调用。

了解了这几个生命周期方法后,我们就来写一个简单Service 。

要使用Service 就要通过继承Service类(或者继承IntentService ,后文会讲)来实现,代码如下:

public class SimpleService extends Service {
    public static final String TAG = "SimpleService";
    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        Log.i(TAG,"call onBind...");
        return null;
    }

    @Override
    public void onCreate() {
        Log.i(TAG,"call onCreate...");
    }

    @Override
    public void onStart(Intent intent, int startId) {
        Log.i(TAG,"call onStart...");
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        Log.i(TAG,"call onStartCommand...");
        return super.onStartCommand(intent, flags, startId);
    }

    @Override
    public void onDestroy() {
        Log.i(TAG,"call onDestroy...");
    }



}

Service类写好了之后,我们需要在清单文件中注册一下,在application标签下:

     <service android:name=".SimpleService"
            android:exported="false"
            />

写好了Service并且在清单文件注册之后,我们就可以启动Service了,启动Service和启动Activity 差不多,通过Intent 来启动,代码如下:

      findViewById(R.id.start_service).setOnClickListener(v -> {
            Intent intent = new Intent(this,SimpleService.class);
            // 启动服务
            startService(intent);
        });
        findViewById(R.id.stop_service).setOnClickListener(v -> {
            Intent service = new Intent(this,SimpleService.class);
            // 停止服务
            stopService(service);
        });

分别点击 startService 和StopService 的button ,看看生命周期回调方法打印的日志:

前三个日志是点击startService出现的

小结:通过startService 方式启动的服务,服务会无限期的在后台运行,直到通过stopService 或 stopSelf 来终止服务。服务独立于启动它的组件,也就是说,当组件启动服务后,组件和服务就在也没有关系了,就算启动它的组件被销毁了,服务照样在后台运行。通过这种方式启动的服务不好与组件之间通信。

bindService 方式启动服务

除了startService 来启动服务之外,另外一种启动服务的方式就是通过bindService 方法了,也就是绑定服务,其实通过它的名字就容易理解,绑定即将启动组件和服务绑定在一起。前面讲的通过startService 方式启动的服务是与组件相独立的,即使启动服务的组件被销毁了,服务仍然在后台运行不受干扰。但是通过bindSerivce 方式绑定的服务就不一样了,它与绑定组件的生命周期是有关的。如下:

多个组件可以绑定到同一个服务上,如果只有一个组件绑定服务,当绑定的组件被销毁时,服务也就会停止了。如果是多个组件绑定到一个服务上,当绑定到该服务的所有组件都被销毁时,服务才会停止。

bindService 绑定服务 和startService 的生命周期是不一样,bindServie 的生命周期如下:onCreate -> onBind -> onUnbind ->onDestroy。其中重要的就是onBind 和onUnbind 方法。

  • onBind(): 当其他组件想通过bindService 与服务绑定时,系统将会回调这个方法,在实现中,你必须返回一个IBinder接口,供客户端与服务进行通信,必须实现此方法,这个方法是Service 的一个抽象方法,但是如果你不允许绑定的话,返回null 就可以了。

  • onUnbind(): 当所有与服务绑定的组件都解除绑定时,就会调用此方法。

了解了这2个方法后,我们来看一下怎么绑定一个服务。
1,首先,添加一个类 继承 Binder ,在Binder 类中添加其他组件要与服务交互的方法,并在onBind() 方法中返回IBinder 实例对象:

public class SimpleService extends Service {
    public static final String TAG = "SimpleService";
    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        Log.i(TAG,"call onBind...");
        //返回IBinder 接口对象
        return new MyBinder();
    }

    @Override
    public boolean onUnbind(Intent intent) {
        Log.i(TAG,"call onUnbind...");
        return super.onUnbind(intent);
    }

    @Override
    public void onCreate() {
        Log.i(TAG,"call onCreate...");
    }

    @Override
    public void onStart(Intent intent, int startId) {
        Log.i(TAG,"call onStart...");
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        Log.i(TAG,"call onStartCommand...");
        return super.onStartCommand(intent, flags, startId);
    }

    @Override
    public void onDestroy() {
        Log.i(TAG,"call onDestroy...");
    }

    // 添加一个类继承Binder
    public  class MyBinder extends Binder{
        // 添加要与外界交互的方法
        public String  getStringInfo(){
          return "调用了服务中的方法";
        }
    }

}

2, 绑定服务的时候,需要提供一个ServiceConnection 接口,在接口回调中获取Binder 对象,与服务进行通信。

 private SimpleService.MyBinder mMyBinder;
    // 绑定/解除绑定 Service 回调接口
    private ServiceConnection mConnection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            // 绑定成功后回调
            //1 ,获取Binder接口对象
            mMyBinder = (SimpleService.MyBinder) service;
            //2, 从服务获取数据
            String content = mMyBinder.getStringInfo();
            // 3,界面提示
            Toast.makeText(ServiceSimpleActivity.this,content,Toast.LENGTH_LONG).show();
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
           // 解除绑定后回调
            mMyBinder = null;
        }
    };

3,绑定和解除绑定服务

        findViewById(R.id.bind_service).setOnClickListener(v -> {
            Intent intent = new Intent(this,SimpleService.class);
            // 绑定服务
            bindService(intent,mConnection, Context.BIND_AUTO_CREATE);
        });
        findViewById(R.id.unbind_service).setOnClickListener(v -> {
            // 解除绑定服务
            unbindService(mConnection);
        });

日志分别如下:

 可以看到,绑定服务的生命周期内依次调用了onCreate ,onBind,onUnbind 和 onDestroy 方法,只有中间两个生命周期方法与startService 启动服务是不同的。一张图就能看清两种方式的生命周期的异同:

tips: Service 的生命周期方法不同于Activity ,不需要调用超类的生命周期方法,如:不用调用 super.onCreate() 

多个组件绑定同一服务

Service 是支持多个组件绑定在同一个服务的,第一个组件绑定是会回调 onCreate 生命周期方法,后续的绑定只会调用onBind方法,返回IBinder给客户端。当绑定在服务上的组件都调用unbindService 解除服务或者组件本身就已经被系统回收,那么服务也就会被停止回收了,会回调onUnbind 和 onDestroy 方法。

Service 与应用组件通信的几种方式

1,BroadcastReceiver

通过前文我们知道,startService方式启动的服务在后台,无限期地运行,并且与启动它的组件是独立的,启动Service 之后也就与启动它的组件没有任何关系了。因此它是不能与启动它的组件之间相互通信的。虽然Service 没有提供这种启动方式的通信方法,我们还是可以通过其他方式来解决的,这就用到了BroadcastReceiver。

场景描述:通过startService 启动一个长期在后台运行的下载图片服务,然后在界面上点击下载按钮,通过intent 传递一个下载链接给Service,在下载完成后,通过BroadcastReceiver 通知Activity 界面显示图片。看一下代码实现:

Service代码如下:

public class DownloadService extends Service {

    public static final String IMAGE = "iamge_url";
    public static final String RECEIVER_ACTION = "simpleservice";
    public static final String ACTION_DOWNLOAD = "startdownload";

    private Looper mServiceLooper;
    private ServiceHandler mServiceHandler;
    private final class ServiceHandler extends Handler{
        public ServiceHandler(Looper looper){
            super(looper);
        }

        @Override
        public void handleMessage(@NonNull Message msg) {
            String url = (String) msg.obj;
            // 工作线程做耗时下载
            try {
                Bitmap bitmap = Picasso.with(getApplicationContext()).load(url).get();
                Intent intent = new Intent();
                intent.putExtra("bitmap",bitmap);
                intent.setAction(RECEIVER_ACTION);
                // 通知显示
                LocalBroadcastManager.getInstance(getApplicationContext()).sendBroadcast(intent);

            } catch (IOException e) {
                e.printStackTrace();
            }
            //工作完成之后,停止服务
            stopSelf();
        }
    }


    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }

    @Override
    public void onCreate() {
        // 开启一个工作线程做耗时工作
        HandlerThread thread = new HandlerThread("ServiceHandlerThread", Process.THREAD_PRIORITY_BACKGROUND);
        thread.start();
        // 获取工作线程的Looper
        mServiceLooper = thread.getLooper();
        // 创建工作线程的Handler
        mServiceHandler = new ServiceHandler(mServiceLooper);
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        if (intent.getAction().equals(ACTION_DOWNLOAD)){
            handleCommand(intent);
        }
        return START_STICKY;
    }

    private void handleCommand(Intent intent) {
        String url = intent.getStringExtra(IMAGE);
        // 发送消息下载
        Message message = mServiceHandler.obtainMessage();
        message.obj = url;
        mServiceHandler.sendMessage(message);
    }
}

新建了一个DownloadService ,在里面启动了一个工作线程,在线程里下载图片,然后通过BroadcastReceiver 通知Activity显示。

Activity的代码很简单,注册BroadcastReceiver,在onReceiver中显示图片就好了,代码如下:

private ImageView mImageView;
    private BroadcastReceiver mReceiver = new BroadcastReceiver() {
        @Override
        public void onReceive(Context context, Intent intent) {
            // 显示图片
            Bitmap bitmap = intent.getParcelableExtra("bitmap");
            mImageView.setImageBitmap(bitmap);
        }
    };

 这里涉及本地广播的使用


    /**
     * 启动下载
     */
    private void startDownload(){
        Intent intent = new Intent(this,DownloadService.class);
        // 启动服务
        intent.putExtra(DownloadService.IMAGE,"https://www.baidu.com/img/PCtm_d9c8750bed0b3c7d089fa7d55720d6cf.png");
        intent.setAction(DownloadService.ACTION_DOWNLOAD);
        startService(intent);
    }

最终成功将图片显示在界面上:

 

2, LocalService 使用Binder 和 服务通信

既然通过startService 启动的服务与启动它的组件是独立的。相互通信比较麻烦,那么Google也提供了两者之间的通信方法,那就是组件绑定服务,也就是上文讲的通过bindService 将组件和服务绑定到一起。组件可以获取Service 通过onBind返回的一个IBinder接口,这样两者就可以通信了,这也是Service 应用类通信比较常用的方式。

下面就模拟一个用服务播放音乐的例子来讲一下组件通过Binder 接口和服务之间通信。
首先定义一个通信的接口 IPlayer:

public interface IPlayer {
    // 播放
    public void play();
    // 暂停
    public void pause();
    // 停止
    public void stop();
    // 获取播放进度
    public int getProgress();
    // 获取时长
    public int getDuration();
}

其中比较重要的就是内部类LocalService ,继承Binder ,里面提供一个getService 方法,返回MusicService 实例,组件通过IBinder 获取到Music 实例后,就可以和Service之间相互通信啦!

Activity中代码如下:

 private MusicService.LocalService mLocalService;
    private MusicService mMusicService;
    // 绑定/解除绑定 Service 回调接口
    private ServiceConnection mConnection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            //1 ,获取Binder接口对象
            mLocalService = (MusicService.LocalService) service;
            //2, 获取MusicService 实例
            mMusicService = mLocalService.getService();

            // 只要拿到Music Service 实例之后,就可以调用接口方法了
            // 可以通过它来播放/暂停音乐,还可以通过它来获取当前播放音乐的进度,时长等等

            mMusicService.play();

            mMusicService.pause();

            mMusicService.stop();

            int progress = mMusicService.getProgress();
            Log.i(MusicService.TAG,"progress:"+progress);

            int duration = mMusicService.getDuration();
            Log.i(MusicService.TAG,"duration:"+duration);
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
           // 解除绑定后回调
            mMusicService = null;
            mLocalService = null;
        }
    };

获取到MusicService 后,就可以调用接口方法了,比如:播放音乐,暂停、停止、获取进度等等。
看一下打印的日志:

 注意:这里涉及的service都得在AndroidManifest文件中进行注册,这是比较容易忘记的点

Service 总结

Service 有2种启动方式,startService 启动服务,服务启动起来后,在后台无限期运行,直到通过stopService 或者 stopSelf 停止服务,服务与组件独立,通信比较困难(但还是有办法的,通过BroadcastReceiver )。另一种方式就是 bindService 即绑定服务,组件和服务绑定在一起,服务的生命后期受组件影响,如果绑定到服务的组件全部被销毁了,那么服务也就会停止了。绑定服务的方式通常用于组件和服务之间 需要相互通信。startService 这种 方式一般用于在后台执行任务,而不需要返回结果给组件。 这两种方式并非完全独立,也就是说,你可以绑定已经通过 startService 启动起来的服务,可以通过在Intent 中添加Action 来标示要执行的动作。比如:通过Intent Action 标记要播放的音乐,调用startService 来启动音乐服务播放音乐,在界面需要显示播放进度的时候,可以通过binderService 来绑定服务,从而获取歌曲信息。这种情况下,Service 需要实现两种方式的生命周期。这种情况下,除非所有客户端都已经取消绑定,否则通过stopService 或者 stopSelf 是不能停止服务的。

Service 是运行在主线程中的,因此不能执行耗时的或者密集型的任务,如果要执行耗时操作或者密集型计算任务,请在服务中开启工作线程,在线程中执行。或者使用下面一节将要讲的IntentService。

IntentService 

IntentService 默认为我们开启了一个工作线程,在任务执行完毕后,自动停止服务,因此在我们大多数的工作中,使用IntentService 就够了,并且IntentService 比较简单,只要实现一个方法OnHandleIntent,接下来看一下示例:

IntentService 扩展类:

public class MyIntentService extends IntentService {

    public static final String TAG ="MyIntentService";
    /**
     * Creates an IntentService.  Invoked by your subclass's constructor.
     *
     */
    public MyIntentService() {
        super("MyIntentService");
    }

    @Override
    protected void onHandleIntent(@Nullable Intent intent) {
        boolean isMainThread = Thread.currentThread() == Looper.getMainLooper().getThread();
        Log.i(TAG, "is main thread:" + isMainThread);

        mockDownLoad();

    }

    private void mockDownLoad(){
        try {
            Thread.sleep(5000);
            Log.i(TAG,"下载完成...");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    @Override
    public void onDestroy() {
        Log.i(TAG,"onDestroy...");
        super.onDestroy();
    }
}

然后启动服务,看一下打印的日志,如下图:

  Intent intent = new Intent(this,MyIntentService.class);
            // 启动服务
            startService(intent);

 判断了是否为主线程,结果为false ,说明是开启了一个工作线程,5s 之后,打印了下载完成,并且自动停止了服务。

IntentService 源码浅析

IntentService 自动为我们开启了一个线程来执行耗时操作,并且在任务完成后自动停止服务,那么它是怎么做的呢?我们看一下源码一探究竟。其实IntentService 的源码非常简单,就一百多行。一起看一下:

   // 1,有一个Looper 变量和一个ServiceHandler 变量,ServiceHander 继承Handler 处理消息
    private volatile Looper mServiceLooper;
    private volatile ServiceHandler mServiceHandler;
    private String mName;
    private boolean mRedelivery;

    private final class ServiceHandler extends Handler {
        public ServiceHandler(Looper looper) {
            super(looper);
        }

        @Override
        public void handleMessage(Message msg) {
           // 在工作线程中调用onHandleIntent,子类根据Intent传递的数据执行具体的操作
            onHandleIntent((Intent)msg.obj);
          // 任务执行完毕后,自动停止Service
            stopSelf(msg.arg1);
        }
    }
//2, 在OnCreate 方法中,创建了一个线程HandlerThread ,并启动线程
// 然后获取工作线程的Looper ,并用Looper 初始化Handler(我们都知道Handler 的创建需要一依赖Looper)
 public void onCreate() {
        // TODO: It would be nice to have an option to hold a partial wakelock
        // during processing, and to have a static startService(Context, Intent)
        // method that would launch the service & hand off a wakelock.

        super.onCreate();
        HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");
        thread.start();

        mServiceLooper = thread.getLooper();
        mServiceHandler = new ServiceHandler(mServiceLooper);
    }
//3, 在onStart()方法中发送消息给Handler,并且把Intent 传给了Handler 处理
 @Override
    public void onStart(@Nullable Intent intent, int startId) {
        Message msg = mServiceHandler.obtainMessage();
        msg.arg1 = startId;
        msg.obj = intent;
        mServiceHandler.sendMessage(msg);
    }
// 4,onStartCommand 直接调用的是onStart方法
public int onStartCommand(@Nullable Intent intent, int flags, int startId) {
        onStart(intent, startId);
        return mRedelivery ? START_REDELIVER_INTENT : START_NOT_STICKY;
    }
// 5 最后就是一个子类需要实现的抽象方法,这个方法在handleMessage中调用,也就是在工作线程中执行。
 protected abstract void onHandleIntent(@Nullable Intent intent);

上面代码中注释得很清楚,下面用一张图来看一下整个过程.

代码很简单,IntentService的源码看着是不是很熟悉?当然很熟悉,前面使用Service的时候,在Service 里面开启工作线程其实就和Intent Service的代码差不多。在onCreate中创建线程,启动,初始化Handler和Looper ,然后在onStartCommand中发送消息给Handler 处理任务。

IntentService 总结

IntentService是Service 的子类,默认给我们开启了一个工作线程执行耗时任务,并且执行完任务后自 动停止服务。扩展IntentService比较简单,提供一个构造方法和实现onHandleIntent 方法就可了,不用重写父类的其他方法。但是如果要绑定服务的话,还是要重写onBind 返回一个IBinder 的。使用Service 可以同时执行多个请求,而使用IntentService 只能同时执行一个请求。这里service的并发执行是指每次的startService都可以在onStartCommand方法开启一个子线程去执行任务。

 另外如果想了解多个任务在IntentService中是如何被顺序执行的,即多次startService IntentService,可查看该博客:多个任务在IntentService中是如何被顺序执行的_网鱼的栈-CSDN博客

以上就是对Service 和IntentService的详细总结,如果问题,欢迎讨论。

转载自:https://juejin.cn/post/6844903477777285134
作者:依然范特稀西
来源:掘金

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Android基础-Service和IntentService知识点详细总结 的相关文章

  • Android主线程的IO操作

    我的问题有两个 是否建议在 Android 的主线程上进行 IO 操作 或者它是否有可能导致我的应用程序崩溃 如果在主线程上执行 IO 操作不理想 我可以使用哪些其他框架 以便当我的应用程序加载时它可以执行一些基本的 IO 文件读取并将值存
  • 如何在应用程序关闭时在 Android 通知中显示操作按钮?

    我有一个安卓应用程序 对于通知 我们必须显示一些操作按钮 当应用程序打开时 我们可以自由地构建通知并显示操作按钮 但是当应用程序关闭时 通知会在 Android 的通知托盘中收到 应用程序开发人员无法控制构建用户界面和操作按钮 我们现在如何
  • 在 Android 上使用 AT 命令与调制解调器对话

    我试图与三星 Galaxy s2 plus 和华为 p1 xl u9200 调制解调器发送 AT 命令 但无法得到任何结果 我使用 adb shell 发送命令并使用 logcat 查看日志 在三星 当我执行 cat 时 proc tty
  • 拖动时跳转 ImageView。 getX() 和 getY() 值正在跳跃

    我创建了一个用于拖动视图的 onTouchListener 如果我使用的话 图像可以顺利拖动getRawX and getRawY 问题是 当您向下放置第二个指针然后抬起第一个指针时 图像将跳转到第二个指针 此 onTouchListene
  • 当路径的点超出视野时,Android Canvas 不会绘制路径

    我在绘制路径时遇到了 Android Canvas 的一些问题 我的情况是 我有一个相对布局工作 如地图视图 不使用 google api 或类似的东西 我必须在该视图上绘制一条路径 canvas drawPath polyPath bor
  • Android 全屏对话框确认和拒绝操作

    材料设计中的全屏对话框应该在操作栏 工具栏上有确认和拒绝操作 我的问题是 我该怎么做 显示对话框 getFragmentManager beginTransaction add R id container new MyDialogFrag
  • 如果使用grifika的ContinualCaptureActivity中的预览方式,相机预览的视野会更小

    我们知道 当相机预览比例设置为时 在相同距离下我们会得到更大的预览视野4 3代替16 9 具体如下 Android Camera API 奇怪的缩放效果 https stackoverflow com questions 20664628
  • 如何禁用操作栏上“向上”按钮的翻转?

    背景 我做了一个 应用程序管理器 https play google com store apps details id com lb app manager 替代应用程序 我希望添加 RTL 从右到左 语言的翻译 因为我知道在某些 And
  • React Native Expo StackNavigator 重叠通知栏

    我正在尝试为我的 React Native Expo 应用程序实现导航栏 这里有一个问题 dependencies expo 18 0 3 react 16 0 0 alpha 12 react native 0 45 1 react na
  • 在新的 intel x86 android 模拟器中访问 google api

    我只是尝试在新的 x86 android 模拟器中运行我公司的应用程序 但是我们的应用程序依赖于 google 地图 API 而这在 google 随 android sdk 版本 17 提供的 x86 系统映像中不可用 我的直觉告诉我答案
  • 如何使用 SharedPreferences 保存多个值?

    我正在开发一个字典应用程序 在我的应用程序中 我假设用户想要保存最喜欢的单词 我决定使用共享首选项保存这些值 我知道 SQLite 和文件更好 但我坚持使用 SharedPreferences 所以继续使用它 下面是我的代码 Overrid
  • 是否可以通过 Android 应用程序来录音?

    我是一名开发人员 希望创建一个 Android 应用程序来记录电话 这是出于我个人的需要 为了我自己的目的和记录而记录电话 是否有可能做到这一点 是否可以访问麦克风以及通过扬声器发出的声音 我对 Android 开发有点陌生 所以请耐心等待
  • 如何为我的 Android Market APK 创建证书?

    我想将我的第一个 APK 应用程序上传到 Android Market 但我收到了此错误 顺便说一下 在 stackoverflow 中搜索时并没有引导我找到正确的链接 市场不接受使用调试证书签名的 APK 创建有效期至少 50 年的新证书
  • 在运行时更改用作背景的 Drawable xml 内的形状纯色

    我有一个 Drawable xml 文件 background xml
  • 安卓。 CalendarView...一次仅显示一个月的日历

    我正在使用 CalendarView 其中我想一次仅查看一个月的日历并滚动查看下个月 但 CalendarView 一次显示所有月份 下面是我的代码
  • Android模拟器中的网络访问

    我试图通过我的 Android 应用程序访问互联网 但我既成功又失败 我在构建应用程序时启动模拟器 并且应用程序安装得很好 我可以使用浏览器访问互联网 但是 当我尝试这个小代码片段时 InetAddress inet try inet In
  • 如何在android中通过蓝牙向配对设备发送短信?

    在我的应用程序中 我想通过蓝牙发送和接收短信 我可以在列表视图中看到配对设备名称和地址的列表 但是当我尝试向配对设备发送文本时 什么也没有发生 在其他设备中没有收到文本 这是我向配对设备发送消息的代码 private void sendDa
  • 丢失应用程序的密钥库文件(但已启用 Google Play 应用程序签名)

    我已经失去了原来的keystore用于签署我的应用程序的文件 我的应用启用了 Google Play 应用签名 如果我联系 Google 支持人员 是否可以重置密钥 以便我可以继续上传到此包 我希望我可以做到这一点 因为应用程序签名已启用
  • 将对象从手机共享到 Android Wear

    我创建了一个应用程序 在此应用程序中 您拥有包含 2 个字符串 姓名和年龄 和一个位图 头像 的对象 所有内容都保存到 sqlite 数据库中 现在我希望可以在我的智能手表上访问这些对象 所以我想实现的是你可以去启动 启动应用程序并向左和向
  • 如何访问我的 Android 程序中的联系人

    我正在制作一个短信应用程序 并且想要访问我的 Android 应用程序中的联系人 我想访问联系人 就像他们在实际联系人列表中一样 选择后 我需要返回到我的活动 在其中我可以向该人发送短信 或者是否可以访问存储联系人的数据库 我的代码如下所示

随机推荐

  • Matlab利用现有模板图换背景——图像处理

    原理介绍 现在有三张图 有人物的图 二值的模板图 即我们要换图的模板 有目标板块 一张背景图 如下 我们的目标是将人物图的背景换成第三张图的背景 抠图换背景的原理 利用模板图 第二张二值图 将目标人物抠出 再将背景图中相应位置的值改为目标人
  • Cortex-M系列:错误异常

    目录 总线错误 1 不存在的地址 2 未对齐访问 3 对私有外设总线 PPB 的非特权访问和默认的存储器访问权限相冲突 存储管理错误 使用错误 HardFault 参考资料 总线错误 维基的解释为 In computing a bus er
  • 英伟达这篇CVPR 2022 Oral火了!2D图像秒变逼真3D物体!虚拟爵士乐队来了!

    点击下方卡片 关注 CVer 公众号 AI CV重磅干货 第一时间送达 你见过乐器自己演奏么 看看这个 图1 活灵活现 的虚拟乐器还是在 NVIDIA 服务器房间里面 尽情 般表演 这正是 NVIDIA Research 在庆祝爵士乐及其发
  • Active Directory 基础 —— 如何理解group的类型

    因为创建一个跨域的组 重新温习了一下最基本的AD知识 所谓温故而知新 把温习的结果整理了一下 AD里面的group类型从范围来说分为global universal 和 local domain 从类型来分分为security和distri
  • Halcon:表面裂纹检测

    原图 处理后的图片 主要方法为 对图像进行拆分 提取RGB三个分量 对B分量进行处理 将其转换为频域内图像 并对其进行高斯卷积 再将卷积处理后的图像转换回空间域图像 随后将B分量图像和处理后的B分量图像用算子sub image做差运算 最后
  • c语言删除json元素,JS 中彻底删除 JSON 对象组成的数组中的元素

    在 JS 中 对于某个由 JSON 对象组成的数组 例如 var test a 1 b 2 a 3 b 4 a 5 b 6 如果我们想要删除其中的第二个json对象 应该怎么做呢 其实方法和操作数组完全相同 在最开始的时候尝试使用了 del
  • IDM下载器

    在Windows PowerShell中运行 需要VPN Enable TLSv1 2 for compatibility with older clients System Net ServicePointManager Security
  • postman介绍

    原文 https blog csdn net weixin 39411616 article details 78655456 请求Request 1 URL 2 Method 根据方法的不同 body编辑器会发生变化 3 Headers
  • pandas.to_datetime() 只保留【年-月-日】

    Outline pandas to datetime 生成的日期会默认带有 2019 07 03 00 00 00 的分钟精度 但有时并不需要这些分钟精度 去掉分钟精度 可以通过pandas中的 dt date 去掉分钟 df just d
  • javaweb-react的入门遇到的问题一:state和props的初始化和设置修改

    引用仿照菜鸟教程的例子 需要网页有一个id为example333的div var HelloMessage React createClass getDefaultProps return password 123456 render fu
  • solidwork放置螺丝孔以及显示螺纹的问题

    放置螺丝孔 放置螺丝孔十分容易 网上便有很多教程 在此我也重复一次 如上图 在特征一行中点击 异型孔向导 在逐步选择所要的螺孔类型 和位置即可 新手如果纠结可以按照我的设置来 前提是你用的是M3的螺丝孔 再着再将螺丝孔放在你想要的位置即可
  • redis配置哨兵(sentinel)模式

    主从复制 当主服务器宕机时 我们需要手动的将从服务器中选取新的主服务器 这样会会需要人工干预 费时费力 而且不能及时的发现服务器宕机 造成一定时间内服务不可用 这并不是一个推荐的方案 我们更加建议使用哨兵模式替代 哨兵模式 Redis Se
  • 微信小程序wx.getLocation接口审核不通过

    审核不通过的原因一般包含这几种 一 当前提审小程序代码包中地理位置相关接口 wx getLocation 暂未开通 分析原因 接口未开通 解决方法 按下图申请开通对应的接口即可 二 你所描述的小程序接口使用场景 目前未符合接入wx getL
  • taro 兼容支付宝小程序和微信小程序<六>-- 自定义密码或验证码输入框组件的input 自动获取焦点(ios支付宝小程序怎么隐藏input的光标)

    项目 tar3 vue3 问题 自定义密码或验证码输入框需要自动聚焦 拉起键盘 A 微信小程序 用了自定义指令达到这个目的 自定义密码或验证码输入框 input自动获取焦点之坑 B 支付宝小程序 官方文档中说不支持自动获取焦点 必须手动拉起
  • 解决mysql经常断开重连的问题

    解决mysql自动断开连接的问题 有三个方法可以解决这个问题 1 修改MySQL配置参数 2 修改JDBC 3 修改第三方的数据库连接池应用 Proxool xml 方法1的解决方案 这个参数的名称是 wait timeout 其默认值为
  • redis 获取 list 中的所有元素

    一种方法是用 lrange key 0 1 这种方法不会影响 redis list 中的数据 List
  • QT 网络编程之https

    HTTP 超文本传输协议 是一个基于请求与响应 无状态的 应用层的协议 常基于TCP IP协议传输数据 互联网上应用最为广泛的一种网络协议 所有的WWW文件都必须遵守这个标准 设计HTTP的初衷是为了提供一种发布和接收HTML页面的方法 H
  • 【Python-3.5】matplotlib做简单折线图

    在matplotlib中使用plot 函数可以做出简单折线图 预期效果如下 代码如下 导入pyplot模块 import matplotlib pyplot as plt 输入横纵坐标数据 months 1 2 3 4 5 6 people
  • 分布式事务解决方案

    一 概述 分布式事务 分布式系统会把一个应用拆分为多个可独立部署的服务 此时要完成事务 就需要这些服务之间远程交互完成事务 简单的说跨JVM进程或者跨数据库实例产生分布式事务 典型的分布式事务场景 跨库事务 一个应用中某个功能需要操作多个库
  • Android基础-Service和IntentService知识点详细总结

    Service 对于广大的Android开发者来说算是耳熟能详了 作为Android的四大组件之一 在我们的开发中也起着重要的作用 在Android面试中 Service相关的问题也是面试官问得比较多的 当别人问你 Service 到底是什