前言
最近有很多朋友问我这个AIDL怎么用,也许由于是工作性质的原因,很多人都没有使用过aidl,所以和他们讲解完以后,感觉对方也是半懂不懂的,所以今天我就从浅到深的分析一下这个aidl具体是怎么用的,希望对大家有帮助。

作为一名合格Android开发人员,如果没听过Service,那就有点说不过去了啊,Service是Android四大组件之一,它是不依赖于用户界面的,就是因为Service不依赖与用户界面,所以我们常常用于进行一些耗时的操作,比如:下载数据等;
有些朋友可能是从事开发工作的时间不是特别的长,所以觉得Service相对与另外两个组件activity、broadcast receiver来说,使用可能并不是特别的多,所以对Service来说,理解不是特别的深入,只是有一个大概的概念,今天就和一块来走一下Service,希望能够帮助到大家对Service有更深入的理解。

Service基本用法——本地服务

我们知道服务分为本地服务和远程服务,而本地服务由于它的启动方式不一样,所以生命周期也就不一样,对Service生命周期不熟悉的朋友,自行去百度一下啊。好了,那么我们分别看一下两种不同的启动方式。我们先创建好Service:
ServiceTest.java

  1. package com.example.administrator.servicetestaidl;
  2.  
  3. import android.app.Service;
  4. import android.content.Intent;
  5. import android.os.IBinder;
  6. import android.support.annotation.IntDef;
  7. import android.util.Log;
  8.  
  9. public class ServiceTest extends Service {
  10.  
  11. @Override
  12. public void onCreate() {
  13. super.onCreate();
  14. Log.d("ServiceTest"," -----> onCreate");
  15. }
  16.  
  17. @Override
  18. public int onStartCommand(Intent intent,int flags, int startId) {
  19.  
  20. Log.d("ServiceTest"," -----> onStartCommand");
  21.  
  22. return super.onStartCommand(intent, flags, startId);
  23. }
  24.  
  25. @Override
  26. public void onDestroy() {
  27. super.onDestroy();
  28.  
  29. Log.d("ServiceTest"," -----> onDestroy");
  30.  
  31. }
  32.  
  33. @Override
  34. public IBinder onBind(Intent intent) {
  35. // TODO: Return the communication channel to the service.
  36. throw new UnsupportedOperationException("Not yet implemented");
  37. }
  38. }

在看看MainActivity的代码:

  1. package com.example.administrator.servicetestaidl;
  2.  
  3. import android.app.Activity;
  4. import android.content.Intent;
  5. import android.os.Bundle;
  6. import android.view.View;
  7. import android.widget.Button;
  8.  
  9. public class MainActivity extends Activity {
  10. private Button startService, stopService;
  11.  
  12. @Override
  13. protected void onCreate(Bundle savedInstanceState) {
  14. super.onCreate(savedInstanceState);
  15. setContentView(R.layout.activity_main);
  16.  
  17. startService = (Button) findViewById(R.id.start_service);
  18. stopService = (Button) findViewById(R.id.stop_service);
  19.  
  20. /**
  21. * 开启服务
  22. */
  23. startService.setOnClickListener(new View.OnClickListener() {
  24. @Override
  25. public void onClick(View v) {
  26. Intent startService = new Intent(MainActivity.this,ServiceTest.class);
  27. startService(startService);
  28.  
  29. }
  30. });
  31.  
  32. /**
  33. * 停止服务
  34. */
  35. stopService.setOnClickListener(new View.OnClickListener() {
  36. @Override
  37. public void onClick(View v) {
  38. Intent stopService = new Intent(MainActivity.this,ServiceTest.class);
  39. stopService(stopService);
  40. }
  41. });
  42.  
  43. }
  44. }

布局activity_main

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. android:orientation="vertical">
  6.  
  7. <Button
  8. android:id="@+id/start_service"
  9. android:layout_width="wrap_content"
  10. android:layout_height="wrap_content"
  11. android:text="开启服务" />
  12.  
  13. <Button
  14. android:id="@+id/stop_service"
  15. android:layout_width="wrap_content"
  16. android:layout_height="wrap_content"
  17. android:text="停止服务" />
  18.  
  19. </LinearLayout>

配置文件AndroidManifest.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="com.example.administrator.servicetestaidl">
  4.  
  5. <application
  6. android:allowBackup="true"
  7. android:icon="@mipmap/ic_launcher"
  8. android:label="@string/app_name"
  9. android:roundIcon="@mipmap/ic_launcher_round"
  10. android:supportsRtl="true"
  11. android:theme="@style/AppTheme">
  12. <activity android:name=".MainActivity">
  13. <intent-filter>
  14. <action android:name="android.intent.action.MAIN" />
  15.  
  16. <category android:name="android.intent.category.LAUNCHER" />
  17. </intent-filter>
  18. </activity>
  19.  
  20. <service
  21. android:name=".ServiceTest"
  22. android:enabled="true"
  23. android:exported="true"></service>
  24. </application>
  25.  
  26. </manifest>

上面的代码很简单,并不难理解,在页面上加两个按钮,一个是启动服务,一个是销毁服务的,并且我们在ServiceTest里面的几个方法都加上了log,那我们点击开启服务,看看Log,如图:

然后我们多次点击开启服务,如图:

我们看到,后面即使多点几下这个开启服务,但是也只会调onStartCommand方法,onCreate方法并不会重复调用,那是因为我们点击Service,由于该service已经存在,所以并不会重新创建,所以onCreate方法只会调用一次。
我们还可以到手机的应用程序管理界面来检查一下Service是不是正在运行,如下图所示:

那当我们点击停止服务按钮呢,看看log:如图

这时候说明了服务已经销毁了。

有些朋友可能注意到了,我们刚刚那种启动服务的方式,好像除了对Service进行开启和销毁以外,很难在activity里进行对Service进行控制,什么意思呢?举个例子,如果说
我现在用Service进行下载某些东西,我现在在Service写有下载这两个东西的方法,方法a,方法b,那么我怎样在activity里面控制什么时候调用方法a,什么时候调用方法b呢,如果
按照原本的启动方式,好像并不好实现,或者说灵活性很差,那么有没有办法办到呢,接着看Service另一种启动方式
在前面我们有一个方法一直都没有动onBind方法,我们就从这个方法入手,先看ServiceTest代码:

  1. package com.example.administrator.servicetestaidl;
  2.  
  3. import android.app.Service;
  4. import android.content.Intent;
  5. import android.os.Binder;
  6. import android.os.IBinder;
  7. import android.support.annotation.IntDef;
  8. import android.util.Log;
  9.  
  10. public class ServiceTest extends Service {
  11.  
  12. @Override
  13. public void onCreate() {
  14. super.onCreate();
  15. Log.d("ServiceTest"," -----> onCreate");
  16. }
  17.  
  18. @Override
  19. public int onStartCommand(Intent intent,int flags, int startId) {
  20.  
  21. Log.d("ServiceTest"," -----> onStartCommand");
  22.  
  23. return super.onStartCommand(intent, flags, startId);
  24. }
  25.  
  26. @Override
  27. public void onDestroy() {
  28. super.onDestroy();
  29.  
  30. Log.d("ServiceTest"," -----> onDestroy");
  31.  
  32. }
  33.  
  34. @Override
  35. public IBinder onBind(Intent intent) {
  36.  
  37. return new Mybind();
  38. }
  39.  
  40. class Mybind extends Binder{
  41. public void getString(){
  42. Log.d("ServiceTest"," -----> getString");
  43. }
  44. }
  45.  
  46. }

在ServiceTest中增加了一个内部类Mybind,并且在Mybind中增加一个getString方法,在方法中打印log,然后在onBind方法中返回Mybind对象。

再看看MainActivity的代码

  1. package com.example.administrator.servicetestaidl;
  2.  
  3. import android.app.Activity;
  4. import android.content.ComponentName;
  5. import android.content.Intent;
  6. import android.content.ServiceConnection;
  7. import android.os.Bundle;
  8. import android.os.IBinder;
  9. import android.view.View;
  10. import android.widget.Button;
  11.  
  12. public class MainActivity extends Activity {
  13.  
  14. private Button startService,stopService,bindService,unbindService;
  15. private ServiceTest.Mybind mybind;
  16.  
  17. private ServiceConnection connection = new ServiceConnection() {
  18. @Override
  19. public void onServiceConnected(ComponentName name, IBinder service) {
  20. mybind = (ServiceTest.Mybind) service;
  21. mybind.getString(); //获取到getString方法
  22. }
  23.  
  24. @Override
  25. public void onServiceDisconnected(ComponentName name) {
  26.  
  27. }
  28. };
  29.  
  30. @Override
  31. protected void onCreate(Bundle savedInstanceState) {
  32. super.onCreate(savedInstanceState);
  33. setContentView(R.layout.activity_main);
  34.  
  35. startService = (Button) findViewById(R.id.start_service);
  36. stopService = (Button) findViewById(R.id.stop_service);
  37. bindService = (Button) findViewById(R.id.bind_service);
  38. unbindService = (Button) findViewById(R.id.unbind_service);
  39.  
  40. /**
  41. * 开启服务
  42. */
  43. startService.setOnClickListener(new View.OnClickListener() {
  44. @Override
  45. public void onClick(View v) {
  46. Intent startService = new Intent(MainActivity.this,ServiceTest.class);
  47. startService(startService);
  48.  
  49. }
  50. });
  51.  
  52. /**
  53. * 停止服务
  54. */
  55. stopService.setOnClickListener(new View.OnClickListener() {
  56. @Override
  57. public void onClick(View v) {
  58. Intent stopService = new Intent(MainActivity.this,ServiceTest.class);
  59. stopService(stopService);
  60. }
  61. });
  62.  
  63. /**
  64. * 绑定服务
  65. */
  66. bindService.setOnClickListener(new View.OnClickListener() {
  67. @Override
  68. public void onClick(View v) {
  69. Intent bindService = new Intent(MainActivity.this,ServiceTest.class);
  70. bindService(bindService,connection,BIND_AUTO_CREATE);
  71. }
  72. });
  73.  
  74. /**
  75. * 解绑服务
  76. */
  77. unbindService.setOnClickListener(new View.OnClickListener() {
  78. @Override
  79. public void onClick(View v) {
  80. unbindService(connection);
  81. }
  82. });
  83. }
  84. }

主页面布局:activity_main

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. android:orientation="vertical">
  6.  
  7. <Button
  8. android:id="@+id/start_service"
  9. android:layout_width="wrap_content"
  10. android:layout_height="wrap_content"
  11. android:text="开启服务" />
  12.  
  13. <Button
  14. android:id="@+id/stop_service"
  15. android:layout_width="wrap_content"
  16. android:layout_height="wrap_content"
  17. android:text="停止服务" />
  18.  
  19. <Button
  20. android:id="@+id/bind_service"
  21. android:layout_width="wrap_content"
  22. android:layout_height="wrap_content"
  23. android:text="绑定服务" />
  24.  
  25. <Button
  26. android:id="@+id/unbind_service"
  27. android:layout_width="wrap_content"
  28. android:layout_height="wrap_content"
  29. android:text="解绑服务" />
  30.  
  31. </LinearLayout>

可以看到,这里我们首先创建了一个ServiceConnection的匿名类,在里面重写了onServiceConnected()方法和onServiceDisconnected()方法,这两个方法分别会在Activity与Service建立关联和解除关联的时候调用。在onServiceConnected()方法中,我们又通过向下转型得到了MyBind的实例,有了这个实例,Activity和Service之间的关系就变得非常紧密了。现在我们可以在Activity中根据具体的场景来调用MyBind中的任何public方法,即实现了Activity指挥Service干什么Service就去干什么的功能。

当我们点击绑定服务的时候,结果如下,如图

点击解绑服务的时候,结果如下,如图

注意:Service 是运行在后台,没有可视化的页面,我们很多时候会把耗时的操作放在Service中执行,但是注意,Service是运行在主线程的,不是在子线程中,
Service和Thread没有半毛钱的关系,所以如果在Service中执行耗时操作,一样是需要开起线程,否则会引起ANR,这个需要区别开来。

好了~~~本地服务基本到这就结束,在后面我会将本地服务这部分代码的链接发出来,需要的可以进行下载~~

远程服务 —— AIDL

AIDL(Android Interface Definition Language)是Android接口定义语言的意思,它可以用于让某个Service与多个应用程序组件之间进行跨进程通信,从而可以实现多个应用程序共享同一个Service的功能。实际上实现跨进程之间通信的有很多,
比如广播,Content Provider,但是AIDL的优势在于速度快(系统底层直接是共享内存),性能稳,效率高,一般进程间通信就用它。

既然是跨进程,那必须的有两个应用,一个是service端,一个是client端,然后实现客户端从服务端获取数据。那么我们创建一个服务端,项目结构如图所示:

服务端

我们在服务端下建立一个MyAIDLService.aidl文件,目录结构为如图所示:

然后,我们在MyAIDLService下增加一个获取字符串的方法。代码如下:(注:刚刚建立的aidl文件中存在一个方法,那个方法可以忽略,可以删掉不要)

  1. // MyAIDLService.aidl
  2. package aidl;
  3.  
  4. // Declare any non-default types here with import statements
  5.  
  6. interface MyAIDLService {
  7. //获取String数据
  8. String getString();
  9. }

创建完aidl文件以后,我们build一下项目,然后会在build - >generated ->source ->aidl->debug下会生成一个aidl文件,那说明AIDL文件已经编译成功。
接着建立一个MyService类,代码如下:

  1. package com.example.service;
  2.  
  3. import android.app.Service;
  4. import android.content.Intent;
  5. import android.os.Binder;
  6. import android.os.IBinder;
  7. import android.os.RemoteException;
  8. import android.util.Log;
  9.  
  10. import java.util.Map;
  11.  
  12. import aidl.MyAIDLService;
  13.  
  14. public class MyService extends Service {
  15.  
  16. @Override
  17. public void onCreate() {
  18. super.onCreate();
  19. }
  20.  
  21. @Override
  22. public int onStartCommand(Intent intent, int flags, int startId) {
  23. return super.onStartCommand(intent, flags, startId);
  24. }
  25.  
  26. @Override
  27. public void onDestroy() {
  28. super.onDestroy();
  29. }
  30.  
  31. @Override
  32. public IBinder onBind(Intent intent) {
  33. return new Mybind();
  34. }
  35.  
  36. class Mybind extends MyAIDLService.Stub {
  37.  
  38. @Override
  39. public String getString() throws RemoteException {
  40. String string = "我是从服务起返回的";
  41.  
  42. return string;
  43. }
  44. }
  45. }

代码看起来是不是很熟悉,唯一不一样的就是原来在本地服务的时候内部类继承的是Binder,而现在继承的是MyAIDLService.Stub,继承的是我们刚刚建立的aidl文件,然后实现我们刚刚的定义的
getString()方法,在这里,我们只是返回一句话,"我是从服务起返回的"~~~~~~~~~~~

客户端

首先将刚刚在服务端创建的MyAIDLService原封不动的复制到客户端来。(注意:路径要一模一样)。接着我们在客户端的MainActivity中加两个按钮,并且和服务端进行相连,代码如下:

MainActivity

  1. package com.example.administrator.servicetestaidl;
  2.  
  3. import android.app.Activity;
  4. import android.content.ComponentName;
  5. import android.content.Intent;
  6. import android.content.ServiceConnection;
  7. import android.os.Bundle;
  8. import android.os.IBinder;
  9. import android.os.RemoteException;
  10. import android.view.View;
  11. import android.widget.Button;
  12. import android.widget.TextView;
  13.  
  14. import aidl.MyAIDLService;
  15.  
  16. public class MainActivity extends Activity {
  17.  
  18. private Button bindService,unbindService;
  19. private TextView tvData;
  20. private MyAIDLService myAIDLService;
  21.  
  22. private ServiceConnection connection = new ServiceConnection() {
  23. @Override
  24. public void onServiceConnected(ComponentName name, IBinder service) {
  25. myAIDLService = MyAIDLService.Stub.asInterface(service);
  26. try {
  27. String str = myAIDLService.getString();
  28. tvData.setText(str);
  29. } catch (RemoteException e) {
  30. e.printStackTrace();
  31. }
  32. }
  33.  
  34. @Override
  35. public void onServiceDisconnected(ComponentName name) {
  36. myAIDLService = null;
  37. }
  38. };
  39.  
  40. @Override
  41. protected void onCreate(Bundle savedInstanceState) {
  42. super.onCreate(savedInstanceState);
  43. setContentView(R.layout.activity_main);
  44.  
  45. bindService = (Button) findViewById(R.id.bind_service);
  46. unbindService = (Button) findViewById(R.id.unbind_service);
  47. tvData = (TextView) findViewById(R.id.tv_data);
  48.  
  49. /**
  50. * 绑定服务
  51. */
  52. bindService.setOnClickListener(new View.OnClickListener() {
  53. @Override
  54. public void onClick(View v) {
  55. Intent intent = new Intent();
  56. intent.setAction("com.example.service.MyService");
  57. //从 Android 5.0开始 隐式Intent绑定服务的方式已不能使用,所以这里需要设置Service所在服务端的包名
  58. intent.setPackage("com.example.service");
  59. bindService(intent, connection, BIND_AUTO_CREATE);
  60.  
  61. }
  62. });
  63.  
  64. /**
  65. * 解绑服务
  66. */
  67. unbindService.setOnClickListener(new View.OnClickListener() {
  68. @Override
  69. public void onClick(View v) {
  70. unbindService(connection);
  71. }
  72. });
  73. }
  74. }

大家是不是感觉和连接本地服务的代码差不多,没错,这里只需要注意两个地方,一个是绑定服务的时候,因为从 Android 5.0开始 隐式Intent绑定服务的方式已不能使用,所以这里需要设置Service所在服务端的包名

那么这个action是怎么来的呢,我们回来服务端的AndroidManifest.xml,代码如下

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="com.example.service">
  4.  
  5. <application
  6. android:allowBackup="true"
  7. android:icon="@mipmap/ic_launcher"
  8. android:label="@string/app_name"
  9. android:roundIcon="@mipmap/ic_launcher_round"
  10. android:supportsRtl="true"
  11. android:theme="@style/AppTheme">
  12. <activity android:name=".MainActivity">
  13. <intent-filter>
  14. <action android:name="android.intent.action.MAIN" />
  15.  
  16. <category android:name="android.intent.category.LAUNCHER" />
  17. </intent-filter>
  18. </activity>
  19.  
  20. <service
  21. android:name=".MyService"
  22. >
  23. <intent-filter>
  24. <action android:name="com.example.service.MyService" />
  25. </intent-filter>
  26.  
  27. </service>
  28. </application>
  29.  
  30. </manifest>

另一个需要注意的就是获取MyAIDLService对象是通过MyAIDLService.Stub.asInterface(service);这里大家需要注意一下的。
不过还有一点需要说明的是,由于这是在不同的进程之间传递数据,Android对这类数据的格式支持是非常有限的,
基本上只能传递Java的基本数据类型、字符串、List或Map等。那么如果我想传递一个自定义的类该怎么办呢?这就必须要让这个类去实现Parcelable接口,
并且要给这个类也定义一个同名的AIDL文件。这部分内容并不复杂,而且和Service关系不大,所以就不再详细进行讲解了,感兴趣的朋友可以自己去查阅一下相关的资料。

注意:从服务器复制过来的aidl文件不能直接放到Java文件夹下面,必须建立一个aidl文件夹存放,否则会编译不成功

好了,到这里,基本上就结束了,附上一张效果图:

最后附上源码链接

本地服务源码:https://github.com/343661629/nativeService

远程服务源码:https://github.com/343661629/remoteService

Service由浅到深——AIDL的使用方式的更多相关文章

  1. 浅入深出之Java集合框架(上)

    Java中的集合框架(上) 由于Java中的集合框架的内容比较多,在这里分为三个部分介绍Java的集合框架,内容是从浅到深,如果已经有java基础的小伙伴可以直接跳到<浅入深出之Java集合框架 ...

  2. 浅入深出之Java集合框架(中)

    Java中的集合框架(中) 由于Java中的集合框架的内容比较多,在这里分为三个部分介绍Java的集合框架,内容是从浅到深,如果已经有java基础的小伙伴可以直接跳到<浅入深出之Java集合框架 ...

  3. 浅入深出之Java集合框架(下)

    Java中的集合框架(下) 由于Java中的集合框架的内容比较多,在这里分为三个部分介绍Java的集合框架,内容是从浅到深,哈哈这篇其实也还是基础,惊不惊喜意不意外 ̄▽ ̄ 写文真的好累,懒得写了.. ...

  4. 浅入深出Vue:环境搭建

    浅入深出Vue:环境搭建 工欲善其事必先利其器,该搭建我们的环境了. 安装NPM 所有工具的下载地址都可以在导航篇中找到,这里我们下载的是最新版本的NodeJS Windows安装程序 下载下来后,直 ...

  5. 浅入深出Vue:工具准备之PostMan安装配置及Mock服务配置

    浅入深出Vue之工具准备(二):PostMan安装配置 由于家中有事,文章没顾得上.在此说声抱歉,这是工具准备的最后一章. 接下来就是开始环境搭建了~尽情期待 工欲善其事必先利其器,让我们先做好准备工 ...

  6. 浅入深出Vue:工具准备之WebStorm安装配置

    浅入深出Vue之工具准备(一):WebStorm安装配置 工欲善其事必先利其器,让我们先做好准备工作吧 导航篇 WebStorm安装配置 所有工具的下载地址都可以在导航篇中找到,这里我们下载的是最新版 ...

  7. 浅入深出Vue系列

    浅入深出Vue导航 导航帖,直接点击标题即可. 文中所有涉及到的资源链接均在最下方列举出来了. 前言 基础篇 浅入深出Vue:工具准备之WebStorm搭建及配置 浅入深出Vue之工具准备(二):Po ...

  8. 浅入深出Vue:前言

    浅入深出Vue系列文章 之前大部分是在做后端,后来出于某些原因开始接触Vue.深感前端变化之大,各种工具.框架令人眼花缭乱.不过正是这些变化,让前端开发更灵活. 博主在刚开始时,参考官网的各个步骤以及 ...

  9. javafx由浅到深的 认识(一)

    javafx是一款比较新兴的语言框架,随着javafx越来越实用,估计许多程序员也会慢慢接触它,故我在这里对它由浅到深进行介绍一下. 首先,要了解javafx,就应该先知道.xml文件的布局软件,以往 ...

随机推荐

  1. Grapher

    [Grapher] You use Grapher to visualize and analyze implicit and explicit equations. You can graph eq ...

  2. Halcon中xld的常见特征的含义总结

    如下图:

  3. [Selenium]怎样验证页面是否有无变化

    验证方法:将两次的Dom结构进行对比 String beforeStr = (String) SeleniumUtil.getInnerHTML(page.getDriver(), page.getD ...

  4. Web01 基础

    网址组成(四部分) 协议  http,https (https是加密的http) 主机  g.cn zhihu.com之类的网址 端口 HTTP协议默认80,因此一般不用填写 路径 下面的 / 和 / ...

  5. LIst和table的转换

    public static class DataTableExtensions { /// <summary> /// 转化一个DataTable /// </summary> ...

  6. 分布式java应用基础与实践

      始读于2014年4月30日,完成于2014年6月6日15:43:39. 阿里巴巴高级研究员林昊早年的书了,这些理论放到今天估计工作一两年的人都耳熟能详了,我个人很早以前就知道此书一直没有找到资源, ...

  7. radiobutton 选中的项不能去掉选择的问题

    代码如下: RadioButton rbtn = new RadioButton(getApplicationContext()); rbtn.setText(String.valueOf(item. ...

  8. VMware安装Ubuntu

    1. VMware Ubuntu安装详细过程:http://blog.csdn.net/u013142781/article/details/50529030 2. 怎样在VMware虚拟机中使用安装 ...

  9. 打开窗口进行选择文件(txt文件),打开所选文件,读入文件

    用mfc编写项目的时候往往需要调用窗口,允许用户通过窗口进行选择文件操作 TCHAR szBuffer[MAX_PATH] = { 0 }; OPENFILENAME ofn = { 0 }; ofn ...

  10. Linux 基础教程 38-文件下载

    什么是wget     wget用原始帮助里面的英文来讲就是:The non-interactive network downloader,非交互式网络下载器.它支持HTTP.HTTPS.FTP等协议 ...