前面将的都是如何使用C/C++文件生成so动态库,那么在使用别人的so动态库的时候应该怎么做呢?这篇文章就是使用一个变声功能的动态库,完成对于以有so动态库的说明。

动态库来源

  • 在互联网中,有着许许多多动态库,很多厂商也对外提供动态库供开发者调用,例如高德地图的动态库,做地图开发的时候还是很方便的

  • 本文主要讲一个可以使声音改变的动态库,这个动态库主要用于游戏中,游戏引擎中有使用到

  • 这就是fmod动态库,首先我们要去下载其动态库文件
    官网地址
    先要注册才能下载其文件,按照步骤来就好

  • 在其下载界面,有FMOD Studio API,这里可以选择版本下载,我写这篇博客的时候,最新版本是1.10.07,在这里就不下载最新的了,我选择的是1.08.28,也就是1.08的最后一个版本。

添加到项目中

  • 解压下载的文件,发现在api文件夹下有三个目录:fsbanklowlevelstudio

  • 这里选择lowlevel,这是基于普遍使用选择的,也可以选择studio,其功能更为强大,不过相对地也更需要运算性能

  • 在Android项目中新建libs目录,将fmod.jar拷贝至libs目录

  • 新建jni目录,将armeabiarmeabi-v7a下的so文件拷贝至jni目录,将lowlevel目录下的inc头文件拷贝至jni文件夹,在这里先实现原声播放的功能,故在lowlevel下的examples下找到play_sound.cpp源文件,将其放在jni目录下,打开文件得知,其依赖的common.h头文件并不在inc中,找到common.h并拷贝至jni中,逐步寻找缺失的依赖文件,导入到jni中,整理完成后的jni文件目录如下:

│  Android.mk
│  common.cpp
│  common.h
│  common_platform.cpp
│  common_platform.h
│  libfmod.so
│  libfmodL.so
│  play_sound.cpp
└─incfmod.hfmod.hppfmod_codec.hfmod_common.hfmod_dsp.hfmod_dsp_effects.hfmod_errors.hfmod_output.h

修改文件使其能够调用

  • lowlevel目录下,有Java的调用示例,在这里直接使用这个MainActivity.java进行修改调用

  • 阅读MainActivity.java源代码,发现其使用的是动态获取权限,为方便使用,直接在清单文件中生命其权限,将其动态申请注释掉,在动态库加载时候,发现加载了一些没有的动态库,将没有的动态库去掉,加上自己的动态库,注意到jni中的调用方法和现有包名不统一,修改之,并将清单文件中的启动活动包名也修改

package org.fmod.example;import android.os.Build;
import android.os.Bundle;
import android.app.Activity;
import android.graphics.Typeface;
import android.util.TypedValue;
import android.view.MotionEvent;
import android.view.View;
import android.view.View.OnTouchListener;
import android.widget.LinearLayout;
import android.widget.LinearLayout.LayoutParams;
import android.widget.TextView;
import android.widget.Button;
import android.content.pm.PackageManager;public class MainActivity extends Activity implements OnTouchListener, Runnable
{private TextView mTxtScreen;private Thread mThread;@Overrideprotected void onCreate(Bundle savedInstanceState){super.onCreate(savedInstanceState);// Create the text areamTxtScreen = new TextView(this);mTxtScreen.setTextSize(TypedValue.COMPLEX_UNIT_SP, 10.0f);mTxtScreen.setTypeface(Typeface.MONOSPACE);// Create the buttonsButton[] buttons = new Button[9];for (int i = 0; i < buttons.length; i++){buttons[i] = new Button(this);buttons[i].setText(getButtonLabel(i));buttons[i].setOnTouchListener(this);buttons[i].setId(i);}// Create the button row layoutsLinearLayout llTopRowButtons = new LinearLayout(this);llTopRowButtons.setOrientation(LinearLayout.HORIZONTAL);LinearLayout llMiddleRowButtons = new LinearLayout(this);llMiddleRowButtons.setOrientation(LinearLayout.HORIZONTAL);LinearLayout llBottomRowButtons = new LinearLayout(this);llBottomRowButtons.setOrientation(LinearLayout.HORIZONTAL);// Create the main view layoutLinearLayout llView = new LinearLayout(this);llView.setOrientation(LinearLayout.VERTICAL);       // Create layout parametersLinearLayout.LayoutParams lpLayout = new LinearLayout.LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT, 1.0f);// Set up the view hierarchyllTopRowButtons.addView(buttons[0], lpLayout);llTopRowButtons.addView(buttons[6], lpLayout);llTopRowButtons.addView(buttons[1], lpLayout);llMiddleRowButtons.addView(buttons[4], lpLayout);llMiddleRowButtons.addView(buttons[8], lpLayout);llMiddleRowButtons.addView(buttons[5], lpLayout);llBottomRowButtons.addView(buttons[2], lpLayout);llBottomRowButtons.addView(buttons[7], lpLayout);llBottomRowButtons.addView(buttons[3], lpLayout);llView.addView(mTxtScreen, lpLayout);llView.addView(llTopRowButtons);llView.addView(llMiddleRowButtons);llView.addView(llBottomRowButtons);setContentView(llView);// Request necessary permissions
//        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M)
//        {//            String[] perms = { "android.permission.RECORD_AUDIO", "android.permission.WRITE_EXTERNAL_STORAGE" };
//            if (checkSelfPermission(perms[0]) == PackageManager.PERMISSION_DENIED ||
//                checkSelfPermission(perms[1]) == PackageManager.PERMISSION_DENIED)
//            {//                requestPermissions(perms, 200);
//            }
//        }org.fmod.FMOD.init(this);mThread = new Thread(this, "Example Main");mThread.start();setStateCreate();}@Overrideprotected void onStart(){super.onStart();setStateStart();}@Overrideprotected void onStop(){setStateStop();super.onStop();}@Overrideprotected void onDestroy(){setStateDestroy();try{mThread.join();}catch (InterruptedException e) { }org.fmod.FMOD.close();super.onDestroy();}@Overridepublic boolean onTouch(View view, MotionEvent motionEvent){if (motionEvent.getAction() == MotionEvent.ACTION_DOWN){buttonDown(view.getId()); }else if (motionEvent.getAction() == MotionEvent.ACTION_UP){buttonUp(view.getId());   }           return true;}@Overridepublic void run(){main();}public void updateScreen(final String text){runOnUiThread(new Runnable(){@Overridepublic void run(){mTxtScreen.setText(text);}});}private native String getButtonLabel(int index);private native void buttonDown(int index);private native void buttonUp(int index);private native void setStateCreate();private native void setStateStart();private native void setStateStop();private native void setStateDestroy();private native void main();static {/** To simplify our examples we try to load all possible FMOD* libraries, the Android.mk will copy in the correct ones* for each example. For real products you would just load* 'fmod' and if you use the FMOD Studio tool you would also* load 'fmodstudio'.*///       // Try debug libraries...
//      try { System.loadLibrary("fmodD");
//            System.loadLibrary("fmodstudioD"); }
//      catch (UnsatisfiedLinkError e) { }
//      // Try logging libraries...
//      try { System.loadLibrary("fmodL");
//            System.loadLibrary("fmodstudioL"); }
//      catch (UnsatisfiedLinkError e) { }
//      // Try release libraries...
//      try { System.loadLibrary("fmod");
//            System.loadLibrary("fmodstudio"); }
//      catch (UnsatisfiedLinkError e) { }
//
//      System.loadLibrary("stlport_shared");
//        System.loadLibrary("example");try { System.loadLibrary("fmod");System.loadLibrary("fmodL"); } catch (UnsatisfiedLinkError e) {e.printStackTrace();}System.loadLibrary("VoiceChangeTest");}
}
  • 使用Android Tools添加本地支持

  • 修改Android.mk文件,并记录动态库的文件名,将其加载至MainActivity.java

LOCAL_PATH := $(call my-dir)include $(CLEAR_VARS)LOCAL_MODULE    := VoiceChangeTest
LOCAL_SRC_FILES := play_sound.cppinclude $(BUILD_SHARED_LIBRARY)

编译项目

  • 此时便可以编译项目了,编译时候会提示有些文件找不到,那是因为包含文件路径不对造成的,此时修改包含文件路径即可

  • 文件包含错误解决以后,再次编译,发现很多函数找不到,此时是因为编译时候那些函数的实现没有编译到项目,修改Android.mk文件,加入依赖实现

LOCAL_PATH := $(call my-dir)include $(CLEAR_VARS)
LOCAL_MODULE := fmod
LOCAL_SRC_FILES := libfmod.so
include $(PREBUILT_SHARED_LIBRARY)include $(CLEAR_VARS)
LOCAL_MODULE := fmodL
LOCAL_SRC_FILES := libfmodL.so
include $(PREBUILT_SHARED_LIBRARY)include $(CLEAR_VARS)
LOCAL_MODULE    := VoiceChangeTest
LOCAL_SRC_FILES := play_sound.cpp \common_platform.cpp \common.cpp
LOCAL_SHARED_LIBRARIES := fmod fmodL
include $(BUILD_SHARED_LIBRARY)
  • 另外,由于用到了STL库,需要在配置里说明,在jni下新建Application.mk文件,写入以下配置
APP_STL := gnustl_static
  • 至此,项目修改完毕,便可以生成apk了。运行界面如下:

仿QQ变声效果实现

在大致了解fmod以后,就可以做一些基于fmod的项目了,正好QQ有一个变声的功能,这里就使用fmod去实现

采集素材

直接将QQ安装包解压,就可以得到图片素材,将其加入到素材中

编辑界面

编写界面,这里直接采用他人布局好的文件

<RelativeLayout 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" ><LinearLayoutandroid:layout_width="fill_parent"android:layout_height="wrap_content"android:layout_alignParentBottom="true"android:orientation="vertical"android:background="#FFF"android:paddingBottom="20dp" ><LinearLayoutandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="horizontal"android:layout_marginTop="20dp"><LinearLayout android:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="vertical"><ImageViewandroid:id="@+id/btn_normal"style="@style/AudioImgStyle"android:src="@drawable/record"android:onClick="mFix"/><TextView style="@style/AudioTextStyle"android:text="原声"/></LinearLayout><LinearLayout android:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="vertical"><ImageViewandroid:id="@+id/btn_luoli"style="@style/AudioImgStyle"android:src="@drawable/luoli"android:onClick="mFix"/><TextView style="@style/AudioTextStyle"android:text="萝莉"/></LinearLayout><LinearLayout android:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="vertical"><ImageViewandroid:id="@+id/btn_dashu"style="@style/AudioImgStyle"android:src="@drawable/dashu"android:onClick="mFix"/><TextView style="@style/AudioTextStyle"android:text="大叔"/></LinearLayout></LinearLayout><LinearLayoutandroid:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="horizontal"android:layout_marginTop="20dp"><LinearLayout android:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="vertical"><ImageViewandroid:id="@+id/btn_jingsong"style="@style/AudioImgStyle"android:src="@drawable/jingsong"android:onClick="mFix"/><TextView style="@style/AudioTextStyle"android:text="惊悚"/></LinearLayout><LinearLayout android:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="vertical"><ImageViewandroid:id="@+id/btn_gaoguai"style="@style/AudioImgStyle"android:src="@drawable/gaoguai"android:onClick="mFix"/><TextView style="@style/AudioTextStyle"android:text="搞怪"/></LinearLayout><LinearLayout android:layout_width="wrap_content"android:layout_height="wrap_content"android:orientation="vertical"><ImageViewandroid:id="@+id/btn_kongling"style="@style/AudioImgStyle"android:src="@drawable/kongling" android:onClick="mFix"/><TextView style="@style/AudioTextStyle"android:text="空灵"/></LinearLayout></LinearLayout></LinearLayout></RelativeLayout>

style文件

<resources><style name="AppBaseTheme" parent="android:Theme.Light"><!--Theme customizations available in newer API levels can go inres/values-vXX/styles.xml, while customizations related tobackward-compatibility can go here.--></style><!-- Application theme. --><style name="AppTheme" parent="AppBaseTheme"><!-- All customizations that are NOT specific to a particular API-level can go here. --></style><style name="AudioImgStyle"><item name="android:layout_width">wrap_content</item><item name="android:layout_height">wrap_content</item><item name="android:layout_marginLeft">25dp</item><item name="android:layout_marginRight">25dp</item></style><style name="AudioTextStyle"><item name="android:layout_width">wrap_content</item><item name="android:layout_height">wrap_content</item><item name="android:layout_marginTop">5dp</item><item name="android:layout_gravity">center_horizontal</item></style>
</resources>

编辑java代码

主活动

package org.fmod.example;import java.io.File;import org.fmod.FMOD;import android.app.Activity;
import android.os.Bundle;
import android.os.Environment;
import android.util.Log;
import android.view.View;public class QQActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);FMOD.init(this);setContentView(R.layout.activity_main);}public void mFix(View v) {String path = Environment.getExternalStorageDirectory().getAbsolutePath() + File.separatorChar + "test.wav";Log.e("java cj5785", path);switch (v.getId()) {case R.id.btn_normal:Log.d("cj5785", "normal");QQVoice.fix(path, QQVoice.MODE_NORMAL);break;case R.id.btn_luoli:Log.d("cj5785", "luoli");QQVoice.fix(path, QQVoice.MODE_LUOLI);break;case R.id.btn_dashu:Log.d("cj5785", "dashu");QQVoice.fix(path, QQVoice.MODE_DASHU);break;case R.id.btn_jingsong:Log.d("cj5785", "jingsong");QQVoice.fix(path, QQVoice.MODE_JINGSONG);break;case R.id.btn_gaoguai:Log.d("cj5785", "gaoguai");QQVoice.fix(path, QQVoice.MODE_GAOGUAI);break;case R.id.btn_kongling:Log.d("cj5785", "kongling");QQVoice.fix(path, QQVoice.MODE_KONGLING);break;}}@Overrideprotected void onDestroy() {super.onDestroy();FMOD.close();}
}

工具类

package org.fmod.example;public class QQVoice {//音效的类型public static final int MODE_NORMAL = 0;public static final int MODE_LUOLI = 1;public static final int MODE_DASHU = 2;public static final int MODE_JINGSONG = 3;public static final int MODE_GAOGUAI = 4;public static final int MODE_KONGLING = 5;public native static void fix(String path, int type);static {System.loadLibrary("fmod");System.loadLibrary("fmodL");System.loadLibrary("QQVoice");}
}

native方法实现

#include <stdlib.h>
#include <unistd.h>#include "inc/fmod.hpp"
#include "org_fmod_example_QQVoice.h"#include <android/log.h>
#define LOGD(FORMAT,...) __android_log_print(ANDROID_LOG_DEBUG,"cj5785",FORMAT,##__VA_ARGS__);
#define LOGE(FORMAT,...) __android_log_print(ANDROID_LOG_ERROR,"cj5785",FORMAT,##__VA_ARGS__);#define MODE_NORMAL 0
#define MODE_LUOLI 1
#define MODE_DASHU 2
#define MODE_JINGSONG 3
#define MODE_GAOGUAI 4
#define MODE_KONGLING 5using namespace FMOD;JNIEXPORT void JNICALL Java_org_fmod_example_QQVoice_fix(JNIEnv *env, jclass jcls, jstring jstr_path, jint type)
{System *system;Sound *sound;Channel *channel;DSP *dsp;bool playing = true;float frequency = 0.0F;const char *path_cstr = env->GetStringUTFChars(jstr_path, NULL);try{//初始化System_Create(&system);system->init(32, FMOD_INIT_NORMAL, NULL);//创建声音system->createSound(path_cstr, FMOD_DEFAULT, NULL, &sound);switch (type){//原声case MODE_NORMAL:LOGD("%s", "NORMAL");system->playSound(sound, 0, false, &channel);break;//萝莉case MODE_LUOLI:LOGD("%s", "LUOLI");system->playSound(sound, 0, false, &channel);system->createDSPByType(FMOD_DSP_TYPE_PITCHSHIFT, &dsp);//设置音调dsp->setParameterFloat(FMOD_DSP_PITCHSHIFT_PITCH, 2.5);//添加至channelchannel->addDSP(0, dsp);break;//大叔case MODE_DASHU:LOGD("%s", "DASHU");system->playSound(sound, 0, false, &channel);system->createDSPByType(FMOD_DSP_TYPE_PITCHSHIFT, &dsp);//设置音调dsp->setParameterFloat(FMOD_DSP_PITCHSHIFT_PITCH, 0.7);//添加至channelchannel->addDSP(0, dsp);break;//惊悚case MODE_JINGSONG:LOGD("%s", "JINGSONG");system->playSound(sound, 0, false, &channel);system->createDSPByType(FMOD_DSP_TYPE_TREMOLO, &dsp);dsp->setParameterFloat(FMOD_DSP_TREMOLO_SKEW, 0.5);channel->addDSP(0, dsp);break;//搞怪case MODE_GAOGUAI:LOGD("%s", "GAOGUAI");system->playSound(sound, 0, false, &channel);//提高说话速度channel->getFrequency(&frequency);frequency *= 2.3;channel->setFrequency(frequency);break;//空灵case MODE_KONGLING:LOGD("%s", "KONGLING");system->playSound(sound, 0, false, &channel);system->createDSPByType(FMOD_DSP_TYPE_ECHO, &dsp);dsp->setParameterFloat(FMOD_DSP_ECHO_DELAY, 300);dsp->setParameterFloat(FMOD_DSP_ECHO_FEEDBACK, 50);channel->addDSP(0, dsp);break;default:break;}}catch(...){LOGE("s", "异常状况发生");goto End;}system->update();while(playing){channel->isPlaying(&playing);usleep(1000 * 1000);}goto End;
End://释放资源env->ReleaseStringUTFChars(jstr_path, path_cstr);sound->release();system->close();system->release();
}

mk文件修改

LOCAL_PATH := $(call my-dir)include $(CLEAR_VARS)
LOCAL_MODULE := fmod
LOCAL_SRC_FILES := libfmod.so
include $(PREBUILT_SHARED_LIBRARY)include $(CLEAR_VARS)
LOCAL_MODULE := fmodL
LOCAL_SRC_FILES := libfmodL.so
include $(PREBUILT_SHARED_LIBRARY)include $(CLEAR_VARS)
LOCAL_MODULE    := QQVoice
LOCAL_SRC_FILES := QQVoice.cpp
LOCAL_SHARED_LIBRARIES := fmod fmodL
LOCAL_LDLIBS := -llog
LOCAL_CPP_FEATURES := exceptionsinclude $(BUILD_SHARED_LIBRARY)

jni目录结构

│  Android.mk
│  Application.mk
│  common.cpp
│  common.h
│  common_platform.cpp
│  common_platform.h
│  effects.cpp
│  libfmod.so
│  libfmodL.so
│  org_fmod_example_QQVoice.h
│  QQVoice.cpp
└─incfmod.hfmod.hppfmod_codec.hfmod_common.hfmod_dsp.hfmod_dsp_effects.hfmod_errors.hfmod_output.h

转载于:https://www.cnblogs.com/cj5785/p/10664668.html

NDK学习笔记-使用现有so动态库相关推荐

  1. NDK学习笔记:一起来变萝莉音!FMOD学习总结(下)

    NDK学习笔记:一起来变萝莉音!FMOD学习总结(下) 一.创建自己的变音demo 上一节我已经能够在AndroidStudio上跑起了fmod的基础教程.还有疑问的同学可以重新阅读跟着来跑一次.这章 ...

  2. NDK学习笔记:RtmpPusher之利用rtmpdump推h264/aac码流

    NDK学习笔记:RtmpPusher之利用rtmpdump推h264/aac码流 本篇将是 RtmpPusher 的最后一篇.在之前的3篇文章里,我们已经把原生的视频YUV格式编码成h264,把音频的 ...

  3. NDK学习笔记:FFmpeg音视频同步3(你追我赶,升级ffmpeg/libyuv支持neon)

    NDK学习笔记:FFmpeg音视频同步3 本篇内容说多不多,但如果要说得明明白白的,可能就有点难度了.所以我决定把我的调试过程日志都呈现出来,方便大家理解.继上一篇文末,我们学习到了什么是DTS/PT ...

  4. Javaweb学习笔记(JSP标准标签库)

    Javaweb学习笔记(JSP标准标签库) JSTL入门 安装和测试JSTL JSTL中的Core标签库 < c:out>标签 标签 标签 < c:catch>标签 标签 标签 ...

  5. 【Android NDK 开发】Android.mk 配置动态库 ( Android Studio 配置动态库 | 动态库加载版本限制 | 本章仅做参考推荐使用 CMake 配置动态库 )

    文章目录 I . Android Studio 中使用 Android.mk 配置动态库 总结 II . 第三方动态库来源 III . 配置 Android.mk 构建脚本路径 IV . 预编译 第三 ...

  6. 梓益C语言学习笔记之链表&动态内存&文件

    梓益C语言学习笔记之链表&动态内存&文件 一.定义: 链表是一种物理存储上非连续,通过指针链接次序,实现的一种线性存储结构. 二.特点: 链表由一系列节点(链表中每一个元素称为节点)组 ...

  7. mysql 临时表 事务_MySQL学习笔记十:游标/动态SQL/临时表/事务

    逆天十三少 发表于:2020-11-12 08:12 阅读: 90次 这篇教程主要讲解了MySQL学习笔记十:游标/动态SQL/临时表/事务,并附有相关的代码样列,我觉得非常有帮助,现在分享出来大家一 ...

  8. NDK学习笔记:FFmpeg解压MP34提取音频PCM(swrContext、swr_alloc_set_opts)

    NDK学习笔记:FFmpeg解压MP34提取音频PCM 承接 FFmpeg解压MP4提取视频YUV ,这次我们需要提取的是音频原始数据PCM.代码流程大同小异,主要区别就是AVFrame->PC ...

  9. NDK学习笔记:JNI调用Java层方法创建Native的AudioTrack播放PCM(方法签名,CallXXXMethod)

    NDK学习笔记:JNI调用Java层方法创建Native的AudioTrack播放PCM 题目有点复杂,不过确实就是那么回事.这章想记录的内容比较多,先列出来: native static 与 nat ...

最新文章

  1. 50个关于IPython的使用技巧,get起来!
  2. python怎么做界面自动化_mac+python3+selenium做pc的界面自动化测试
  3. json如何把键名作为变量?
  4. delphi edit里面的文字如何添加下划线_标题设计如何处理更吸引人?来看设计高手的实用技巧...
  5. 大数据技术 学习之旅_数据-数据科学之旅的起点
  6. 用户画像之Spark ML实现
  7. UI设计实用排版法则,优秀可临摹案例,剖析设计实例
  8. PHP多线程(Pthread初探)
  9. 理解Scala - 核心规则
  10. flex 还有人用么_那些北京人用过的老物件,能认出10个说明您老了!
  11. 华为交换机基本配置,秒看秒懂~~~
  12. 动易html编辑器,动易后台编辑器在IE8下无效的解决方法
  13. 股票投资的基本面量化
  14. jadx重新打包_Android改机系列:一.Android一键新机原理刨析
  15. win10无法装载iso文件_win10专业版系统无法加载iso文件怎么办
  16. Conflux DAO 社区技术委员会成立 助力生态繁荣发展
  17. 对计算机的认识与感想
  18. Java笔试题(一)单选题
  19. 《乡土中国》读书笔记
  20. python中super().__init__()

热门文章

  1. 计算机科学与技术专业《计算机网络原理》课程实验指导书,计算机科学导论,课程实验指导书解读.pdf...
  2. 华中师范大学邮箱matlab,18春[华中师范大学]华师《Matlab基础与应用》在线作业1(100分)...
  3. es springboot 不设置id_springboot整合ES_文档ID删除
  4. 计算机三种引用方式,单元格的引用方式有哪几种
  5. 热敏电阻温度特性曲线_热敏电阻与体温计的应用关系
  6. java编译找不到符号_关于久违的Javac,编译出现“找不到符号”
  7. java飞鸽传书_feige 飞鸽传书源代码java 实现不错的联系网络编程的资料飞鸽传书的GUI(java实现) - 下载 - 搜珍网...
  8. ac ap方案 华为_华为无线_AC+AP小型无线网络配置实验_v1
  9. C++统计微妙级时间消耗(chrono)
  10. ruby array_Ruby中带有示例的Array.select方法