我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Android无障碍服务performAction怎么调用

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Android无障碍服务performAction怎么调用

这篇文章主要介绍“Android无障碍服务performAction怎么调用”,在日常操作中,相信很多人在Android无障碍服务performAction怎么调用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Android无障碍服务performAction怎么调用”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

无障碍服务可以模拟一些用户操作,无障碍可以处理的对象,通过类 AccessibilityNodeInfo 表示,通过无障碍服务,可以通过它的 performAction 方法来触发一些 action ,包括:

ACTION_FOCUS // 获取焦点ACTION_CLEAR_FOCUS // 清除焦点ACTION_SELECT // 选中ACTION_CLEAR_SELECTION // 清除选中状态ACTION_ACCESSIBILITY_FOCUS // 无障碍焦点ACTION_CLEAR_ACCESSIBILITY_FOCUS // 清除无障碍焦点ACTION_CLICK // 点击ACTION_LONG_CLICK // 长按ACTION_NEXT_AT_MOVEMENT_GRANULARITY // 下一步移动ACTION_PREVIOUS_AT_MOVEMENT_GRANULARITY // 上一步移动ACTION_NEXT_HTML_ELEMENT // 下一个 html 元素ACTION_PREVIOUS_HTML_ELEMENT // 上一个 html 元素ACTION_SCROLL_FORWARD // 向前滑动ACTION_SCROLL_BACKWARD // 向后滑动

他们都可以通过performAction方法进行处理:

// in AccessibilityNodeInfopublic boolean performAction(int action) {    enforceSealed();    if (!canPerformRequestOverConnection(mConnectionId, mWindowId, mSourceNodeId)) {        return false;    }    AccessibilityInteractionClient client = AccessibilityInteractionClient.getInstance();    return client.performAccessibilityAction(mConnectionId, mWindowId, mSourceNodeId,            action, null);}

在这个方法中,第一步是检查 perform 是否可以通过 connection 请求,这里 connection 检查是根据通过 binder 通信传递过来的 id 检查连接是否正常。 然后通过AccessibilityInteractionClient对象,调用它的performAccessibilityAction方法去进行实际操作的。

AccessibilityInteractionClient

这个类是一个执行可访问性交互的单例,它可以根据 View 的快照查询远程的 View 层次结构,以及通过 View 层次结构,来请求对 View 执行某项操作。

基本原理:内容检索 API 从客户端的角度来看是同步的,但在内部它们是异步的。客户端线程调用系统请求操作并提供回调以接收结果,然后等待该结果的超时。系统强制执行安全性并将请求委托给给定的视图层次结构, 在该视图层次结构中发布消息(来自 Binder 线程),描述 UI 线程要执行的内容,其结果是通过上述回调传递的。但是,被阻塞的客户端线程和目标视图层次结构的主 UI 线程可以是同一个线程,例如无障碍服务和 Activity 在同一个进程中运行,因此它们在同一个主线程上执行。 在这种情况下,检索将会失败,因为 UI 线程在等待检索结果,会导致阻塞。 为了避免在进行调用时出现这种情况,客户端还会传递其进程和线程 ID,以便访问的视图层次结构可以检测发出请求的客户端是否正在其主 UI 线程中运行。 在这种情况下,视图层次结构,特别是对它执行 IPC 的绑定线程,不会发布要在 UI 线程上运行的消息,而是将其传递给单例交互客户端,通过该客户端发生所有交互,后者负责执行开始等待通过回调传递的异步结果之前的消息。在这种情况下,已经收到预期的结果,因此不执行等待。

上面是官方备注的描述,大概意思最好不要在主线程执行检索操作。

继续跟进它的performAccessibilityAction方法:

    public boolean performAccessibilityAction(int connectionId, int accessibilityWindowId,            long accessibilityNodeId, int action, Bundle arguments) {        try {            IAccessibilityServiceConnection connection = getConnection(connectionId);            if (connection != null) {                final int interactionId = mInteractionIdCounter.getAndIncrement();                final long identityToken = Binder.clearCallingIdentity();                final boolean success;                try {                    success = connection.performAccessibilityAction(                            accessibilityWindowId, accessibilityNodeId, action, arguments,                            interactionId, this, Thread.currentThread().getId()); // 【*】                } finally {                    Binder.restoreCallingIdentity(identityToken);                }                if (success) {                    return getPerformAccessibilityActionResultAndClear(interactionId);                }            }        } catch (RemoteException re) {            Log.w(LOG_TAG, "Error while calling remote performAccessibilityAction", re);        }        return false;    }

这里通过 getConnection(connectionId) 获取了一个 IAccessibilityServiceConnection 。

public static IAccessibilityServiceConnection getConnection(int connectionId) {    synchronized (sConnectionCache) {        return sConnectionCache.get(connectionId);    }}

这里的 sConnectionCache 通过 AccessibilityInteractionClient 的addConnection添加数据的,addConnection在 AccessbilityService 创建初始化时调用的:

case DO_INIT: {    mConnectionId = message.arg1;    SomeArgs args = (SomeArgs) message.obj;    IAccessibilityServiceConnection connection = (IAccessibilityServiceConnection) args.arg1;    IBinder windowToken = (IBinder) args.arg2;    args.recycle();    if (connection != null) {        AccessibilityInteractionClient.getInstance(mContext).addConnection(mConnectionId, connection);        mCallback.init(mConnectionId, windowToken);        mCallback.onServiceConnected();    } else {        AccessibilityInteractionClient.getInstance(mContext).removeConnection(mConnectionId);        mConnectionId = AccessibilityInteractionClient.NO_ID;        AccessibilityInteractionClient.getInstance(mContext).clearCache();        mCallback.init(AccessibilityInteractionClient.NO_ID, null);    }    return;}

也就是说,在 AccessbilityService 创建时,会将一个表示连接的对象存到 AccessibilityInteractionClient 的连接缓存中。

IAccessibilityServiceConnection

它是 AccessibilityManagerService 向 AccessbilityService 暴露的 AIDL 接口,提供给 AccessbilityService 调用AccessibilityManagerService 的能力。 上面的 performAction 流程中,调用到了 connection 的performAccessibilityAction方法。 而 IAccessibilityServiceConnection 有两个实现类,AccessibilityServiceConnectionImplAbstractAccessibilityServiceConnection,前者都是空实现,显然不是我们要调用到的地方,后者的performAccessibilityAction

@Overridepublic boolean performAccessibilityAction(int accessibilityWindowId,        long accessibilityNodeId, int action, Bundle arguments, int interactionId,        IAccessibilityInteractionConnectionCallback callback, long interrogatingTid)        throws RemoteException {    final int resolvedWindowId;    synchronized (mLock) {        if (!hasRightsToCurrentUserLocked()) {            return false;        }        resolvedWindowId = resolveAccessibilityWindowIdLocked(accessibilityWindowId);        if (!mSecurityPolicy.canGetAccessibilityNodeInfoLocked(                mSystemSupport.getCurrentUserIdLocked(), this, resolvedWindowId)) {            return false;        }    }    if (!mSecurityPolicy.checkAccessibilityAccess(this)) {        return false;    }    return performAccessibilityActionInternal(            mSystemSupport.getCurrentUserIdLocked(), resolvedWindowId, accessibilityNodeId,            action, arguments, interactionId, callback, mFetchFlags, interrogatingTid);}

最后的一行调用:

    private boolean performAccessibilityActionInternal(int userId, int resolvedWindowId, long accessibilityNodeId, int action, Bundle arguments, int interactionId, IAccessibilityInteractionConnectionCallback callback, int fetchFlags, long interrogatingTid) {        RemoteAccessibilityConnection connection;        IBinder activityToken = null;        // 同步获取 connection        synchronized (mLock) {            connection = mA11yWindowManager.getConnectionLocked(userId, resolvedWindowId);            if (connection == null)  {                return false;            }            final boolean isA11yFocusAction = (action == ACTION_ACCESSIBILITY_FOCUS) || (action == ACTION_CLEAR_ACCESSIBILITY_FOCUS);            if (!isA11yFocusAction) {                final WindowInfo windowInfo = mA11yWindowManager.findWindowInfoByIdLocked(resolvedWindowId);                if (windowInfo != null) activityToken = windowInfo.activityToken;            }            final AccessibilityWindowInfo a11yWindowInfo = mA11yWindowManager.findA11yWindowInfoByIdLocked(resolvedWindowId);            if (a11yWindowInfo != null && a11yWindowInfo.isInPictureInPictureMode() && mA11yWindowManager.getPictureInPictureActionReplacingConnection() != null && !isA11yFocusAction) {                connection = mA11yWindowManager.getPictureInPictureActionReplacingConnection();            }        }        // 通过 connection 调用到远程服务的performAccessibilityAction        final int interrogatingPid = Binder.getCallingPid();        final long identityToken = Binder.clearCallingIdentity();        try {            // 无论操作是否成功,它都是由用户操作的无障碍服务生成的,因此请注意用户Activity。            mPowerManager.userActivity(SystemClock.uptimeMillis(), PowerManager.USER_ACTIVITY_EVENT_ACCESSIBILITY, 0);            if (action == ACTION_CLICK || action == ACTION_LONG_CLICK) {                mA11yWindowManager.notifyOutsideTouch(userId, resolvedWindowId);            }            if (activityToken != null) {                LocalServices.getService(ActivityTaskManagerInternal.class).setFocusedActivity(activityToken);            }            connection.getRemote().performAccessibilityAction(accessibilityNodeId, action, arguments, interactionId, callback, fetchFlags, interrogatingPid, interrogatingTid);        } catch (RemoteException re) {            if (DEBUG) {                Slog.e(LOG_TAG, "Error calling performAccessibilityAction: " + re);            }            return false;        } finally {            Binder.restoreCallingIdentity(identityToken);        }        return true;    }

在这个方法中,通过 connection 调用到了远端的 performAccessibilityAction 方法。

关键的一行是:

connection.getRemote().performAccessibilityAction(accessibilityNodeId, action, arguments, interactionId, callback, fetchFlags, interrogatingPid, interrogatingTid);

这里的 connection 类型定义成了RemoteAccessibilityConnection

RemoteAccessibilityConnection

RemoteAccessibilityConnection 是AccessibilityWindowManager的内部类,它的getRemote()返回类型是IAccessibilityInteractionConnection

AccessibilityWindowManager

此类为 AccessibilityManagerService 提供 API 来管理 AccessibilityWindowInfo 和 WindowInfos。

IAccessibilityInteractionConnection

这是一个 AIDL 中定义的接口,用来进行 给定 window 中 AccessibilityManagerService 和 ViewRoot 之间交互的接口。

也就是说getRemote(). performAccessibilityAction(...)最终来到了 ViewRootImpl 中。

AccessibilityInteractionConnection

ViewRootImpl 中存在一个内部类AccessibilityInteractionConnection,它是这个 ViewAncestor 提供给 AccessibilityManagerService 的一个接口,后者可以与这个 ViewAncestor 中的视图层次结构进行交互。

它的performAccessibilityAction实现是:

@Overridepublic void performAccessibilityAction(long accessibilityNodeId, int action, Bundle arguments, int interactionId, IAccessibilityInteractionConnectionCallback callback, int flags, int interrogatingPid, long interrogatingTid) {    ViewRootImpl viewRootImpl = mViewRootImpl.get();    if (viewRootImpl != null && viewRootImpl.mView != null) {        viewRootImpl.getAccessibilityInteractionController().performAccessibilityActionClientThread(accessibilityNodeId, action, arguments,                    interactionId, callback, flags, interrogatingPid, interrogatingTid);    } else {        // We cannot make the call and notify the caller so it does not wait.        try {            callback.setPerformAccessibilityActionResult(false, interactionId);        } catch (RemoteException re) {                    }    }}

内部又是通过代理调用 ,ViewRootImpl 的 getAccessibilityInteractionController() 返回了一个 AccessibilityInteractionController 对象。

AccessibilityInteractionController

它的 performAccessibilityActionClientThread :

public void performAccessibilityActionClientThread(long accessibilityNodeId, int action,        Bundle arguments, int interactionId,        IAccessibilityInteractionConnectionCallback callback, int flags, int interrogatingPid,        long interrogatingTid) {    Message message = mHandler.obtainMessage();    message.what = PrivateHandler.MSG_PERFORM_ACCESSIBILITY_ACTION;    message.arg1 = flags;    message.arg2 = AccessibilityNodeInfo.getAccessibilityViewId(accessibilityNodeId);    SomeArgs args = SomeArgs.obtain();    args.argi1 = AccessibilityNodeInfo.getVirtualDescendantId(accessibilityNodeId);    args.argi2 = action;    args.argi3 = interactionId;    args.arg1 = callback;    args.arg2 = arguments;    message.obj = args;    scheduleMessage(message, interrogatingPid, interrogatingTid, CONSIDER_REQUEST_PREPARERS);}

组装了一个 message ,并通过 scheduleMessage 方法去执行:

private void scheduleMessage(Message message, int interrogatingPid, long interrogatingTid, boolean ignoreRequestPreparers) {    if (ignoreRequestPreparers || !holdOffMessageIfNeeded(message, interrogatingPid, interrogatingTid)) {        if (interrogatingPid == mMyProcessId && interrogatingTid == mMyLooperThreadId                && mHandler.hasAccessibilityCallback(message)) {            AccessibilityInteractionClient.getInstanceForThread(                    interrogatingTid).setSameThreadMessage(message);        } else {            if (!mHandler.hasAccessibilityCallback(message) && Thread.currentThread().getId() == mMyLooperThreadId) {                mHandler.handleMessage(message);            } else {                mHandler.sendMessage(message);            }        }    }}

这里实际上,如果是在主线程,则处理消息,如果不是,则发送消息到主线程处理。handler 的类型是 PrivateHandler ,在 AccessibilityInteractionController 内部定义。

它的处理消息方法的实现是:

@Overridepublic void handleMessage(Message message) {    final int type = message.what;    switch (type) {        // ...        case MSG_PERFORM_ACCESSIBILITY_ACTION: {            performAccessibilityActionUiThread(message);        } break;        // ...        default:            throw new IllegalArgumentException("Unknown message type: " + type);    }}

执行到了 performAccessibilityActionUiThread(message); :

    private void performAccessibilityActionUiThread(Message message) {        // ...         boolean succeeded = false;        try {            // ...            final View target = findViewByAccessibilityId(accessibilityViewId);            if (target != null && isShown(target)) {                mA11yManager.notifyPerformingAction(action);                if (action == R.id.accessibilityActionClickOnClickableSpan) {                    // 单独处理这个 hidden action                    succeeded = handleClickableSpanActionUiThread(target, virtualDescendantId, arguments);                } else {                    AccessibilityNodeProvider provider = target.getAccessibilityNodeProvider();                    if (provider != null) {                        succeeded = provider.performAction(virtualDescendantId, action, arguments);                    } else if (virtualDescendantId == AccessibilityNodeProvider.HOST_VIEW_ID) {                        succeeded = target.performAccessibilityAction(action, arguments);                    }                }                mA11yManager.notifyPerformingAction(0);            }        } finally {            try {                mViewRootImpl.mAttachInfo.mAccessibilityFetchFlags = 0;                callback.setPerformAccessibilityActionResult(succeeded, interactionId);            } catch (RemoteException re) {                            }        }    }

在这个流程中,分为三种情况去真正执行 performAction :

1. action == R.id.accessibilityActionClickOnClickableSpan

private boolean handleClickableSpanActionUiThread(        View view, int virtualDescendantId, Bundle arguments) {    Parcelable span = arguments.getParcelable(ACTION_ARGUMENT_ACCESSIBLE_CLICKABLE_SPAN);    if (!(span instanceof AccessibilityClickableSpan)) {        return false;    }    // Find the original ClickableSpan if it's still on the screen    AccessibilityNodeInfo infoWithSpan = null;    AccessibilityNodeProvider provider = view.getAccessibilityNodeProvider();    if (provider != null) {        infoWithSpan = provider.createAccessibilityNodeInfo(virtualDescendantId);    } else if (virtualDescendantId == AccessibilityNodeProvider.HOST_VIEW_ID) {        infoWithSpan = view.createAccessibilityNodeInfo();    }    if (infoWithSpan == null) {        return false;    }    // Click on the corresponding span    ClickableSpan clickableSpan = ((AccessibilityClickableSpan) span).findClickableSpan(            infoWithSpan.getOriginalText());    if (clickableSpan != null) {        clickableSpan.onClick(view);        return true;    }    return false;}

2. View. AccessibilityNodeProvider != null

当能够通过 View 获取到 AccessibilityNodeProvider 对象是,通过它的 performAction 方法,去执行真正的调用,它的真正调用在 AccessibilityNodeProviderCompat中,这个 Compat 的实现在ExploreByTouchHelper中的内部类MyNodeProvider中:

@Overridepublic boolean performAction(int virtualViewId, int action, Bundle arguments) {    return ExploreByTouchHelper.this.performAction(virtualViewId, action, arguments);}

在 ExploreByTouchHelper 中继续查看:

boolean performAction(int virtualViewId, int action, Bundle arguments) {    switch (virtualViewId) {        case HOST_ID:            return performActionForHost(action, arguments);        default:            return performActionForChild(virtualViewId, action, arguments);    }}
private boolean performActionForHost(int action, Bundle arguments) {    return ViewCompat.performAccessibilityAction(mHost, action, arguments);}private boolean performActionForChild(int virtualViewId, int action, Bundle arguments) {    switch (action) {        case AccessibilityNodeInfoCompat.ACTION_ACCESSIBILITY_FOCUS:            return requestAccessibilityFocus(virtualViewId);        case AccessibilityNodeInfoCompat.ACTION_CLEAR_ACCESSIBILITY_FOCUS:            return clearAccessibilityFocus(virtualViewId);        case AccessibilityNodeInfoCompat.ACTION_FOCUS:            return requestKeyboardFocusForVirtualView(virtualViewId);        case AccessibilityNodeInfoCompat.ACTION_CLEAR_FOCUS:            return clearKeyboardFocusForVirtualView(virtualViewId);        default:            return onPerformActionForVirtualView(virtualViewId, action, arguments);    }}

前者调用到了 ViewCompat :

public static boolean performAccessibilityAction(@NonNull View view, int action,        Bundle arguments) {    if (Build.VERSION.SDK_INT >= 16) {        return view.performAccessibilityAction(action, arguments);    }    return false;}

然后是 View 的 :

public boolean performAccessibilityAction(int action, Bundle arguments) {  if (mAccessibilityDelegate != null) {      return mAccessibilityDelegate.performAccessibilityAction(this, action, arguments);  } else {      return performAccessibilityActionInternal(action, arguments);  }}

mAccessibilityDelegate.performAccessibilityAction的实现是:

public boolean performAccessibilityAction(View host, int action, Bundle args) {    return host.performAccessibilityActionInternal(action, args);}

也是调用到了 View 的performAccessibilityActionInternal 。 performAccessibilityActionInternal 的实现是:

// in View.javapublic boolean performAccessibilityActionInternal(int action, Bundle arguments) {    if (isNestedScrollingEnabled()            && (action == AccessibilityNodeInfo.ACTION_SCROLL_BACKWARD            || action == AccessibilityNodeInfo.ACTION_SCROLL_FORWARD            || action == R.id.accessibilityActionScrollUp            || action == R.id.accessibilityActionScrollLeft            || action == R.id.accessibilityActionScrollDown            || action == R.id.accessibilityActionScrollRight)) {        if (dispatchNestedPrePerformAccessibilityAction(action, arguments)) {            return true;        }    }    switch (action) {        case AccessibilityNodeInfo.ACTION_CLICK: {            if (isClickable()) {                performClickInternal();                return true;            }        } break;        case AccessibilityNodeInfo.ACTION_LONG_CLICK: {            if (isLongClickable()) {                performLongClick();                return true;            }        } break;        // ...    }    return false;}

以 AccessibilityNodeInfo.ACTION_CLICK 为例,内部调用是:

private boolean performClickInternal() {    // Must notify autofill manager before performing the click actions to avoid scenarios where    // the app has a click listener that changes the state of views the autofill service might    // be interested on.    notifyAutofillManagerOnClick();    return performClick();}

这样就调用到了 View 的点击事件。

3. View. AccessibilityNodeProvider == null && virtualDescendantId == AccessibilityNodeProvider.HOST_VIEW_ID

target.performAccessibilityAction(action, arguments);

这里 target 是个 View, 也是走的 View 的 performAccessibilityAction ,和上面流程一样。

View 的 performClick 方法是同步的还是异步的?

public boolean performClick() {    // We still need to call this method to handle the cases where performClick() was called    // externally, instead of through performClickInternal()    notifyAutofillManagerOnClick();    final boolean result;    final ListenerInfo li = mListenerInfo;    if (li != null && li.mOnClickListener != null) {        playSoundEffect(SoundEffectConstants.CLICK);        li.mOnClickListener.onClick(this);        result = true;    } else {        result = false;    }    sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_CLICKED);    notifyEnterOrExitForAutoFillIfNeeded(true);    return result;}

同步的。

到此,关于“Android无障碍服务performAction怎么调用”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Android无障碍服务performAction怎么调用

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Android无障碍服务performAction怎么调用

这篇文章主要介绍“Android无障碍服务performAction怎么调用”,在日常操作中,相信很多人在Android无障碍服务performAction怎么调用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答
2023-07-02

android无障碍服务功能怎么实现

Android无障碍服务功能可以通过编写无障碍服务来实现。以下是实现无障碍服务功能的一般步骤:1. 创建一个继承自AccessibilityService的类,例如 MyAccessibilityService。2. 在AndroidMan
2023-10-07

Android无障碍监听通知怎么实现

本篇内容主要讲解“Android无障碍监听通知怎么实现”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Android无障碍监听通知怎么实现”吧!监听通知Android 中的 Accessibili
2023-07-02

Android无障碍服务执行全局动作与手势

学会Android的无障碍服务可以进行很多自动化的操作,下面就一起来看下如何玩转无障碍服务 1.创建一个继承AccessibilityService 的类 class MyAccessibilityService : Accessibili
2022-06-06

Android无需权限调用系统相机拍照怎么实现

这篇“Android无需权限调用系统相机拍照怎么实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Android无需权限调用
2023-07-05

云服务器怎么调用

云服务器可以通过以下步骤进行调用:1. 登录云服务器的管理控制台。这通常需要使用您的云服务提供商提供的用户名和密码进行身份验证。2. 在管理控制台中,找到您想要调用的云服务器实例,并确保该实例处于运行状态。3. 确定您要使用的访问方式。云服
2023-09-27

云服务器怎么调用

云服务器调用云服务器通过API调用、CLI工具和ControlPanel提供便捷调用方式。API调用适合有编程能力的开发者,使用SDK或直接发送HTTP请求。CLI工具适用于命令行操作,使用官方或第三方工具。ControlPanel为非技术人员提供图形化管理界面,可以使用云服务提供商的ControlPanel或第三方ControlPanel。选择调用方式应考虑开发者技能、功能要求和环境限制。最佳实践包括使用安全凭证、处理错误、使用速率限制器、监控API调用和保持更新。
云服务器怎么调用
2024-04-11

调用阿里云服务器怎么调用网络

简介在当今数字化时代,云计算已经成为企业和个人构建和管理IT基础设施的重要选择之一。作为全球领先的云计算服务提供商,阿里云提供了强大的云服务器(ECS)服务,让用户可以轻松地调用和管理自己的服务器资源。本文将介绍如何正确地调用阿里云服务器,并充分利用其提供的网络功能。1.使用阿里云API进行调用阿里云提供了一套完善的A
调用阿里云服务器怎么调用网络
2024-01-17

SpringCloud远程服务怎么调用

这篇文章主要介绍SpringCloud远程服务怎么调用,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!笔记在微服务中,若想要使用远程调用,需要引入spring-cloud-starter-openfeign(在使用注册
2023-06-21

Win7怎么删除无用服务

今天小编给大家分享一下Win7怎么删除无用服务的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。删除方法:1、点击开始→运行输入
2023-06-28

云服务器调用本地摄像头怎么调

如果您要调用云服务器上的摄像头,需要先在本地安装相应的摄像头驱动程序,下载并解压缩到指定路径,例如c:\DocumentsandSettings\Adobe\AdobeCameraStudio\QuickTime/v2\lib。然后在命令行中输入以下命令来调用摄像头:clfdriver.server.yml其中:clfdriver.server.yml:指定要在本地进行操作的远程摄
2023-10-26

Springboot HTTP怎么调用其他服务

这篇“Springboot HTTP怎么调用其他服务”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Springboot HT
2023-06-29

云服务器调用本地打印机怎么调

如果您需要在云服务器上使用本地打印机,可以使用以下步骤来调节其打印设置:打开云端控制台。在“文件”>“选项”中,找到“打印服务”选项并单击。在打开的打印服务配置界面中,选择“启动打印服务”选项。在该界面中,单击“停止打印服务”按钮以停止本地打印机的打印服务。如果您是在本地计算机上使用本地打印机,您可以根据需要在界面中添加相应的配置文件以将其添加到打印服务配置中。最后,单击&ldqu
2023-10-26

轻量应用服务器带宽调整怎么调

轻量应用服务器通常是指使用更轻量级的处理器、内存和存储器件的应用服务器,例如基于Java的轻量级Web应用服务器、基于XML的轻量级数据库服务器、基于Python的轻量级机器学习服务器等。这些轻量应用服务器的处理能力比较低,通常只有几十至几百个核,而且它们所使用的CPU和内存资源都相对较少。针对轻量应用服务器的带宽需求
轻量应用服务器带宽调整怎么调
2023-10-28

WCF服务怎么利用JAVA进行调用

这篇文章给大家介绍WCF服务怎么利用JAVA进行调用,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。.生成WCF客户端使用的就是上面提到的wsimport,如果您正确的配置了java环境变量,就可以直接在控制台中使用了,
2023-05-31

轻量应用服务器带宽调整怎么调的

轻量应用服务器可以通过配置高带宽或低带宽来适应不同的应用场景。以下是一些可能需要对轻量应用服务器进行带宽调整的方法:检查硬件设置:检查所选服务器的硬件设置,以确定它是否足够处理高峰流量。可以通过调整硬件配置或优化处理器、内存、网络带宽等多种因素来提高硬件性能。自动调整:有时候轻量应用服务器需要自动调整其带宽。这可以通过配置应用程序的带宽策略来实现。例如,可以在应用程序中启用高带宽配置,或在
2023-10-26

云服务器调用本地摄像头怎么调整

如果您的云服务器上提供了视频共享服务,则可以通过在云服务器上调用本地摄像头来调整视频流。以下是一些方法:将视频从摄像头中分离出来,并保存到本地文件夹中。在云服务器上的视频共享界面上,找到要应用于本地的视频,并将其传输到本地电脑。在本地计算机上查找该视频的文件夹,并将其从摄像头上分离出来。打开本地存储设备,例如视频文件,将摄像头连接到该设备上。在本地电脑上播放该视频,并将其设置为自动播
2023-10-26

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录