- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在为我的 Android 手机开发一个应用程序,我正在尝试启用 Wifi 热点。我正在使用 Qt 5.4.1,所以我用 C++ 开发。由于在 NDK 中没有任何函数可以执行此操作,因此我使用 JNI 来调用 Java 方法。
我的 java 代码是(感谢 Ashish Sahu 在 stackoverflow thread 中的回答):
package org.app.test;
import android.content.*;
import android.net.wifi.*;
import java.lang.reflect.*;
//Class that handles Wifi Hotspot (access point) configuration
public class ApManager {
//Is Wifi hotspot on or off ?
public static boolean isApOn(Context context) {
WifiManager wifimanager = (WifiManager) context.getSystemService(context.WIFI_SERVICE);
try {
Method method = wifimanager.getClass().getDeclaredMethod("isWifiApEnabled");
method.setAccessible(true);
return (Boolean) method.invoke(wifimanager);
}
catch (Throwable ignored) {}
return false;
}
//Turn Wifi hotspot on or off
public static boolean configApState(Context context, boolean b) {
WifiManager wifimanager = (WifiManager) context.getSystemService(context.WIFI_SERVICE);
WifiConfiguration wificonfiguration = null;
try {
//if Wifi is on, turn it off
if(isApOn(context)) {
wifimanager.setWifiEnabled(false);
}
Method method = wifimanager.getClass().getMethod("setWifiApEnabled", WifiConfiguration.class, boolean.class);
method.invoke(wifimanager, wificonfiguration, b);
return true;
}
catch(Exception e) {
e.printStackTrace();
}
return false;
}
}
C++ 代码示例:
setWifiApEnabled(QAndroidJniObject context, bool b)
{
return QAndroidJniObject::callStaticMethod<jboolean>("org/app/test/ApManager"
, "configApState"
, "(Ljava/lang/Object;Z)Z" //Or (Landroid/content/Context;Z)Z ???
, context.object<jobject>()
, b);
}
但是现在我遇到了一个问题;如何获取参数 context
在我调用它时传递给函数 setWifiApEnabled(context, b)
?
我有点迷茫,我读了一些关于这个问题的帖子 (like this one),但我不完全理解回答者的意思。
你能帮我解决这个问题吗?
编辑:我在这个 stackoverflow thread 上找到了一种获取上下文的方法:
interface = QApplication::platformNativeInterface();
activiti = (jobject)interface->nativeResourceForIntegration("QtActivity");
at = new QAndroidJniObject(activiti);
appctx = at->callObjectMethod("getApplicationContext", "()Landroid/content/Context;");
if(appctx.isValid()) qDebug() << "I am valid !";
else qDebug() << "I ain't valid !";
appctx
有效,但 Wifi 热点未启用,我无法获取其状态。
编辑 2::我成功地使用 Android Studio 在 Java 中启用了 Wifi 热点。代码如下:
WifiApManager.java :
public class WifiApManager {
private WifiManager wifiMan;
protected Method setWifiApEnabledMethod, isWifiApEnabledMethod;
protected final static int MAX_ITER = 10;
public WifiApManager(WifiManager wifiMan) {
this.wifiMan = wifiMan;
getHiddenMethods();
}
private void getHiddenMethods() {
try {
setWifiApEnabledMethod = wifiMan.getClass().getMethod("setWifiApEnabled", WifiConfiguration.class, boolean.class);
isWifiApEnabledMethod = wifiMan.getClass().getMethod("isWifiApEnabled");
} catch (NoSuchMethodException e) {
e.printStackTrace();
}
}
public boolean isWifiApEnabled() {
try {
return (Boolean)isWifiApEnabledMethod.invoke(wifiMan);
} catch (Exception e) {
return false;
}
}
public boolean isWifiEnabled() {
return wifiMan.isWifiEnabled();
}
public boolean setWifiApEnabled(WifiConfiguration conf, boolean enabled) {
try {
return (Boolean) setWifiApEnabledMethod.invoke(wifiMan, conf, true);
} catch (Exception e) {
return false;
}
}
public boolean toggleWifi(String ssid) {
// WifiConfiguration creation:
WifiConfiguration conf = new WifiConfiguration();
conf.SSID = ssid;
conf.allowedAuthAlgorithms.set(WifiConfiguration.AuthAlgorithm.OPEN);
// If AP Wifi is enabled, disables it and returns:
if(isWifiApEnabled()) {
//setWifiApEnabled(null, false); Won't work, see two further lines
wifiMan.setWifiEnabled(true);
wifiMan.setWifiEnabled(false);
int maxIter = MAX_ITER;
while (isWifiApEnabled() && maxIter-- >= 0) {
try {Thread.sleep(500);} catch (Exception e) {}
}
return isWifiApEnabled();
}
// If standard Wifi is enabled, disables it:
if (isWifiEnabled()) {
if (wifiMan.setWifiEnabled(false)) {
int maxIter = MAX_ITER;
while (wifiMan.isWifiEnabled() && maxIter-- >= 0) {
try {Thread.sleep(500);} catch (Exception e) {}
}
}
if (isWifiEnabled()) {
return false;
}
}
// Enables AP Wifi
try {
if (! setWifiApEnabled(conf, true)) {
System.out.println("setWifiApEnabledMethod failed.");
return false;
}
int maxIter = MAX_ITER;
while (! isWifiApEnabled() && maxIter-- > 0) {
try {Thread.sleep(500);} catch (Exception e) {}
}
} catch(Exception e) {
e.printStackTrace();
return false;
}
return true;
}
}
主类:
public class AndroidMenusActivity extends Activity implements OnClickListener {
private WifiApManager wifiMan;
private ToggleButton wifiButton;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
wifiMan = new WifiApManager((WifiManager) this.getSystemService(Context.WIFI_SERVICE));
setContentView(R.layout.activity_android_menus);
makeUI();
}
private void makeUI() {
LinearLayout subLayout = (LinearLayout) findViewById(R.id.subLayout);
wifiButton = new ToggleButton(this);
wifiButton.setTextOn("Disable Wifi");
wifiButton.setTextOff("Enable AP Wifi");
wifiButton.setChecked(wifiMan.isWifiApEnabled());
wifiButton.setOnClickListener(this);
subLayout.addView(wifiButton);
}
@Override
public void onClick(View sender) {
if (!wifiButton.equals(sender))
return;
AsyncTask<Object, Void, Boolean> task = new AsyncTask<Object, Void, Boolean>() {
private ToggleButton bt;
private WifiApManager wm;
@Override
protected Boolean doInBackground(Object... args) {
bt = (ToggleButton) args[0];
wm = (WifiApManager) args[1];
return wm.toggleWifi("test.com");
}
@Override
protected void onPostExecute (Boolean result) {
bt.setChecked(result.booleanValue());
bt.setEnabled(true);
}
};
wifiButton.setEnabled(false);
task.execute(wifiButton, wifiMan);
}
}
但是我找不到在 C++ 中做同样事情的方法,有什么帮助吗?
最佳答案
只是为了将上下文对象从 C++ 传输到 JNI,使用这样的自动注入(inject):
QtAndroid::androidActivity().callStaticMethod<void>(
"com/company/project/MyApp",
"myFunc", // method name
"(Landroid/content/Context;)V" // auto-injection
);
关于java - 如何将 android.content.Context 参数(从 native 代码)发送到 Java fct(通过 JNI),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30762826/
回答this question about overload resolution with enums时出现了这个问题. 虽然 long long 的情况肯定是 MSVC2012NovCTP 中的错
回答this question about overload resolution with enums时出现这个问题. 虽然 long long 的情况绝对是 MSVC2012NovCTP 中的一个
我努力让 lambda 函数通过引用返回一个值,而不复制引用值。我下面的代码示例说明了这个问题。它编译并运行正常,但是使用“//”注释行而不是上面的行,它没有。我找到了两种解决方法(均在我的示例中进行
我在"Artificial Intelligence: A modern approach"中遇到过代码存储库下面是我以前从未见过的代码: def __init__(self, state, pa
我正在 Google Apps 脚本中运行某个函数,有时它会超时并停止运行。我想添加一些东西来检查(连续或每 x 秒一次)我的函数是否正在运行,如果没有,则启动它。对此最好的解决方案是什么? 基本上,
[dcl.fct.default]/10 : A virtual function call (10.3) uses the default arguments in the declaration
[dcl.fct.default]/3 (重点是我的): A default argument shall be specified only in the parameter-declaration
我知道,例如 void *(*myFuncName)(void*) 是一个函数指针,它接受并返回 void*。 这是一个有两个参数的指针吗?void 指针是该类型的另一个返回 void* 和 void
DR 2145 中提出的更改负责[dcl.fct.def.general]/2的变化从C++14到C++17,如下: C++14: The declarator in a function-defin
我正在为我的 Android 手机开发一个应用程序,我正在尝试启用 Wifi 热点。我正在使用 Qt 5.4.1,所以我用 C++ 开发。由于在 NDK 中没有任何函数可以执行此操作,因此我使用 JN
C++14 中的§12.3.2 [class.conv.fct]/1: A member function of a class X having no parameters with a name
我是一名优秀的程序员,十分优秀!