act_type = act_type.lower()

时间: 2025-08-03 13:06:02 浏览: 1
<think>我们正在讨论的是字符串方法`lower()`的用法。用户提到了`act_type=act_type.lower()`,这行代码的作用是将字符串转换为小写形式。下面我将详细解释其用法和应用场景。</think> ### Python 字符串方法 `.lower()` 的用法解析 `act_type = act_type.lower()` 的作用是将字符串变量 `act_type` 的内容**统一转换为小写形式**。这是一个基础的字符串处理操作,常用于**标准化输入数据**或**消除大小写敏感问题**。 --- ### 一、核心功能与语法 #### 1. 基本语法 ```python 字符串.lower() ``` - **返回值**: 返回字符串的全小写副本(不修改原始字符串) - **非字母字符**: 数字、符号等不受影响 - **原始对象**: 不改变原字符串(字符串为不可变对象) #### 2. 典型使用场景 ```python act_type = "ReLU" # 原始值含大写 act_type = act_type.lower() # 转换后为 "relu" ``` 此时执行过程等效于: $$ \text{act\_type}_{\text{new}} = \text{lower}(\text{act\_type}_{\text{old}}) $$ --- ### 二、实际应用示例 #### 案例 1: 配置神经网络激活函数 ```python def build_layer(act_type): act_type = act_type.lower() # 标准化输入 if act_type == "relu": return nn.ReLU() elif act_type == "sigmoid": return nn.Sigmoid() else: raise ValueError(f"无效激活函数: {act_type}") # 测试(大小写输入均有效) layer1 = build_layer("ReLU") # ✅ 自动转为 "relu" layer2 = build_layer("siGMoid") # ✅ 转为 "sigmoid" ``` > **作用**: 避免因用户输入大小写不一致导致判断失效[^2] --- #### 案例 2: 文件类型校验 ```python allowed_extensions = [".jpg", ".png", ".gif"] filename = "PHOTO.JPG" if filename.lower().endswith(tuple(allowed_extensions)): print("文件格式支持") # ✅ 输出此行 ``` > **原理**: 将 `"PHOTO.JPG"` → `"photo.jpg"` 后再比对后缀[^4] --- ### 三、与其他方法的对比 | **方法** | 作用 | 示例输入 → 输出 | |---------------|--------------------|------------------------| | `.lower()` | 全转小写 | `"PyThon"` → `"python"` | | `.upper()` | 全转大写 | `"PyThon"` → `"PYTHON"` | | `.capitalize()`| 仅首字母大写 | `"pyThon"` → `"Python"` | | `casefold()` | 更激进的小写转换 | `"SS"` → `"ss"` (德语ß) | > 说明:`casefold()` 对 Unicode 特殊字符处理更严格(如德语 ß → ss) --- ### 四、注意事项 1. **非字符串类型需先转换** ```python num = 123 # num.lower() ❌ 抛出 AttributeError str(num).lower() # ✅ 先转为字符串 ``` 2. **处理特殊语言字符** ```python "İstanbul".lower() # → "i̇stanbul" (土耳其语) ``` > 某些语言(如土耳其语)的大小写转换需特殊处理[^3] ---
阅读全文

相关推荐

# Copyright 2020 Huawei Technologies Co., Ltd # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # ============================================================================ """wide and deep model""" import time import numpy as np from mindspore import nn, context from mindspore import Parameter, ParameterTuple import mindspore.common.dtype as mstype import mindspore.ops as ops from mindspore.nn import Dropout from mindspore.nn.optim import Adam, FTRL from mindspore.common.initializer import Uniform, initializer from mindspore.context import ParallelMode from mindspore.nn.wrap.grad_reducer import DistributedGradReducer from mindspore.communication.management import get_group_size np_type = np.float32 ms_type = mstype.float32 def init_method(method, shape, name, max_val=1.0): ''' parameter init method ''' if method in ['uniform']: params = Parameter(initializer( Uniform(max_val), shape, ms_type), name=name) elif method == "one": params = Parameter(initializer("ones", shape, ms_type), name=name) elif method == 'zero': params = Parameter(initializer("zeros", shape, ms_type), name=name) elif method == "normal": params = Parameter(initializer("normal", shape, ms_type), name=name) return params def init_var_dict(init_args, in_vars): ''' var init function ''' var_map = {} _, _max_val = init_args for _, item in enumerate(in_vars): key, shape, method = item if key not in var_map.keys(): if method in ['random', 'uniform']: var_map[key] = Parameter(initializer( Uniform(_max_val), shape, ms_type), name=key) elif method == "one": var_map[key] = Parameter(initializer( "ones", shape, ms_type), name=key) elif method == "zero": var_map[key] = Parameter(initializer( "zeros", shape, ms_type), name=key) elif method == 'normal': var_map[key] = Parameter(initializer( "normal", shape, ms_type), name=key) return var_map class DenseLayer(nn.Cell): """ Dense Layer for Deep Layer of WideDeep Model; Containing: activation, matmul, bias_add; Args: """ def __init__(self, input_dim, output_dim, weight_bias_init, act_str, keep_prob=0.5, use_activation=True, convert_dtype=True, drop_out=False): super(DenseLayer, self).__init__() weight_init, bias_init = weight_bias_init self.weight = init_method( weight_init, [input_dim, output_dim], name="weight") self.bias = init_method(bias_init, [output_dim], name="bias") self.act_func = self._init_activation(act_str) self.matmul = ops.MatMul(transpose_b=False) self.bias_add = ops.BiasAdd() self.cast = ops.Cast() self.dropout = Dropout(keep_prob=(1 - keep_prob)) self.use_activation = use_activation self.convert_dtype = convert_dtype self.drop_out = drop_out def _init_activation(self, act_str): act_str = act_str.lower() if act_str == "relu": act_func = ops.ReLU() elif act_str == "sigmoid": act_func = ops.Sigmoid() elif act_str == "tanh": act_func = ops.Tanh() return act_func def construct(self, x): ''' Construct Dense layer ''' if self.training and self.drop_out: x = self.dropout(x) if self.convert_dtype: x = self.cast(x, mstype.float16) weight = self.cast(self.weight, mstype.float16) bias = self.cast(self.bias, mstype.float16) wx = self.matmul(x, weight) wx = self.bias_add(wx, bias) if self.use_activation: wx = self.act_func(wx) wx = self.cast(wx, mstype.float32) else: wx = self.matmul(x, self.weight) wx = self.bias_add(wx, self.bias) if self.use_activation: wx = self.act_func(wx) return wx class WideDeepModel(nn.Cell): """ From paper: " Wide & Deep Learning for Recommender Systems" Args: config (Class): The default config of Wide&Deep """ def __init__(self, config): super(WideDeepModel, self).__init__() self.batch_size = config.batch_size host_device_mix = bool(config.host_device_mix) parameter_server = bool(config.parameter_server) parallel_mode = context.get_auto_parallel_context("parallel_mode") is_auto_parallel = parallel_mode in (ParallelMode.SEMI_AUTO_PARALLEL, ParallelMode.AUTO_PARALLEL) if is_auto_parallel: self.batch_size = self.batch_size * get_group_size() sparse = config.sparse self.field_size = config.field_size self.emb_dim = config.emb_dim self.weight_init, self.bias_init = config.weight_bias_init self.deep_input_dims = self.field_size * self.emb_dim self.all_dim_list = [self.deep_input_dims] + config.deep_layer_dim + [1] init_acts = [('Wide_b', [1], config.emb_init)] var_map = init_var_dict(config.init_args, init_acts) self.wide_b = var_map["Wide_b"] self.dense_layer_1 = DenseLayer(self.all_dim_list[0], self.all_dim_list[1], config.weight_bias_init, config.deep_layer_act, convert_dtype=True, drop_out=config.dropout_flag) self.dense_layer_2 = DenseLayer(self.all_dim_list[1], self.all_dim_list[2], config.weight_bias_init, config.deep_layer_act, convert_dtype=True, drop_out=config.dropout_flag) self.dense_layer_3 = DenseLayer(self.all_dim_list[2], self.all_dim_list[3], config.weight_bias_init, config.deep_layer_act, convert_dtype=True, drop_out=config.dropout_flag) self.dense_layer_4 = DenseLayer(self.all_dim_list[3], self.all_dim_list[4], config.weight_bias_init, config.deep_layer_act, convert_dtype=True, drop_out=config.dropout_flag) self.dense_layer_5 = DenseLayer(self.all_dim_list[4], self.all_dim_list[5], config.weight_bias_init, config.deep_layer_act, use_activation=False, convert_dtype=True, drop_out=config.dropout_flag) self.wide_mul = ops.Mul() self.deep_mul = ops.Mul() self.reduce_sum = ops.ReduceSum(keep_dims=False) self.reshape = ops.Reshape() self.deep_reshape = ops.Reshape() self.square = ops.Square() self.concat = ops.Concat(axis=1) self.unique = ops.Unique().shard(((1,),)) self.wide_gatherv2 = ops.Gather() self.deep_gatherv2 = ops.Gather() if is_auto_parallel and sparse and not config.field_slice and not parameter_server: target = 'CPU' if host_device_mix else 'DEVICE' self.wide_embeddinglookup = nn.EmbeddingLookup(config.vocab_size, 1, target=target, slice_mode=nn.EmbeddingLookup.TABLE_ROW_SLICE) if config.deep_table_slice_mode == "column_slice": self.deep_embeddinglookup = nn.EmbeddingLookup(config.vocab_size, self.emb_dim, target=target, slice_mode=nn.EmbeddingLookup.TABLE_COLUMN_SLICE) if config.use_sp: self.dense_layer_1.matmul.shard(((1, get_group_size()), (get_group_size(), 1))) self.dense_layer_1.bias_add.shard(((get_group_size(), 1), (1,))) self.deep_mul.shard(((1, 1, get_group_size()), (1, 1, 1))) else: self.dense_layer_1.dropout.dropout.shard(((1, get_group_size()),)) self.dense_layer_1.matmul.shard(((1, get_group_size()), (get_group_size(), 1))) self.deep_mul.shard(((1, 1, get_group_size()), (1, 1, 1))) self.dense_layer_1.matmul.add_prim_attr("field_size", self.field_size) self.deep_reshape.add_prim_attr("skip_redistribution", True) else: self.deep_embeddinglookup = nn.EmbeddingLookup(config.vocab_size, self.emb_dim, target=target, slice_mode=nn.EmbeddingLookup.TABLE_ROW_SLICE) self.reduce_sum.add_prim_attr("cross_batch", True) self.embedding_table = self.deep_embeddinglookup.embedding_table elif is_auto_parallel and host_device_mix and config.field_slice and config.full_batch and config.manual_shape: manual_shapes = tuple((s[0] for s in config.manual_shape)) self.deep_embeddinglookup = nn.EmbeddingLookup(config.vocab_size, self.emb_dim, slice_mode=nn.EmbeddingLookup.FIELD_SLICE, manual_shapes=manual_shapes) self.wide_embeddinglookup = nn.EmbeddingLookup(config.vocab_size, 1, slice_mode=nn.EmbeddingLookup.FIELD_SLICE, manual_shapes=manual_shapes) self.deep_mul.shard(((1, get_group_size(), 1), (1, get_group_size(), 1))) self.wide_mul.shard(((1, get_group_size(), 1), (1, get_group_size(), 1))) self.reduce_sum.shard(((1, get_group_size(), 1),)) self.dense_layer_1.dropout.dropout.shard(((1, get_group_size()),)) self.dense_layer_1.matmul.shard(((1, get_group_size()), (get_group_size(), 1))) self.embedding_table = self.deep_embeddinglookup.embedding_table elif parameter_server: cache_enable = config.vocab_cache_size > 0 target = 'DEVICE' if cache_enable else 'CPU' if not cache_enable: sparse = True if is_auto_parallel and config.full_batch and cache_enable: self.deep_embeddinglookup = nn.EmbeddingLookup(config.vocab_size, self.emb_dim, target=target, slice_mode=nn.EmbeddingLookup.TABLE_ROW_SLICE, sparse=sparse, vocab_cache_size=config.vocab_cache_size) self.wide_embeddinglookup = nn.EmbeddingLookup(config.vocab_size, 1, target=target, slice_mode=nn.EmbeddingLookup.TABLE_ROW_SLICE, sparse=sparse, vocab_cache_size=config.vocab_cache_size) else: self.deep_embeddinglookup = nn.EmbeddingLookup(config.vocab_size, self.emb_dim, target=target, sparse=sparse, vocab_cache_size=config.vocab_cache_size) self.wide_embeddinglookup = nn.EmbeddingLookup(config.vocab_size, 1, target=target, sparse=sparse, vocab_cache_size=config.vocab_cache_size) self.embedding_table = self.deep_embeddinglookup.embedding_table self.deep_embeddinglookup.embedding_table.set_param_ps() self.wide_embeddinglookup.embedding_table.set_param_ps() else: self.deep_embeddinglookup = nn.EmbeddingLookup(config.vocab_size, self.emb_dim, target='DEVICE', sparse=sparse, vocab_cache_size=config.vocab_cache_size) self.wide_embeddinglookup = nn.EmbeddingLookup(config.vocab_size, 1, target='DEVICE', sparse=sparse, vocab_cache_size=config.vocab_cache_size) self.embedding_table = self.deep_embeddinglookup.embedding_table def construct(self, id_hldr, wt_hldr): """ Args: id_hldr: batch ids; wt_hldr: batch weights; """ # Wide layer wide_id_weight = self.wide_embeddinglookup(id_hldr) # Deep layer deep_id_embs = self.deep_embeddinglookup(id_hldr) mask = self.reshape(wt_hldr, (self.batch_size, self.field_size, 1)) # Wide layer wx = self.wide_mul(wide_id_weight, mask) wide_out = self.reshape(self.reduce_sum(wx, 1) + self.wide_b, (-1, 1)) # Deep layer vx = self.deep_mul(deep_id_embs, mask) deep_in = self.deep_reshape(vx, (-1, self.field_size * self.emb_dim)) deep_in = self.dense_layer_1(deep_in) deep_in = self.dense_layer_2(deep_in) deep_in = self.dense_layer_3(deep_in) deep_in = self.dense_layer_4(deep_in) deep_out = self.dense_layer_5(deep_in) out = wide_out + deep_out return out, self.embedding_table class NetWithLossClass(nn.Cell): """" Provide WideDeep training loss through network. Args: network (Cell): The training network config (Class): WideDeep config """ def __init__(self, network, config): super(NetWithLossClass, self).__init__(auto_prefix=False) host_device_mix = bool(config.host_device_mix) parameter_server = bool(config.parameter_server) sparse = config.sparse parallel_mode = context.get_auto_parallel_context("parallel_mode") is_auto_parallel = parallel_mode in (ParallelMode.SEMI_AUTO_PARALLEL, ParallelMode.AUTO_PARALLEL) self.no_l2loss = (is_auto_parallel if (host_device_mix or config.field_slice) else parameter_server) if sparse: self.no_l2loss = True self.network = network self.l2_coef = config.l2_coef self.loss = ops.SigmoidCrossEntropyWithLogits() self.square = ops.Square() self.reduceMean_false = ops.ReduceMean(keep_dims=False) if is_auto_parallel: self.reduceMean_false.add_prim_attr("cross_batch", True) self.reduceSum_false = ops.ReduceSum(keep_dims=False) def construct(self, batch_ids, batch_wts, label): ''' Construct NetWithLossClass ''' predict, embedding_table = self.network(batch_ids, batch_wts) log_loss = self.loss(predict, label) wide_loss = self.reduceMean_false(log_loss) if self.no_l2loss: deep_loss = wide_loss else: l2_loss_v = self.reduceSum_false(self.square(embedding_table)) / 2 deep_loss = self.reduceMean_false(log_loss) + self.l2_coef * l2_loss_v return wide_loss, deep_loss class IthOutputCell(nn.Cell): def __init__(self, network, output_index): super(IthOutputCell, self).__init__() self.network = network self.output_index = output_index def construct(self, x1, x2, x3): predict = self.network(x1, x2, x3)[self.output_index] return predict class TrainStepWrap(nn.Cell): """ Encapsulation class of WideDeep network training. Append Adam and FTRL optimizers to the training network after that construct function can be called to create the backward graph. Args: network (Cell): The training network. Note that loss function should have been added. sens (Number): The adjust parameter. Default: 1024.0 host_device_mix (Bool): Whether run in host and device mix mode. Default: False parameter_server (Bool): Whether run in parameter server mode. Default: False """ def __init__(self, network, sens=1024.0, host_device_mix=False, parameter_server=False, sparse=False, cache_enable=False): super(TrainStepWrap, self).__init__() parallel_mode = context.get_auto_parallel_context("parallel_mode") is_auto_parallel = parallel_mode in (ParallelMode.SEMI_AUTO_PARALLEL, ParallelMode.AUTO_PARALLEL) self.network = network self.network.set_train() self.trainable_params = network.trainable_params() weights_w = [] weights_d = [] for params in self.trainable_params: if 'wide' in params.name: weights_w.append(params) else: weights_d.append(params) self.weights_w = ParameterTuple(weights_w) self.weights_d = ParameterTuple(weights_d) if (sparse and is_auto_parallel) or (sparse and parameter_server): self.optimizer_d = Adam( self.weights_d, learning_rate=5e-4, eps=1e-8, loss_scale=sens, use_lazy=True) self.optimizer_w = FTRL(learning_rate=1e-3, params=self.weights_w, l1=1e-8, l2=1e-8, initial_accum=1.0, loss_scale=sens) if host_device_mix or (parameter_server and not cache_enable): self.optimizer_w.target = "CPU" self.optimizer_d.target = "CPU" else: self.optimizer_d = Adam( self.weights_d, learning_rate=5e-4, eps=1e-8, loss_scale=sens) self.optimizer_w = FTRL(learning_rate=1e-3, params=self.weights_w, l1=1e-8, l2=1e-8, initial_accum=1.0, loss_scale=sens) self.hyper_map = ops.HyperMap() self.grad_w = ops.GradOperation(get_by_list=True, sens_param=True) self.grad_d = ops.GradOperation(get_by_list=True, sens_param=True) self.sens = sens self.loss_net_w = IthOutputCell(network, output_index=0) self.loss_net_d = IthOutputCell(network, output_index=1) self.loss_net_w.set_grad() self.loss_net_d.set_grad() self.reducer_flag = False self.grad_reducer_w = None self.grad_reducer_d = None self.reducer_flag = parallel_mode in (ParallelMode.DATA_PARALLEL, ParallelMode.HYBRID_PARALLEL) if self.reducer_flag: mean = context.get_auto_parallel_context("gradients_mean") degree = context.get_auto_parallel_context("device_num") self.grad_reducer_w = DistributedGradReducer(self.optimizer_w.parameters, mean, degree) self.grad_reducer_d = DistributedGradReducer(self.optimizer_d.parameters, mean, degree) def construct(self, batch_ids, batch_wts, label): ''' Construct wide and deep model ''' weights_w = self.weights_w weights_d = self.weights_d loss_w, loss_d = self.network(batch_ids, batch_wts, label) sens_w = ops.Fill()(ops.DType()(loss_w), ops.Shape()(loss_w), self.sens) sens_d = ops.Fill()(ops.DType()(loss_d), ops.Shape()(loss_d), self.sens) grads_w = self.grad_w(self.loss_net_w, weights_w)(batch_ids, batch_wts, label, sens_w) grads_d = self.grad_d(self.loss_net_d, weights_d)(batch_ids, batch_wts, label, sens_d) if self.reducer_flag: grads_w = self.grad_reducer_w(grads_w) grads_d = self.grad_reducer_d(grads_d) return ops.depend(loss_w, self.optimizer_w(grads_w)), ops.depend(loss_d, self.optimizer_d(grads_d)) class PredictWithSigmoid(nn.Cell): """ Predict definition """ def __init__(self, network): super(PredictWithSigmoid, self).__init__() self.network = network self.sigmoid = ops.Sigmoid() parallel_mode = context.get_auto_parallel_context("parallel_mode") full_batch = context.get_auto_parallel_context("full_batch") is_auto_parallel = parallel_mode in (ParallelMode.SEMI_AUTO_PARALLEL, ParallelMode.AUTO_PARALLEL) if is_auto_parallel and full_batch: self.sigmoid.shard(((1, 1),)) def construct(self, batch_ids, batch_wts, labels): logits, _, = self.network(batch_ids, batch_wts) pred_probs = self.sigmoid(logits) return logits, pred_probs, labels # Pre processing def pre_process_criteo_wide_deep(x): return x class WideDeepPostProcess: def __init__(self): self.good = 0 self.total = 0 self.roc_auc = 0 self.results = [] self.labels = [] def __call__(self, results, expected=None, result_dict=None): processed_results = [] n = len(results) for idx in range(0, n): result = results['auc'] processed_results.append(result) self.good += 1 self.total += 1 return processed_results def add_results(self, labels, results): self.results.append(results) self.labels.append(labels) def start(self): self.good = 0 self.total = 0 self.roc_auc = 0 self.results = [] def finalize(self, result_dict, ds=False, output_dir=None): result_dict["good"] = self.good result_dict["total"] = self.total 这段代码的作用是什么?

/********************************************************************* * * Copyright (C), 2010-2022 Oplus. All rights reserved.. * * VENDOR_EDIT * * File : AddFilePanelFragment.kt * * Description : AddFilePanelFragment * * Version : 1.0 * * Date : 2025/02/08 * * Author : W9085798 * * * * ---------------------Revision History: ---------------------------- * * <author> <data> <version> <desc> ***********************************************************************/ package com.oplus.filemanager.addfilepanel.ui import android.annotation.SuppressLint import android.app.Activity import android.os.SystemClock import android.view.LayoutInflater import android.view.View import android.view.ViewGroup import android.widget.ImageView import android.widget.RelativeLayout import android.widget.TextView import androidx.annotation.VisibleForTesting import androidx.fragment.app.Fragment import androidx.fragment.app.FragmentActivity import androidx.lifecycle.Lifecycle import androidx.lifecycle.ViewModelProvider import com.coui.appcompat.button.COUIButton import com.coui.appcompat.button.SingleButtonWrap import com.coui.appcompat.contextutil.COUIContextUtil import com.coui.appcompat.panel.COUIBottomSheetDialog import com.coui.appcompat.panel.COUIBottomSheetDialogFragment import com.coui.appcompat.panel.COUIPanelFragment import com.coui.appcompat.panel.COUIPanelMultiWindowUtils import com.coui.appcompat.tablayout.COUITab import com.coui.appcompat.tablayout.COUITabLayout import com.coui.appcompat.toolbar.COUIToolbar import com.filemanager.common.MyApplication.appContext import com.filemanager.common.base.BaseFileBean import com.filemanager.common.base.BaseFragmentAdapter import com.filemanager.common.constants.Constants import com.filemanager.common.controller.LoadingController import com.filemanager.common.controller.PrivacyPolicyController import com.filemanager.common.helper.MimeTypeHelper import com.filemanager.common.helper.uiconfig.UIConfigMonitor import com.filemanager.common.utils.Injector import com.filemanager.common.utils.KtViewUtils import com.filemanager.common.utils.Log import com.filemanager.common.utils.ModelUtils import com.filemanager.common.utils.SdkUtils import com.filemanager.common.utils.StatusBarUtil import com.filemanager.common.utils.Utils import com.filemanager.common.utils.dp2px import com.filemanager.common.view.ViewPagerWrapperForPC import com.filemanager.common.view.viewpager.RTLViewPager import com.oplus.filemanager.addfilepanel.R import com.oplus.filemanager.addfilepanel.bean.AddFileBean import com.oplus.filemanager.addfilepanel.viewmodel.AddFilePanelViewModel import com.oplus.filemanager.interfaze.categorydoc.IDocumentExtensionType import com.oplus.filemanager.interfaze.main.IMain import kotlin.properties.Delegates class AddFilePanelFragment : COUIPanelFragment(), COUITabLayout.OnTabSelectedListener, PrivacyPolicyController.OnPrivacyPolicyListener { companion object { private const val TAG = "ShortcutFolderAddFilePanelFragment" private const val NUM = 1024 private const val DECIMAL_NUM = 1024f private const val DISMISS_TIME_GAP = 1000L private const val DELAY_TIME = 50L private const val PANEL_MAX_HEIGHT = "panel_max_height" } var lifeCycle: Lifecycle by Delegates.notNull() var title: String? = null var mCurrentPath: String = "" private var disMissTime: Long = 0L private var toolBar: COUIToolbar? = null private var mTabTitle = arrayListOf( appContext.resources.getString(com.filemanager.common.R.string.total), appContext.resources.getString(com.filemanager.common.R.string.string_photos), appContext.resources.getString(com.filemanager.common.R.string.string_videos), appContext.resources.getString(com.filemanager.common.R.string.string_audio), appContext.resources.getString(com.filemanager.common.R.string.string_documents), appContext.resources.getString(com.filemanager.common.R.string.string_other) ) private val mTabTitles = intArrayOf( Constants.TAB_ALL, Constants.TAB_IMAGE, Constants.TAB_VIDEO, Constants.TAB_AUDIO, Constants.TAB_DOCUMENT, Constants.TAB_OTHER ) private var mAddFileDialogInterface: AddFileDialogInterface? = null private var mPages: ArrayList<AddFileListFragment> = ArrayList() private var mViewPager: RTLViewPager? = null private var mTabView: COUITabLayout? = null private var mViewPagerWrapper: ViewPagerWrapperForPC? = null private var addFileBtn: COUIButton? = null private var verticalButtonWrap: SingleButtonWrap? = null private var rootView: ViewGroup? = null private var addFilePanelViewModel: AddFilePanelViewModel? = null private var statisticsSelectTitleView: TextView? = null private var statisticsSelectBodyView: TextView? = null private var contentLayout: RelativeLayout? = null private var buttonDivider: View? = null private var statisticsArrowUpView: ImageView? = null private var statisticsRootView: ViewGroup? = null private var loadingController: LoadingController? = null private var mPosition: Int = 0 var limitCount: Int = 0 private val allFiles: MutableList<BaseFileBean> = mutableListOf() private val imageFiles: MutableList<BaseFileBean> = mutableListOf() private val videoFiles: MutableList<BaseFileBean> = mutableListOf() private val audioFiles: MutableList<BaseFileBean> = mutableListOf() private val docFiles: MutableList<BaseFileBean> = mutableListOf() private val otherFiles: MutableList<BaseFileBean> = mutableListOf() private val mSelectArrayByID: MutableSet<Long> = mutableSetOf() var addFileClickCallback: (selectData: List<AddFileBean>) -> Unit = {} var isExternalReference: Boolean = false private fun setPanelMaxHeight() { if (activity == null) return rootView?.layoutParams?.apply { if (UIConfigMonitor.isCurrentZoomWindowShow() && SdkUtils.isAtLeastR() && !isExternalReference) { val savedHeight: Int = arguments?.getInt(PANEL_MAX_HEIGHT) ?: 0 Log.d(TAG, "setPanelMaxHeight savedHeight = $savedHeight") if (savedHeight == 0) { // 处理浮窗 height = handleZoomWindowHeight(requireActivity(), height) arguments?.putInt(PANEL_MAX_HEIGHT, height) } else { height = savedHeight } Log.d(TAG, "setPanelMaxHeight height = $height") } else { val screenHeight = dp2px(requireActivity(), requireActivity().resources.configuration.screenHeightDp) val panelMaxHeight = COUIPanelMultiWindowUtils.getPanelMaxHeight(requireActivity(), requireActivity().resources.configuration) Log.d(TAG, "setPanelMaxHeight screenHeight:$screenHeight panelH:$panelMaxHeight height:$height") //当是分屏 if (requireActivity().isInMultiWindowMode) { if (StatusBarUtil.checkIsGestureNavMode(requireActivity())) { //全面屏手势导航 height = handleGestureNavModeHeight(requireActivity(), height) } else { // 底部导航栏 height = Math.min(screenHeight, panelMaxHeight) - requireActivity().resources.getDimensionPixelOffset( com.filemanager.common.R.dimen.dimen_40dp ) height = handleNavModeHeight(requireActivity(), height) } } rootView?.layoutParams = this } } } /** * 处理浮窗模式下的高度 */ private fun handleZoomWindowHeight(activity: Activity, height: Int): Int { var result = height val statusBarH = COUIPanelMultiWindowUtils.getStatusBarHeight(activity) val configuration = activity.resources.configuration val paddingBottom = COUIPanelMultiWindowUtils.getPanelMarginBottom(context, configuration) val rect = COUIPanelMultiWindowUtils.getCurrentWindowVisibleRect(activity) if (rect != null) { if (rect.top == 0) { Log.d(TAG, "handleZoomWindowHeight: rect $rect top is invalid, reset") rect.top = activity.resources.getDimensionPixelOffset(com.filemanager.common.R.dimen.panel_min_top_margin) } if (rect.bottom == 0) { val windowH = KtViewUtils.getWindowSize(activity).y rect.bottom = windowH Log.d(TAG, "handleZoomWindowHeight: rect $rect bottom is invalid, reset $windowH") } result = (rect.bottom - rect.top) - paddingBottom - statusBarH Log.d(TAG, "handleZoomWindowHeight: height $result = ${rect.bottom}-${rect.top}-$paddingBottom-$statusBarH") } return result } /** * 处理底部导航栏导航时的高度 */ private fun handleNavModeHeight(it: Activity, height: Int): Int { val navBarHeight = KtViewUtils.getSoftNavigationBarSize(it) var result = height when { ModelUtils.isTablet() -> { //平板竖屏分屏时 if (UIConfigMonitor.instance.isDevicePortrait(it)) { result = height - navBarHeight Log.d(TAG, "handleNavMode pad portrait height:$result") } } UIConfigMonitor.instance.isFoldable(it) -> { // 折叠屏 result = height - navBarHeight Log.d(TAG, "handleNavMode fold height:$result") } else -> { //手机且分屏处于下屏 if (!COUIPanelMultiWindowUtils.isDisplayInUpperWindow(it)) { result = height - navBarHeight Log.d(TAG, "handleNavMode phone lower split screen height:$result") } } } return result } /** * 处理全面屏手势导航时的高度 */ @VisibleForTesting fun handleGestureNavModeHeight(activity: Activity, height: Int): Int { if (COUIPanelMultiWindowUtils.isDisplayInUpperWindow(activity)) { // 上分屏,不用处理 return height } // 处于下分屏,并且显示了手势指示条 if (StatusBarUtil.checkShowGestureNavBar(activity)) { val result = height - activity.resources.getDimensionPixelOffset(com.filemanager.common.R.dimen.dimen_16dp) Log.d(TAG, "handleGestureNavMode lower split screen with show gesture nvaBar height:$result") return result } return height } override fun onDestroy() { Log.d(TAG, "onDestroy") super.onDestroy() verticalButtonWrap?.release() clearDataList() addFilePanelViewModel?.onDestroy() } override fun initView(panelView: View?) { activity?.let { activity -> LayoutInflater.from(activity).inflate(getLayoutResId(), null, false)?.let { (contentView as? ViewGroup)?.addView(it) initContentView(it) initViewModel(activity) startObserver() initData() } } } private fun getLayoutResId(): Int { return R.layout.add_file_panel_dialog } private fun initViewModel(act: FragmentActivity) { addFilePanelViewModel = ViewModelProvider(act)[AddFilePanelViewModel::class.java] } private fun startObserver() { addFilePanelViewModel?.mAllFiles?.observe(this) { it ?: return@observe showLoadingView(false) setViewPagerData(it) } addFilePanelViewModel?.mAllSelectFiles?.observe(this) { it ?: return@observe updateStatisticsSelectView(it) } } fun notifyUpdate() { mSelectArrayByID.clear() addFilePanelViewModel?.mAllSelectFiles?.value?.forEach { bean -> if (bean.isSelected) { bean.mFileID?.let { mSelectArrayByID.add(it) } } } getCurrentFragment()?.apply { setSelectedData(mSelectArrayByID) notifyUpdate() } } private fun initContentView(view: View) { Log.d(TAG, "initContentView") hideDragView() rootView = view.findViewById(R.id.dialog_layout) toolBar = view.findViewById(R.id.toolbar) mTabView = view.findViewById<COUITabLayout?>(R.id.tab_layout) mViewPager = view.findViewById(R.id.viewPager) mViewPagerWrapper = view.findViewById<ViewPagerWrapperForPC?>(R.id.view_pager_wrapper).apply { notifyMainViewPager = object : ((Boolean) -> Unit) { override fun invoke(enable: Boolean) { activity?.let { val mainAction = Injector.injectFactory<IMain>() mainAction?.setViewPagerScrollEnabled(it, enable) } } } } addFileBtn = view.findViewById<COUIButton?>(R.id.btn_add_file).apply { verticalButtonWrap = SingleButtonWrap(this, SingleButtonWrap.Type.Small) setOnClickListener { Log.d(TAG, "addFileBtn click. select:${addFilePanelViewModel?.mAllSelectFiles?.value?.size}") val selectList = addFilePanelViewModel?.mAllSelectFiles?.value selectList?.let { addFileClickCallback.invoke(it) } } } contentLayout = view.findViewById(R.id.content_layout) buttonDivider = view.findViewById(R.id.button_divider) statisticsSelectTitleView = view.findViewById<TextView?>(R.id.select_title_content) statisticsSelectBodyView = view.findViewById<TextView?>(R.id.select_body_content) statisticsArrowUpView = view.findViewById(R.id.select_arraw_up) statisticsRootView = view.findViewById<ViewGroup?>(R.id.select_root_view).apply { setOnClickListener { if (Utils.isQuickClick()) { Log.w(TAG, "click too fast, try later") return@setOnClickListener } addFilePanelViewModel?.mAllSelectFiles?.value?.let { mAddFileDialogInterface?.replaceSelectedPanelFragment(it) } } } setPanelMaxHeight() } private fun initData() { initToolBar() resetStatisticsView() val fragments = childFragmentManager.fragments for (i in mTabTitles.indices) { initFragment(i, fragments) } initViewPager() if (PrivacyPolicyController.hasAgreePrivacy()) { loadData() } else { PrivacyPolicyController.bindPrivacyPolicyListener(this) } } private fun loadData() { Log.d(TAG, "loadData") showLoadingView(true) // 加载数据 activity?.let { addFilePanelViewModel?.loadAllFiles(it) } } private fun resetStatisticsView() { addFileBtn?.isEnabled = false statisticsSelectTitleView?.apply { visibility = View.VISIBLE text = appContext.resources.getString(com.filemanager.common.R.string.not_selected_file) } statisticsSelectBodyView?.apply { visibility = View.GONE text = "" } statisticsArrowUpView?.apply { visibility = View.GONE } statisticsRootView?.apply { isClickable = false } } private fun initToolBar() { toolBar?.apply { visibility = View.VISIBLE title = appContext.resources.getString(com.filemanager.common.R.string.encryption_file_select_title) isTitleCenterStyle = true inflateMenu(R.menu.add_file_panel_dialog_menu_bar) menu.findItem(R.id.cancel).setOnMenuItemClickListener { mAddFileDialogInterface?.dismissAddFileDialog() true } } if ((mTabView != null) && (mViewPager != null)) { mTabView?.let { it.setupWithViewPager(mViewPager, false) it.addOnTabSelectedListener(this) it.isUpdateindicatorposition = true } } } fun setAddFileDialogInterface(addFileDialog: AddFileDialogInterface) { mAddFileDialogInterface = addFileDialog } private fun initViewPager() { mViewPager?.let { it.offscreenPageLimit = mTabTitles.size it.adapter = ViewPagerFragmentStateAdapter(this) it.currentItem = 0 it.overScrollMode = View.OVER_SCROLL_NEVER } } private fun initFragment(position: Int, fragments: List<Fragment>): Fragment { Log.d(TAG, "initFragment") var fragment = if (fragments.isEmpty()) null else fragments[position] if (fragment == null) { fragment = AddFileListFragment() fragment.isExternalReference = isExternalReference fragment.lifeCycle = lifeCycle fragment.mOnItemSelectCallback = { _, fileID -> mSelectArrayByID.add(fileID) addFilePanelViewModel?.updateAllSelectFiles(mSelectArrayByID) } fragment.mOnItemUnSelectCallback = { _, fileID -> mSelectArrayByID.remove(fileID) addFilePanelViewModel?.updateAllSelectFiles(mSelectArrayByID) } fragment.mFileEmptyCallback = { isShowEmpty -> handleFileEmptyView(isShowEmpty) } fragment.mCurrentPath = mCurrentPath fragment.limitCount = limitCount } if (fragment is AddFileListFragment) { mPages.add(fragment) } return fragment } private fun handleFileEmptyView(isShowEmptyView: Boolean) { if (isShowEmptyView) { showLoadingView(false) buttonDivider?.alpha = 0F } else { buttonDivider?.alpha = 1F } } private fun showLoadingView(isShow: Boolean) { if (isShow) { Log.d(TAG, "showLoadingView disMissTime $disMissTime") activity?.let { loadingController = LoadingController(it, this) //如果异常loadingView刚刚调用,就不调用显示loadingView,防止loadingView一直显示 Log.d(TAG, "showLoadingView getLoadingShowStartTime ${loadingController?.getLoadingShowStartTime()}") if (SystemClock.elapsedRealtime() - disMissTime >= DISMISS_TIME_GAP || loadingController?.getLoadingShowStartTime() == 0L) { loadingController?.showLoading(contentLayout) } } } else { Log.d(TAG, "showLoadingView dismissLoading") loadingController?.dismissLoading() disMissTime = SystemClock.elapsedRealtime() } } override fun onShow(isShowOnFirstPanel: Boolean?) { super.onShow(isShowOnFirstPanel) ((parentFragment as? COUIBottomSheetDialogFragment)?.dialog as? COUIBottomSheetDialog) ?.setPanelBackgroundTintColor(COUIContextUtil.getAttrColor(context, com.support.appcompat.R.attr.couiColorBackgroundElevatedWithCard)) } private fun clearDataList() { allFiles.clear() imageFiles.clear() videoFiles.clear() audioFiles.clear() docFiles.clear() otherFiles.clear() mSelectArrayByID.clear() } private fun setViewPagerData(data: MutableList<AddFileBean>) { Log.d(TAG, "setViewPagerData") clearDataList() allFiles.addAll(data) val documentExtensionType = Injector.injectFactory<IDocumentExtensionType>() data.forEach { file -> val localType = file.mLocalType if (MimeTypeHelper.isImageType(localType)) { imageFiles.add(file) } else if (MimeTypeHelper.isVideoType(localType)) { videoFiles.add(file) } else if (MimeTypeHelper.isAudioType(localType)) { audioFiles.add(file) } else if ((MimeTypeHelper.isDocType(localType) || MimeTypeHelper.isOtherDocType(localType)) && documentExtensionType?.isCategoryDocSupportType(file) == true ) { //isCategoryDocSupportType确保面板上文档分类显示的内容和文管文档分类一致(部分未知格式后缀会识解析TXT类型) docFiles.add(file) } else { otherFiles.add(file) } } updateData(mTabTitles[mPosition]) } private fun updateStatisticsSelectView(data: MutableList<AddFileBean>) { if (data.size <= 0) { Log.d(TAG, "unselect file") resetStatisticsView() return } addFileBtn?.isEnabled = true statisticsSelectTitleView?.apply { visibility = View.VISIBLE text = appContext.resources.getQuantityString( com.filemanager.common.R.plurals.mark_selected_items_new, data.size, data.size ) } statisticsSelectBodyView?.apply { visibility = View.VISIBLE val size: Long = data.sumOf { it.mSize } val totalSize: String = updateSizeFormat(size) text = appContext.resources.getString(com.filemanager.common.R.string.selected_size, totalSize) } statisticsArrowUpView?.apply { visibility = View.VISIBLE } statisticsRootView?.apply { isClickable = true } } @SuppressLint("DefaultLocale") fun updateSizeFormat(size: Long): String { return if (size < NUM) { size.toString() + "B" } else if (size < NUM * NUM) { String.format("%.1f", (size / DECIMAL_NUM)) + " KB" } else if (size < NUM * NUM * NUM) { String.format("%.1f", (size / NUM / DECIMAL_NUM)) + " MB" } else { String.format("%.1f", (size / NUM / NUM / DECIMAL_NUM)) + " GB" } } private fun updateData(fragmentType: Int) { val fragment = getCurrentFragment() fragment?.mFragmentType = fragmentType when (fragmentType) { Constants.TAB_ALL -> fragment?.setData(allFiles) Constants.TAB_IMAGE -> fragment?.setData(imageFiles) Constants.TAB_VIDEO -> fragment?.setData(videoFiles) Constants.TAB_AUDIO -> fragment?.setData(audioFiles) Constants.TAB_DOCUMENT -> fragment?.setData(docFiles) Constants.TAB_OTHER -> fragment?.setData(otherFiles) } fragment?.setSelectedData(mSelectArrayByID) Log.d( TAG, "allFiles=${allFiles.size}, imageFiles=${imageFiles.size}, videoFiles=${videoFiles.size}, " + "audioFiles=${audioFiles.size}, docFiles=${docFiles.size}, otherFiles=${otherFiles.size}" ) } override fun onTabSelected(tab: COUITab?) { Log.d(TAG, "onTabSelected tab.postion=${tab?.position}") tab?.let { mPosition = it.position if (loadingController?.isShowing() != true) { updateData(mTabTitles[mPosition]) } } } override fun onTabUnselected(p0: COUITab?) {} override fun onTabReselected(p0: COUITab?) {} private fun getCurrentFragment(): AddFileListFragment? { return if (mPosition < mPages.size) { mPages[mPosition] } else { null } } inner class ViewPagerFragmentStateAdapter(fragment: Fragment) : BaseFragmentAdapter(fragment) { override fun getItemCount(): Int { return mTabTitle.size } override fun createFragment(position: Int): Fragment { return mPages[position] } override fun getPageTitle(position: Int): CharSequence { return mTabTitle[position] } } override fun onAgreeResult(agree: Boolean, noLongerRemind: Boolean) { loadData() } } interface AddFileDialogInterface { fun dismissAddFileDialog() fun replaceSelectedPanelFragment(selectData: MutableList<AddFileBean>) }这是对应的kotlin代码,怎么处理

#pragma once #include <QMainWindow> #include <QMap> #include <QVector> #include "qcustomplot.h" // 数据结构定义 struct MatchResult { double time; double alerttime; double das_azimuth; double das_ele; double act_azi; double act_ele; double distance; double azidiffs; double elediffs; }; struct DasMatchResult { QString target_type; int num; double end_time; double start_time; QVector<MatchResult> data; }; namespace Ui { class Plotter; } class Plotter : public QMainWindow { Q_OBJECT public: explicit Plotter(QWidget *parent = nullptr); ~Plotter(); void processData(const QMap<QString, DasMatchResult> &results); private slots: void onItemDoubleClicked(QListWidgetItem *item); void updatePlots(); void onCustomPlotClicked(QMouseEvent *event); void onNextPoint(); void onPrevPoint(); private: Ui::Plotter *ui; QMap<QString, DasMatchResult> m_result; QString currentKey; QCustomPlot* m_currentCustomPlot = nullptr; // 修复未定义问题[5](@ref) // 点选标记组件 QCPItemStraightLine *m_selectionLine = nullptr; QCPItemText *m_selectionLabel = nullptr; QVector<QCPItemEllipse*> m_selectionDots; // 当前选中数据 int m_currentIndex = -1; QVector<double> m_currentXData; QVector<QVector<double>> m_currentYData; // 辅助函数 void initComboBoxes(); void plotData(QCustomPlot *plot, const QVector<MatchResult> &data, const QString &field1, const QString &field2); double getValueByField(const MatchResult &mr, const QString &field); void findNearestPoint(QCustomPlot* plot, double clickX, double clickY); void updateSelectionDisplay(QCustomPlot* plot); void clearSelection(QCustomPlot* plot); }; #include "plotter.h" #include "ui_plotter.h" #include <QMouseEvent> #include <QListWidgetItem> #include <QDebug> #include <float.h> Plotter::Plotter(QWidget *parent) : QMainWindow(parent), ui(new Ui::Plotter) { ui->setupUi(this); // 初始化UI交互 connect(ui->listWidget, &QListWidget::itemDoubleClicked, this, &Plotter::onItemDoubleClicked); initComboBoxes(); // 初始化点选标记组件 m_selectionLine = new QCPItemStraightLine(ui->customPlot1); m_selectionLine->setVisible(false); m_selectionLabel = new QCPItemText(ui->customPlot1); m_selectionLabel->setVisible(false); // 连接鼠标点击信号[5](@ref) connect(ui->customPlot1, &QCustomPlot::mousePress, this, &Plotter::onCustomPlotClicked); connect(ui->customPlot2, &QCustomPlot::mousePress, this, &Plotter::onCustomPlotClicked); // 连接导航按钮 connect(ui->nextButton, &QPushButton::clicked, this, &Plotter::onNextPoint); connect(ui->prevButton, &QPushButton::clicked, this, &Plotter::onPrevPoint); // 初始化QCustomPlot交互功能 ui->customPlot1->setInteractions(QCP::iRangeDrag | QCP::iRangeZoom); ui->customPlot2->setInteractions(QCP::iRangeDrag | QCP::iRangeZoom); } Plotter::~Plotter() { delete ui; } void Plotter::processData(const QMap<QString, DasMatchResult> &results) { m_result = results; ui->listWidget->clear(); // 填充列表控件 for (auto it = m_result.begin(); it != m_result.end(); ++it) { const QString& key = it.key(); const DasMatchResult& data = it.value(); QString itemText = QString("%1--开始时间:%2--结束时间:%3") .arg(key) .arg(data.start_time, 0, 'f', 2) .arg(data.end_time, 0, 'f', 2); QListWidgetItem* item = new QListWidgetItem(itemText, ui->listWidget); item->setData(Qt::UserRole, key); } } void Plotter::initComboBoxes() { QStringList dataFields = { "", // 空选项(不绘制曲线) "alerttime", "das_azimuth", "das_ele", "act_azi", "act_ele", "distance", "azidiffs", "elediffs" }; ui->comboBox1->addItems(dataFields); ui->comboBox2->addItems(dataFields); ui->comboBox3->addItems(dataFields); ui->comboBox4->addItems(dataFields); // 设置默认空选项 ui->comboBox1->setCurrentIndex(0); ui->comboBox2->setCurrentIndex(0); ui->comboBox3->setCurrentIndex(0); ui->comboBox4->setCurrentIndex(0); // 连接combobox变化信号 connect(ui->comboBox1, &QComboBox::currentTextChanged, this, &Plotter::updatePlots); connect(ui->comboBox2, &QComboBox::currentTextChanged, this, &Plotter::updatePlots); connect(ui->comboBox3, &QComboBox::currentTextChanged, this, &Plotter::updatePlots); connect(ui->comboBox4, &QComboBox::currentTextChanged, this, &Plotter::updatePlots); } void Plotter::onItemDoubleClicked(QListWidgetItem *item) { currentKey = item->data(Qt::UserRole).toString(); updatePlots(); clearSelection(ui->customPlot1); // 清除旧选择 clearSelection(ui->customPlot2); } void Plotter::updatePlots() { if (!m_result.contains(currentKey)) return; const DasMatchResult& dasData = m_result[currentKey]; plotData(ui->customPlot1, dasData.data, ui->comboBox1->currentText(), ui->comboBox2->currentText()); plotData(ui->customPlot2, dasData.data, ui->comboBox3->currentText(), ui->comboBox4->currentText()); } void Plotter::plotData(QCustomPlot *plot, const QVector<MatchResult> &data, const QString &field1, const QString &field2) { plot->clearGraphs(); QVector<double> x, y1, y2; // 准备数据 for (const MatchResult &mr : data) { x.append(mr.time); if (!field1.isEmpty()) y1.append(getValueByField(mr, field1)); if (!field2.isEmpty()) y2.append(getValueByField(mr, field2)); } // 绘制第一条曲线 if (!y1.isEmpty()) { plot->addGraph(); plot->graph(0)->setData(x, y1); plot->graph(0)->setName(field1); plot->graph(0)->setPen(QPen(Qt::blue, 2)); } // 绘制第二条曲线 if (!y2.isEmpty()) { int graphIndex = (y1.isEmpty() ? 0 : 1); plot->addGraph(); plot->graph(graphIndex)->setData(x, y2); plot->graph(graphIndex)->setName(field2); plot->graph(graphIndex)->setPen(QPen(Qt::red, 2)); } // 配置坐标轴 plot->xAxis->setLabel("时间(s)"); plot->yAxis->setLabel("数值"); plot->legend->setVisible(true); plot->rescaleAxes(); plot->replot(); } double Plotter::getValueByField(const MatchResult &mr, const QString &field) { if (field == "alerttime") return mr.alerttime; if (field == "das_azimuth") return mr.das_azimuth; if (field == "das_ele") return mr.das_ele; if (field == "act_azi") return mr.act_azi; if (field == "act_ele") return mr.act_ele; if (field == "distance") return mr.distance; if (field == "azidiffs") return mr.azidiffs; if (field == "elediffs") return mr.elediffs; return 0.0; } // 鼠标点击处理函数[6](@ref) void Plotter::onCustomPlotClicked(QMouseEvent *event) { m_currentCustomPlot = qobject_cast<QCustomPlot*>(sender()); // 右键清除选择 if (event->buttons() & Qt::RightButton) { // 修复event->button()问题 clearSelection(m_currentCustomPlot); return; } // 左键处理 if (event->buttons() & Qt::LeftButton) { double clickX = m_currentCustomPlot->xAxis->pixelToCoord(event->pos().x()); findNearestPoint(m_currentCustomPlot, clickX, 0); updateSelectionDisplay(m_currentCustomPlot); } } void Plotter::findNearestPoint(QCustomPlot* plot, double clickX, double) { m_currentIndex = -1; double minDistance = DBL_MAX; m_currentXData.clear(); m_currentYData.clear(); // 收集当前图表数据 for (int i = 0; i < plot->graphCount(); ++i) { QCPGraph* graph = plot->graph(i); auto data = graph->data(); QVector<double> yValues; for (auto it = data->begin(); it != data->end(); ++it) { if (i == 0) m_currentXData.append(it->key); yValues.append(it->value); } m_currentYData.append(yValues); } // 查找最近点(按X轴距离) for (int i = 0; i < m_currentXData.size(); ++i) { double distance = fabs(m_currentXData[i] - clickX); if (distance < minDistance) { minDistance = distance; m_currentIndex = i; } } } void Plotter::updateSelectionDisplay(QCustomPlot* plot) { clearSelection(plot); if (m_currentIndex < 0 || m_currentXData.isEmpty()) return; double x = m_currentXData[m_currentIndex]; // 绘制垂直线 m_selectionLine->setPen(QPen(Qt::red, 1, Qt::DashLine)); m_selectionLine->point1->setCoords(x, plot->yAxis->range().lower); m_selectionLine->point2->setCoords(x, plot->yAxis->range().upper); m_selectionLine->setVisible(true); // 显示坐标文本 QString coordText = QString("时间: %1\n").arg(x, 0, 'f', 2); for (int g = 0; g < m_currentYData.size(); g++) { if (m_currentIndex < m_currentYData[g].size()) { double y = m_currentYData[g][m_currentIndex]; coordText += QString("曲线%1: %2\n").arg(g+1).arg(y, 0, 'f', 2); // 绘制数据点标记 QCPItemEllipse* dot = new QCPItemEllipse(plot); dot->topLeft->setCoords(x-0.2, y+0.2); dot->bottomRight->setCoords(x+0.2, y-0.2); dot->setPen(QPen(g == 0 ? Qt::blue : Qt::red)); dot->setBrush(QBrush(g == 0 ? Qt::blue : Qt::red)); m_selectionDots.append(dot); } } m_selectionLabel->setText(coordText); m_selectionLabel->setFont(QFont("Arial", 9)); m_selectionLabel->position->setCoords(x, plot->yAxis->range().upper); m_selectionLabel->setPen(QPen(Qt::black)); m_selectionLabel->setBrush(QBrush(Qt::white)); m_selectionLabel->setPadding(QMargins(3, 3, 3, 3)); m_selectionLabel->setVisible(true); plot->replot(); } void Plotter::clearSelection(QCustomPlot* plot) { if (!plot) return; m_selectionLine->setVisible(false); m_selectionLabel->setVisible(false); for (auto dot : m_selectionDots) { plot->removeItem(dot); delete dot; } m_selectionDots.clear(); plot->replot(); } void Plotter::onNextPoint() { if (m_currentIndex < 0 || !m_currentCustomPlot || m_currentXData.isEmpty()) return; if (++m_currentIndex >= m_currentXData.size()) m_currentIndex = m_currentXData.size() - 1; updateSelectionDisplay(m_currentCustomPlot); } void Plotter::onPrevPoint() { if (m_currentIndex < 0 || !m_currentCustomPlot || m_currentXData.isEmpty()) return; if (--m_currentIndex < 0) m_currentIndex = 0; updateSelectionDisplay(m_currentCustomPlot); }理解这个类

MediaSessionService: dispatchVolumeKeyEvent, pkg=com.sgrl.pjj.factorymode, opPkg=com.sgrl.pjj.factorymode, pid=8753, uid=1000, asSystem=true, event=KeyEvent { action=ACTION_DOWN, keyCode=KEYCODE_VOLUME_DOWN, scanCode=114, metaState=0, flags=0x8, repeatCount=18, eventTime=424403976947, downTime=422339335000, deviceId=2, source=0x101, displayId=-1 }, stream=-2147483648, musicOnly=false 01-01 14:38:38.737202 1468 6566 D MediaSessionService: Adjusting suggestedStream=-2147483648 by -1. flags=4113, preferSuggestedStream=false, session=null 01-01 14:38:38.767065 2994 3482 I vol.Events: writeEvent level_changed STREAM_MUSIC 14 01-01 14:38:38.771218 2994 3482 I vol.Events: writeEvent key STREAM_MUSIC 14 01-01 14:38:38.781765 8753 8753 D VUMeter : amplitude =854 01-01 14:38:38.781907 8753 8753 D VUMeter : mCurrentAngle = 0.7897484 01-01 14:38:38.788692 2994 2994 I vol.VolumeDialogImpl: showH r=volume_changed 01-01 14:38:38.789350 2994 2994 I vol.VolumeDialogImpl: rescheduleTimeout 3000 com.android.systemui.volume.VolumeDialogImpl.-$$Nest$mshowH:45 01-01 14:38:38.790324 2994 2994 I vol.Events: writeEvent show_dialog volume_changed keyguard=false 01-01 14:38:38.790379 1468 6566 D MediaSessionService: dispatchVolumeKeyEvent, pkg=com.sgrl.pjj.factorymode, opPkg=com.sgrl.pjj.factorymode, pid=8753, uid=1000, asSystem=true, event=KeyEvent { action=ACTION_DOWN, keyCode=KEYCODE_VOLUME_DOWN, scanCode=114, metaState=0, flags=0x8, repeatCount=19, eventTime=424454178870, downTime=422339335000, deviceId=2, source=0x101, displayId=-1 }, stream=-2147483648, musicOnly=false 01-01 14:38:38.790621 1468 6566 D MediaSessionService: Adjusting suggestedStream=-2147483648 by -1. flags=4113, preferSuggestedStream=false, session=null 01-01 14:38:38.798833 786 886 I libPerfCtl: xgfGetFPS pid:2994 fps:57 01-01 14:38:38.798955 786 886 I libPerfCtl: xgfGetFPS pid:8753 fps:23 01-01 14:38:38.798973 786 886 I libPowerHal: [updateFPS] pid(8753) : -1 => 60 01-01 14:38:38.799314 786 906 I libPowerHal: [perfNotifyAppState] pack:com.sgrl.pjj.factorymode, act:com.sgrl.pjj.factorymode.testItem.MicTestActivity, state:5, pid:8753, uid:1000, fps:60 01-01 14:38:38.801122 786 906 E UxUtility: notifyAppState error = NULL 01-01 14:38:38.801175 786 906 I TouchUtility: notifyAppState error = NULL 01-01 14:38:38.810670 2994 3482 I vol.Events: writeEvent level_changed STREAM_MUSIC 13 01-01 14:38:38.813463 2994 3482 I vol.Events: writeEvent key STREAM_MUSIC 13 01-01 14:38:38.816819 2994 2994 D vol.VolumeDialogImpl: onAnimationCancel 01-01 14:38:38.820277 2994 2994 I vol.VolumeDialogImpl: showH r=volume_changed 01-01 14:38:38.820677 2994 2994 I vol.VolumeDialogImpl: rescheduleTimeout 3000 com.android.systemui.volume.VolumeDialogImpl.-$$Nest$mshowH:45 01-01 14:38:38.821256 2994 2994 I vol.Events: writeEvent show_dialog volume_changed keyguard=false 01-01 14:38:38.836439 1468 4396 D MediaSessionService: dispatchVolumeKeyEvent, pkg=com.sgrl.pjj.factorymode, opPkg=com.sgrl.pjj.factorymode, pid=8753, uid=1000, asSystem=true, event=KeyEvent { action=ACTION_DOWN, keyCode=KEYCODE_VOLUME_DOWN, scanCode=114, metaState=0, flags=0x8, repeatCount=20, eventTime=424504556332, downTime=422339335000, deviceId=2, source=0x101, displayId=-1 }, stream=-2147483648, musicOnly=false 01-01 14:38:38.836845 1468 4396 D MediaSessionService: Adjusting suggestedStream=-2147483648 by -1. flags=4113, preferSuggestedStream=false, session=null 01-01 14:38:38.865736 8753 8753 D VUMeter : amplitude =930 01-01 14:38:38.865825 8753 8753 D VUMeter : mCurrentAngle = 0.60974836 01-01 14:38:38.872666 2994 3482 I vol.Events: writeEvent level_changed STREAM_MUSIC 12 01-01 14:38:38.875468 2994 3482 I vol.Events: writeEvent key STREAM_MUSIC 12请分析这段log内容

一、目的和意义 随着人工智能技术的快速发展,智能语音处理作为其重要分支,正在深刻改变人机交互的方式。智能语音助手、语音识别、自然语言理解等技术已广泛应用于智能家居、智能客服、车载系统等领域,成为现代生活中不可或缺的一部分。本任务书旨在配合《智能语音处理综合实践》课程的教学目标,帮助学生通过实践掌握智能语音处理的核心技术,培养解决实际问题的能力。 本指导书适用于智能科学与技术及相关专业的学生,要求学习者具备一定的编程基础和智能语音处理基础知识。通过系列的学习,学生将能够全面了解智能语音处理的基本流程与关键技术,掌握从数据预处理到模型训练与评估的完整实践能力。同时,实践还将引导学生思考智能语音处理技术在实际应用中的挑战与解决方案,培养工程管理、文档撰写和团队协作等综合素质,为后续的学术研究或工程开发奠定坚实的基础。 [1] 掌握智能语音处理的核心技术,包括语音识别(ASR)、自然语言处理(NLP)及意图识别的实现流程。 [2] 理解预训练模型(如BERT、GPT等)在语音转文本、指令分类及槽位填充中的应用原理。 [3] 能够基于Python及开源工具(如Hugging Face Transformers、Speech Recognition)实现智能家居系统的语音交互功能。 [4] 通过实践对比不同模型在意图识别中的性能差异,分析其适用场景。 [5] 能够通过实践对比不同分词方法的优缺点。 二、考查项目及评分标准 1.考查项目 构建一个智能家居控制助手。 2.项目要求 (1)语音识别与文本转换: 任务目标:完成语音转文本模型的开发,实现以下核心功能: ①实时语音检测:通过麦克风持续监听环境音,基于音频能量阈值或语音活动检测(VAD)算法,自动识别用户语音的起始和结束节点,精准捕捉有效语音段。 ②智能录音控制:仅在检测到用户语音时触发音频录制,支持两种停止机制(完成一种即可): 静音超时自动停止:当检测到静音时长超过预设阈值(如 5 秒)时,自动终止录音; 手动按钮停止:提供交互按钮,允许用户随时主动结束录音。 ③语音转文本:对录制的音频进行降噪预处理后,通过语音识别模型将其转换为文本,输出语音识别结果。 ④环境自适应扩展:通过可调参数配置(如降噪等级、音量敏感度阈值),使模型能够适应不同环境条件(如背景噪音强度、用户说话音量差异),提升识别鲁棒性 ⑤结果展示: (2)数据集准备: ①要实现识别家居控制类文本并输出控制设备及指令,需要下述两个模型: 二分类模型:根据用户的语音或者文本输入的语句,判断当前句子是否与家居控制有关。 多分类模型:从单条语句中同时识别以下三元组属性,实现细粒度家居控制指令解析: 设备位置(如厨房、卧室等) 设备名称(如灯、空调等) 控制指令(开关打开或者关闭) ②针对以上模型开发任务,需构建适配的标注数据集: 二分类模型数据要求: 正样本:包含明确家居控制意图的语句(如 “打开客厅空调”“关闭卧室灯光”)。 负样本:非家居控制场景的语句(如 “今天天气如何”“帮我订外卖”)。 标注规则:每条样本需标注布尔型标签(1 = 家居控制类,0 = 非家居控制类)。 参考生成的数据集如下图所示: 多分类模型数据要求: 设备位置:需覆盖常见家居场景(如客厅、卧室、厨房、卫生间等,支持自定义扩展)。 设备名称:包含主流家电品类(如灯、空调、风扇、窗帘、电视等,需与地点逻辑匹配)。 控制指令:二值标签(0 = 关闭,1 = 开启;支持 “调节” 等模糊状态的语义映射处理)。 参考生成的数据集如下图所示: (3)二分类模型: 开发一个中文文本二分类模型,用于判断用户输入语句是否为家居控制指令(如开关灯光、调节空调等),实现以下核心功能。  对短文本进行语义理解与意图识别  准确区分家具控制指令与普通日常对话  保存最优模型参数并提供交互式预测接口  如下图所示: (4)多分类模型: 开发一个的多任务分类模型,从单条语句中同时识别以下三元组属性,实现细粒度家居控制指令解析: 设备位置(如厨房、卧室等) 设备名称(如灯、空调等) 控制指令(开关打开或者关闭) 如下图所示: (5)界面设计: 任务目标:设计智能家居可视化界面,实现以下核心功能: ①支持语音输入和键盘输入双模式指令交互 ②通过深度学习模型实现多维度指令分析(地点/设备/状态) ③提供直观的图形化操作界面,符合现代UI设计规范 ④实现实时录音状态反馈与静音端点检测功能 ⑤参考图如下:

大家在看

recommend-type

tanner pro 集成电路设计和布局实战指导 配套 CD

《tanner pro 集成电路设计和布局实战指导》配套CD , 包含多有例子:有Layout和 code,学习非常方便。
recommend-type

SmartSVN license

和谐SmartSVN试用版的license,下载后覆盖即可。(注意:覆盖前最好备份原license)
recommend-type

AAA2.5及汉化补丁

Advanced Aircraft Analysis V2.5.1.53 (3A) 在win7 64位上安装测试。有注册机和安装视频。支持winxp和win732位和64位系统。 Darcorp Advanced Aircraft Analysis V2.5.1.53 (AAA) 软件是一款面向于高级用户的飞机设计和仿真分析软件,目前广泛应用于数十个国家的各种机构,已然成为飞机设计、开发、稳定性分析以及飞行控制的工业标准软件。适用于 FAR23、FAR25、UAV无人驾驶飞机与 Military 规范,为全球飞机公司(如波音公司)、政府部门(如 FAA)与学校采用于飞机初步设计、分析、与 3-D 绘图的一套完整软件工具。 Advanced Aircraft Analysis (AAA) 是行业标准的飞机设计,稳定性和控制分析软件。 安装在超过45个国家,AAA所使用的主要航空工程大学,飞机制造商和世界各地的军事组织。 Advanced Aircraft Analysis(AAA)是行业标准的飞机设计 AAA提供了一个功能强大的框架,以支持飞机初步设计迭代和非独特的过程。 AAA计划允许学生和初步设计工程师从早期的大小通过开环和闭环动态稳定性和灵敏度分析的重量,而该机的配置工作在监管和成本的限制。
recommend-type

TDC-GP22资料.zip

TDC-GP22 STM32F103和MSP430源代码以及TDC中英文芯片手册,现场图片和测试总结
recommend-type

Frequency-comb-DPLL:数字锁相环软件,用于使用Red Pitaya锁定频率梳

数字锁相环,用于使用红色火龙果锁定频率梳 固件/软件允许使用此硬件来锁相频率梳。 更一般而言,它与硬件一起提供了一个数字控制盒,该数字控制盒可以支持双通道锁相环,包括输入rf信号的前端IQ检测。 因此,虽然此数字控制盒可用于锁相其他系统,但下面的讨论假定用户正在操作频率梳。 入门 从“发布部分”( )下载所需的文件: 可以访问Python GUI的完整源代码存储库; b。 红火龙果的SD卡映像(red_pitaya_dpll_2017-05-31.zip) 阅读并遵循“ RedPitaya DPLL.pdf的说明和操作手册”文件。 软件版本 所需的Python发行版是WinPython-64bit-3.7.2( )。 FPGA Vivado项目在Vivado 2015.4中进行了编译,但是仅使用该软件就不需要安装Vivado。 附加信息 可以从NIST数字控制箱的说明手册中获得更多信

最新推荐

recommend-type

Photoshop电子教案PPT学习课件.ppt

Photoshop电子教案PPT学习课件.ppt
recommend-type

移动通信技术与网络优化第6章路测.ppt

移动通信技术与网络优化第6章路测.ppt
recommend-type

大数据分析—安全.ppt

大数据分析—安全.ppt
recommend-type

Delphi 12.3控件之IOCompV404Sp2 fix OPC and display bugs (Blockke fix)for D12.7z

Delphi 12.3控件之IOCompV404Sp2 fix OPC and display bugs (Blockke fix)for D12.7z
recommend-type

用于学习的1553B总结资料

用于学习的1553B总结资料
recommend-type

19年国赛服务器答案深度解析:网络搭建与应用

网络搭建与应用是一门涉及计算机网络规划、配置、管理和维护的技术学科。在19年的国家竞赛中,参与者需要展示他们对网络架构、网络设备、协议、安全等方面的知识,以及他们在真实世界问题解决中的实际应用能力。在网络搭建与应用19国赛服务器答案中,涉及的知识点可能包括但不限于以下几个方面: 1. 网络基础知识 - 了解网络的基本概念,包括网络的定义、分类(如LAN、WAN等)、网络的功能和网络协议栈(如TCP/IP模型)。 - 理解网络设备的功能和作用,例如交换机、路由器、防火墙等。 - 掌握网络通信的基本原理,包括数据链路层、网络层、传输层和应用层的协议和功能。 2. 网络设计与规划 - 学习如何根据不同的需求设计网络拓扑结构,例如星形、环形、总线型等。 - 掌握IP地址规划和子网划分的方法,如CIDR、VLSM等技术。 - 了解如何进行网络流量分析和带宽规划,以确保网络性能和稳定性。 3. 网络设备配置与管理 - 掌握交换机和路由器的配置命令,例如VLAN划分、路由协议配置、端口安全等。 - 理解网络设备的管理和维护策略,包括日志管理、性能监控和故障诊断。 4. 网络安全 - 学习网络安全的基本原则,包括数据加密、访问控制、入侵检测系统(IDS)和入侵防御系统(IPS)。 - 掌握常见的网络攻击手段及其防御措施,例如DDoS攻击、ARP欺骗、病毒和恶意软件的防御。 5. 服务器搭建与应用 - 了解不同类型的服务器和它们的应用场景,如Web服务器、数据库服务器、邮件服务器等。 - 学习服务器的安装、配置和管理方法,包括操作系统的安装、服务软件的部署、用户管理等。 6. 实践操作 - 通过搭建真实或虚拟的网络环境来实践理论知识,包括使用网络模拟软件(如GNS3、Packet Tracer等)进行网络设备配置和故障排除。 - 学习如何准备和参加网络相关的竞赛,包括时间管理和应对突发问题的策略。 在给定的压缩包子文件的文件名称列表中,“19年国赛服务器的答案”表明该文件包含具体的竞赛解答步骤和方法。文件可能涉及的是如何根据具体的网络搭建与应用的题目,一步一步地提供解题策略、配置命令和执行步骤等。具体内容可能包括: - 对竞赛题目需求的分析和理解。 - 对应的网络设计解决方案和实现步骤。 - 关键的配置命令和参数设置。 - 对于各种网络功能和安全性的测试方法。 - 可能遇到的问题及其解决方案。 综上所述,对于网络搭建与应用19国赛服务器答案的深入学习和掌握,能够帮助专业人士和学生扩展网络知识,提高解决复杂网络问题的能力,并为将来的IT行业工作奠定扎实的基础。通过针对竞赛题目的实际操作演练,还能增强个人的实践技能和创新思维能力。
recommend-type

【VS2010模块化秘籍】:提升项目管理效率的10个技巧

# 1. VS2010模块化编程概述 在现代软件开发中,模块化编程已经成为提高代码质量和促进团队协作的关键技术之一。本章将介绍在Visual Studio 2010(VS2010)环境中如何理解和实施模块化编程。 ## 1.1 什么是模块化编程? 模块化编程是一种组织程序结构的方法,将程序分解为独立且可管理的模块,每个模块实现一个特定的功能。在V
recommend-type

数据分析师发展前景

<think>嗯,用户现在想了解数据分析师的职业发展和未来前景。根据之前的对话,用户是土木本科想转行,之前我建议过数据分析作为可能的方向之一,现在他可能想更深入了解这个领域的潜力。 首先,我需要参考之前提供的引用内容。引用1提到了数据分析师的职业路径,从专员到专家,还有技能提升的重要性。引用2和3强调了行业需求增长和技能的必要性,引用4则讨论了成为行业专家的长期发展优势。这些都需要整合到回答中。 用户的问题集中在职业前景和趋势,所以我要覆盖市场需求、职业阶段、技能要求、行业趋势和转行建议。考虑到用户是转行者,需要突出土木背景如何与数据分析结合,比如提到的BIM、GIS或者工程数据分析,这样
recommend-type

Elasticsearch及IK分词器安装包资源汇总

标题中提到的知识点包括Elasticsearch安装包和IK分词器,这是进行搜索引擎搭建和数据文本分析的重要组件。Elasticsearch是一个基于Lucene构建的开源搜索引擎,具有水平可伸缩性、高可用性和易用性的特点。它提供了全文搜索功能,同时支持结构化搜索和分析,常被用于大数据分析场景中。 描述中涉及的版本信息表明了所附的安装包和分词器支持不同版本的Elasticsearch。Elasticsearch版本6.x和7.x分别对应了两个主要的版本线,而IK分词器是专门为Elasticsearch设计的中文分词插件。 IK分词器是一款支持中文分词的扩展插件,可以根据中文语境进行智能分词,包括正向匹配、正向最大匹配和逆向最大匹配等算法,对中文文本进行处理。分词器的版本通常会与Elasticsearch的版本相匹配,以保证兼容性和最佳性能。 提到的logstash是与Elasticsearch配合使用的数据处理管道工具,负责收集、处理和转发数据。logstash可以作为事件的中介来处理各种来源的数据,然后将其发送到Elasticsearch进行存储。本压缩包中的logstash-6.4.3.tar.gz对应的版本表明了它的兼容性,适用于Elasticsearch 6.x版本。 压缩包文件名称列表中的文件包含了不同软件的多个版本。其中,“elasticsearch-head-master.zip”是一个可以对Elasticsearch进行可视化管理的Chrome插件,它提供了包括集群管理、索引管理、数据操作和查询在内的功能。 另外,“mysql-connector-java-5.1.41.jar”是一个MySQL数据库的Java驱动程序,用于连接Java应用程序和MySQL数据库,但这似乎与Elasticsearch及IK分词器直接关联不大,可能是一个辅助组件,用于在某些集成场景下将数据从MySQL迁移到Elasticsearch。 从标签内容来看,Elasticsearch被归类于源码软件、大数据和搜索引擎类别。它是一个重要的大数据处理组件,特别是在全文搜索和文本分析领域。在大数据背景下,Elasticsearch凭借其卓越的搜索和分析能力,已经成为企业构建和维护搜索引擎的首选技术之一。 总结来说,本次提供的压缩包包含了多个关键组件,它们共同支持构建一个功能强大的搜索引擎和数据分析平台。Elasticsearch自身及其配套的IK分词器、logstash和可视化插件elasticsearch-head,均对大数据和搜索领域有着重要意义。尽管这些工具具有复杂性,但它们的组合使用使得数据的索引、搜索、可视化和分析变得简单、快速和强大。
recommend-type

从零开始:Axure插件开发入门指南,构建自定义Chrome工具

# 摘要 随着原型设计工具Axure RP的广泛应用,定制化插件开发变得日益重要。本文旨在为开发者提供一个完整的Axure插件开发指南,从基础概念到环境搭建,再到核心编程和界面定制,最终实现交互逻辑与发布维护。通过详细解析插件的API、JavaScript编程、数据管理、界面设计及发布流程,本文帮助开发者深入理解并掌握Axure插件开发的关键技术,确保开发过程的高效性和插件的稳定运行。 # 关键字 Axur