Unity编辑器拓展-Odin

1.相比于原生Unity的优势

  • Unity不支持泛型类型序列化,例如字典原生Unity不支持序列化,而Odin可以继承序列化的Mono实现
  • 功能强大且使用简单,原生Unity想实现一些常见的功能需要额外自己编写Unity扩展的编码,实现功能只需要加一个特性即可Odin帮忙写好了内部管理和实现
  • 编辑器的窗口实现简单且美观

2.常用功能代码总结

通过Tools-Odin Inspector-Attribute Overview即可打开一个预览各个特性的效果的窗口,可供参考

OdinValueDrawer

类继承自OdinValueDrawer,其中T为自定义数据类型,之后重写DrawPropertyLayout方法,实现自定义绘制,实现了对于我们自定义数据类型的自定义绘制,重写Initialize来做对其下序列化部分的初始化,例如Selector的生成,选择变化的事件监听,初始值的定义。同时在DrawPropertyLayout中定义对应的按钮,在按下的时候用selector.ShowInPopup();打开对应的selector

#if UNITY_EDITOR
namespace Sirenix.OdinInspector.Demos
{using UnityEngine;using System;#if UNITY_EDITORusing Sirenix.OdinInspector.Editor;using UnityEditor;using Sirenix.Utilities;#endif// 演示如何为自定义类型生成自定义drawer的示例。[TypeInfoBox("此示例演示如何为自定义结构或类实现自定义drawer")]public class CustomDrawerExample : MonoBehaviour{public MyStruct MyStruct;[ShowInInspector]public static float labelWidth = 10;}// 自定义数据结构,用于演示。[Serializable]public struct MyStruct{public float X;public float Y;}#if UNITY_EDITORpublic class CustomStructDrawer : OdinValueDrawer<MyStruct>{protected override void DrawPropertyLayout(GUIContent label){//获取我们绘制类的值MyStruct value = this.ValueEntry.SmartValue;//获取要绘制的区域(rect)var rect = EditorGUILayout.GetControlRect();//在Odin中,标签是可选项,可以为空,所以我们必须考虑到这一点。if (label != null){rect = EditorGUI.PrefixLabel(rect, label);}//保存原始labelWidth的宽度,此label为struct中对应的X,Yvar prev = EditorGUIUtility.labelWidth;//设定新的label宽度EditorGUIUtility.labelWidth = CustomDrawerExample.labelWidth;//根据slider对应的值进行赋值value.X = EditorGUI.Slider(rect.AlignLeft(rect.width * 0.5f), "X", value.X, 0, 1);value.Y = EditorGUI.Slider(rect.AlignRight(rect.width * 0.5f), "Y", value.Y, 0, 1);//恢复设定原始label宽度EditorGUIUtility.labelWidth = prev;//将新的Struct赋值给我们定义的MyStructthis.ValueEntry.SmartValue = value;}}
#endif
}
#endif

OdinSelector

一个选择框,可以选择提供的列表,同时选择后触发对应的事件

BuildSelectionTree

BuildSelectionTree的作用是在OdinSelector中构建一个选择树,也就是显示在弹出窗口中的树形结构的列表。你可以通过重写这个方法,来决定选择树中包含哪些值,以及它们的层级关系和排序方式。

protected override void BuildSelectionTree(OdinMenuTree tree)  
{  
//搜索框tree.Config.AutoFocusSearchBar = true;  tree.Config.DrawSearchToolbar = true;  tree.Selection.SupportsMultiSelect = false;  tree.MenuItems.Clear();  foreach (结合excel枚举对应的pb)  {      tree.Add(path, instance);}}

在外部进行的初始化

例如在上述的Drawer的Init中进行初始化,需要new出这个类,并且增加监听事件,且设置初始值。同时监听外部按钮,打开这个selector

m_WeaponIDSelector = new WeaponIDSelector();  
m_WeaponIDSelector.SelectionTree.UpdateMenuTree();  
m_WeaponIDSelector.SetSelection(WeaponData.weaponKey.thingID); //初始值 
m_WeaponIDSelector.SelectionChanged += OnWeaponThingIDChanged;//改变时候监听
// 重写Initialize方法
protected override void Initialize()
{
// 调用基类的Initialize方法
base.Initialize();// 实例化OdinMenuTree对象
tree = new OdinMenuTree();// 添加一些菜单项到选择树中,例如一些GameObject或者其他自定义对象
tree.Add("Cube", GameObject.CreatePrimitive(PrimitiveType.Cube));
tree.Add("Sphere", GameObject.CreatePrimitive(PrimitiveType.Sphere));
tree.Add("MyObject", new MyObject());// 设置选择树的配置,例如是否支持多选,是否显示搜索栏等
tree.Config.DrawSearchToolbar = true;
tree.Config.MultiSelect = true;// 设置选择树的事件,例如当菜单项被选中或者双击时执行一些操作
tree.Selection.SelectionChanged += OnSelectionChanged;
tree.MenuItems.OnDoubleClick += OnDoubleClick;// 在Initialize的时候调用UpdateMenuTree方法,以便初始化选择树中的菜单项,以及处理搜索和排序等功能
tree.UpdateMenuTree();
}

在外部按钮按下进行的显示这个selector

//FocusType是一个枚举类型,用于表示按钮是否可以通过键盘选择。FocusType有三个可能的值:Passive、Keyboard和Native.Passive表示按钮不会接收键盘焦点,只能通过鼠标点击选择;Keyboard表示按钮可以通过Tab键或方向键在其他控件之间切换焦点;Native表示按钮使用平台本地的焦点行为,例如在Mac上使用Option+Tab键切换焦点.在你的代码中,FocusType.Passive表示你的下拉按钮不需要通过键盘选择,只能通过鼠标点击打开下拉菜单。
if (EditorGUILayout.DropdownButton(new GUIContent("武器选择: " + m_WeaponIDSelector.CurSelectName), FocusType.Passive))  
{  m_WeaponIDSelector.RebuildSelectionTree();  m_WeaponIDSelector.EnsureSingleClickToSelect();  m_WeaponIDSelector.ShowInPopup();  
}
public void RebuildSelectionTree()  
{  List<uint> curSelections = GetCurrentSelection().ToList();  BuildSelectionTree(SelectionTree);  EnableIDChangeEvent = false;  SetSelection(curSelections);  EnableIDChangeEvent = true;  SelectionTree.Config.AutoFocusSearchBar = true;  
}
public void EnsureSingleClickToSelect()  
{  
//设置选择树的Config.SelectionConfirmNavigationKey属性为KeyCode.None,这样就可以取消双击确认选择的功能EnableSingleClickToSelect();  //设置了SelectionTree.Config.ConfirmSelectionOnDoubleClick属性为false,这个属性也是用于控制是否需要双击确认选择的功能虽然这个属性和Config.SelectionConfirmNavigationKey属性有重复的作用,但是为了保险起见,最好都设置一下。SelectionTree.Config.ConfirmSelectionOnDoubleClick = false;  //设置了SelectionTree.Config.AutoFocusSearchBar属性为true,这个属性可以让选择树在打开时自动聚焦搜索栏,方便用户输入搜索关键词SelectionTree.Config.AutoFocusSearchBar = true;  
}

处理Selector的改变

这里需要注意的是要用var first = col.FirstOrDefault(); 来取,并且需要做判空,一般是取完keyID之后再去读表刷上其他的数据,或者是根据

selector.SelectionChanged += col => //添加SelectionChanged事件的处理方法
{
var first = col.FirstOrDefault(); //获取第一个选择的对象
if (first != null) //如果不为空
{
selectedName = first.name; //更新selectedName为对象的名称
}

缩进级别

  • GUIHelper.PushIndentLevel(1);
  • GUIHelper.PopIndentLevel();
    结合使用,增加一个缩进级别,可以让GUI的元素更加有层次感和结构化。例如,你可以用这个方法来创建一个树形的菜单或列表。

结构化API

  • EditorGUILayout.BeginHorizontal();这两个老朋友了就不解释了
  • EditorGUILayout.BeginVertical();
  • GUIHelper.PushGUIEnabled(bool);//表示接下来的一个范围内我要检查一下这个bool值来觉得这些UI是否可以被选
  • GUIHelper.PopGUIEnabled();
//GUIHelper.PushGUIEnabled和GUIHelper.PopGUIEnabled是两个用于控制GUI元素是否可用的方法。它们的作用是在一段代码中临时改变GUI元素的可用状态,而不影响其他代码中的GUI元素。具体来说,GUIHelper.PushGUIEnabled方法可以接受一个布尔值作为参数,表示要设置的GUI元素的可用状态。这个方法会将当前的GUI元素的可用状态压入一个栈中,然后设置新的可用状态。GUIHelper.PopGUIEnabled方法则会从栈中弹出之前保存的GUI元素的可用状态,并恢复它。这样就可以在一段代码中临时禁用或启用一些GUI元素,而不影响其他代码中的GUI元素。
GUIHelper.PushGUIEnabled(toggle); //根据复选框的状态设置按钮的可用状态
if (GUILayout.Button("Click Me")) //绘制按钮会受到是否可选的影响
{
Debug.Log("You clicked the button!"); //打印日志
}
GUIHelper.PopGUIEnabled(); //恢复之前的可用状态

补充

可以通过UnityToolbarExtender来打开目标窗口,使用起来也非常简单

static ToolbarGUIUtility()  
{  uxToolPrefabOpen = EditorPrefs.GetBool("UXToolPrefabOpen", false);  uxToolPowerOpen = EditorPrefs.GetBool("UXToolPowerOpen", false);  ToolbarExtender.LeftToolbarGUI.Add(OnLeftToolbarGUI);  ToolbarExtender.RightToolbarGUI.Add(OnRightToolbarGUI);  
}
private static GUIStyle ToolbarButtonLeftStyle  
{  get  {  if (toolbarButtonLeftStyle == null) toolbarButtonLeftStyle = new GUIStyle("toolbarbuttonLeft");  return toolbarButtonLeftStyle;  }}
static void OnLeftToolbarGUI()  
{  if (TestCommon.TestHelper.ColorButton(new GUIContent("XXX配置", "XXXToolTip"), new Color(0.55f, 0.37f, 1f), GUILayoutOptions.Width(90), ToolbarButtonLeftStyle))  {        TestWindow.OpenWindow();  }

快速创建Button的管理者类

#if UNITY_EDITOR  
using System;  
using System.Collections.Generic;  
using System.Reflection;  
using Sirenix.Utilities;  
using Sirenix.Utilities.Editor;  
using UnityEditor;  
using UnityEngine;  namespace TestCommon
{  public static class TestHelper{  private static GUIStyle ColoredButtonStyle = new GUIStyle(GUI.skin.button);  public static bool ColorButton(string text, Color color, int width, int height, GUIStyle guiStyle = null, GUIStyle labelGUIStyle = null)  {            ColoredButtonStyle.normal.textColor = Color.white;  GUIHelper.PushColor(color);  Rect buttonRect = GUILayoutUtility.GetRect(width, height);  bool res = guiStyle != null ? GUI.Button(buttonRect, "", guiStyle) : GUI.Button(buttonRect, "");  GUIHelper.PopColor();  GUI.Label(buttonRect, text, labelGUIStyle ?? SirenixGUIStyles.LabelCentered);  return res;  }  public static bool ColorButton(string text, Color color, GUILayoutOption[] guiLayoutOptions, GUIStyle guiStyle = null, GUIStyle labelGUIStyle = null)  {            ColoredButtonStyle.normal.textColor = Color.white;  GUIHelper.PushColor(color);  GUIHelper.PushContentColor(Color.clear);  bool res = guiStyle != null ? GUILayout.Button(text, guiStyle, guiLayoutOptions) : GUILayout.Button(text, guiLayoutOptions);  GUIHelper.PopContentColor();  GUIHelper.PopColor();  Rect buttonRect = GUILayoutUtility.GetLastRect();  GUI.Label(buttonRect, text, labelGUIStyle ?? SirenixGUIStyles.LabelCentered);  return res;  }  public static bool ColorButton(GUIContent guiContent, Color color, GUILayoutOption[] guiLayoutOptions, GUIStyle guiStyle = null, GUIStyle labelGUIStyle = null)  {            ColoredButtonStyle.normal.textColor = Color.white;  GUIHelper.PushColor(color);  GUIHelper.PushContentColor(Color.clear);  bool res = guiStyle != null ? GUILayout.Button(guiContent, guiStyle, guiLayoutOptions) : GUILayout.Button(guiContent, guiLayoutOptions);  GUIHelper.PopContentColor();  GUIHelper.PopColor();  Rect buttonRect = GUILayoutUtility.GetLastRect();  GUI.Label(buttonRect, guiContent.text, labelGUIStyle ?? SirenixGUIStyles.LabelCentered);  return res;  }  public static bool ColorButton(Rect rect, string text, Color color, GUIStyle guiStyle = null, GUIStyle labelGUIStyle = null)  {            GUIHelper.PushColor(color);  bool res = guiStyle != null ? GUI.Button(rect, "", guiStyle) : GUI.Button(rect, "");  GUIHelper.PopColor();  GUI.Label(rect, text, labelGUIStyle ?? SirenixGUIStyles.LabelCentered);  return res;  }  public static bool ColorButton(Rect rect, Texture2D texture2D, Color color, GUIStyle guiStyle = null)  {            GUIHelper.PushColor(color);  bool res = guiStyle != null ? GUI.Button(rect, texture2D, guiStyle) : GUI.Button(rect, texture2D);  GUIHelper.PopColor();  return res;  }  public static void RichLabel(Rect rect, string text, Color color, int fontSize = 12, bool italic = false, bool bold = false, GUIStyle guiStyle = null)  {            string richText = text;  richText = $"<size={fontSize}><color=#{ColorUtility.ToHtmlStringRGB(color)}>{richText}</color></size>";  if (italic) richText = $"<i>{richText}</i>";  if (bold) richText = $"<b>{richText}</b>";  if (guiStyle != null)  GUI.Label(rect, richText, guiStyle);  else  GUI.Label(rect, richText);  }  public static void RichLabel(Rect rect, GUIContent guiContent, Color color, int fontSize = 12, bool italic = false, bool bold = false, GUIStyle guiStyle = null)  {            string richText = guiContent.text;  richText = $"<size={fontSize}><color=#{ColorUtility.ToHtmlStringRGB(color)}>{richText}</color></size>";  if (italic) richText = $"<i>{richText}</i>";  if (bold) richText = $"<b>{richText}</b>";  guiContent.text = richText;  if (guiStyle != null)  GUI.Label(rect, guiContent, guiStyle);  else  GUI.Label(rect, guiContent);  }  public static void RichLabel(string text, Color color, int fontSize = 12, bool italic = false, bool bold = false, GUIStyle guiStyle = null, GUILayoutOption[] guiLayoutOptions = null)  {            string richText = text;  richText = $"<size={fontSize}><color=#{ColorUtility.ToHtmlStringRGB(color)}>{richText}</color></size>";  if (italic) richText = $"<i>{richText}</i>";  if (bold) richText = $"<b>{richText}</b>";  if (guiStyle != null)  {                if (guiLayoutOptions != null)  GUILayout.Label(richText, guiStyle, guiLayoutOptions);  else  GUILayout.Label(richText, guiStyle);  }            else  {  if (guiLayoutOptions != null)  GUILayout.Label(richText, guiLayoutOptions);  else  GUILayout.Label(richText);  }        }  public static void RichLabel(GUIContent guiContent, Color color, int fontSize = 12, bool italic = false, bool bold = false, GUIStyle guiStyle = null, GUILayoutOption[] guiLayoutOptions = null)  {            string richText = guiContent.text;  richText = $"<size={fontSize}><color=#{ColorUtility.ToHtmlStringRGB(color)}>{richText}</color></size>";  if (italic) richText = $"<i>{richText}</i>";  if (bold) richText = $"<b>{richText}</b>";  guiContent.text = richText;  if (guiStyle != null)  {                if (guiLayoutOptions != null)  GUILayout.Label(guiContent, guiStyle, guiLayoutOptions);  else  GUILayout.Label(guiContent, guiStyle);  }            else  {  if (guiLayoutOptions != null)  GUILayout.Label(guiContent, guiLayoutOptions);  else  GUILayout.Label(guiContent);  }        }  public enum Alignment  {  UpperLeft,  UpperCenter,  UpperRight,  MiddleLeft,  MiddleCenter,  MiddleRight,  BottomLeft,  BottomCenter,  BottomRight,  }  public static bool IsUpper(this Alignment alignment)  {            return alignment == Alignment.UpperRight || alignment == Alignment.MiddleRight || alignment == Alignment.BottomRight;  }  public static bool IsMiddle(this Alignment alignment)  {            return alignment == Alignment.MiddleLeft || alignment == Alignment.MiddleCenter || alignment == Alignment.MiddleRight;  }  public static bool IsBottom(this Alignment alignment)  {            return alignment == Alignment.BottomLeft || alignment == Alignment.BottomCenter || alignment == Alignment.BottomRight;  }  public static bool IsLeft(this Alignment alignment)  {            return alignment == Alignment.UpperLeft || alignment == Alignment.MiddleLeft || alignment == Alignment.BottomLeft;  }  public static bool IsRight(this Alignment alignment)  {            return alignment == Alignment.UpperRight || alignment == Alignment.MiddleRight || alignment == Alignment.BottomRight;  }  public static bool IsCenter(this Alignment alignment)  {            return alignment == Alignment.UpperCenter || alignment == Alignment.MiddleCenter || alignment == Alignment.BottomCenter;  }  public static void DrawResponsiveLayout(int rowWidth, List<int> elementWidths, Action<int> drawActions, Alignment alignment)  {            int accumulatedWidth = 0;  GUILayout.BeginVertical(GUILayoutOptions.ExpandWidth(false).ExpandHeight(false));  {                if (alignment.IsMiddle() || alignment.IsBottom()) GUILayout.FlexibleSpace();  GUILayout.BeginHorizontal();  if (alignment.IsRight() || alignment.IsCenter()) GUILayout.FlexibleSpace();  for (int index = 0; index < elementWidths.Count; index++)  {                    int elementWidth = elementWidths[index] + 4;  if (accumulatedWidth + elementWidth > rowWidth)  {                        accumulatedWidth = 0;  if (alignment.IsLeft() || alignment.IsCenter()) GUILayout.FlexibleSpace();  GUILayout.EndHorizontal();  GUILayout.BeginHorizontal();  if (alignment.IsRight() || alignment.IsCenter()) GUILayout.FlexibleSpace();  }  accumulatedWidth += elementWidth;                    drawActions.Invoke(index);  }  if (alignment.IsLeft() || alignment.IsCenter()) GUILayout.FlexibleSpace();  GUILayout.EndHorizontal();  if (alignment.IsMiddle() || alignment.IsUpper()) GUILayout.FlexibleSpace();  }            GUILayout.EndVertical();  }  [InitializeOnLoad]  public class ScrollableTextArea  {  private delegate string ScrollableTextAreaInternalDelegate(  Rect position,  string text,  ref Vector2 scrollPosition,  GUIStyle style);  private static readonly ScrollableTextAreaInternalDelegate EditorGUI_ScrollableTextAreaInternal;  static ScrollableTextArea()  {                MethodInfo method = typeof(EditorGUI).GetMethod("ScrollableTextAreaInternal", BindingFlags.Static | BindingFlags.Public | BindingFlags.NonPublic);  if (method == null)  return;  EditorGUI_ScrollableTextAreaInternal = (ScrollableTextAreaInternalDelegate)Delegate.CreateDelegate(typeof(ScrollableTextAreaInternalDelegate), method);  }  public static string EditorGUITextArea(Rect rect, string text, ref Vector2 scrollPos, int minLines = 0, int maxLines = 10)  {                return EditorGUI_ScrollableTextAreaInternal(rect, text, ref scrollPos, EditorStyles.textArea);  }  public static string EditorGUILayoutTextArea(string text, ref Vector2 scrollPos, int minLines = 0, int maxLines = 10)  {                float height = 32f + (float)((Mathf.Clamp(Mathf.CeilToInt(EditorStyles.textArea.CalcHeight(GUIHelper.TempContent(text), GUIHelper.ContextWidth) / 13f), minLines, maxLines) - 1) * 13);  Rect controlRect = EditorGUILayout.GetControlRect(false, height);  return EditorGUI_ScrollableTextAreaInternal(controlRect, text, ref scrollPos, EditorStyles.textArea);  }        }  public static Rect BeginColorBox(string label, Color color, params GUILayoutOption[] options)  {            Rect rect = SirenixEditorGUI.BeginBox(options);  SirenixEditorGUI.DrawSolidRect(rect, color);  SirenixEditorGUI.DrawBorders(rect, 1, new Color(0.24f, 0.24f, 0.24f));  SirenixEditorGUI.BeginBoxHeader();  float fieldWidth = EditorGUIUtility.fieldWidth;  EditorGUIUtility.fieldWidth = 10f;  Rect controlRect = EditorGUILayout.GetControlRect(false);  EditorGUIUtility.fieldWidth = fieldWidth;  GUI.Label(controlRect, label);  SirenixEditorGUI.EndBoxHeader();  return rect;  }  public static void EndColorBox()  {            SirenixEditorGUI.EndBox();  }  public static Color Grey(float value)  {            return new Color(value, value, value);  }  public static Color WithGrey(this Color color, float value)  {            return new Color(color.r * value, color.g * value, color.b * value);  }  public static Color WithAlpha(this Color color, float alpha)  {            color.a = alpha;  return color;  }    }}  
#endif

参考

十分钟入门Unity革命性编辑器拓展插件——Odin_哔哩哔哩_bilibili
Odin常用功能整理 | 登峰造极者,殊途亦同归。 (lfzxb.top)
marijnz/unity-toolbar-extender at dbd76ed996483d219c39d240f785b1790aa039ed (github.com)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/97005.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Django 前端模板显示换行符、日期格式

linebreaksbr 显示换行符 <td>{{ data.sku_list|default:"无"|linebreaksbr }}</td> date:"Y年m月d日 H:i" 设置日期格式 <td>{{ data.submit_time|date:"Y年m月d日 H:i" }}</td> 其他语法 forloop 获取循环的索引 …

【audio】alsa pcm音频路径

文章目录 AML方案音频路径分析dump alsa pcm各个音频路径的原始音频流数据 AML方案音频路径分析 一个Audio Patch用来表示一个或多个source端到一个或多个sink端。这个是从代码的注释翻译来的&#xff0c;大家可以把它比作大坝&#xff0c;可以有好几个入水口和出水口&#xf…

nginx 配置

一、nginx安装 下载地址&#xff1a;http://nginx.org/en/download.html&#xff0c;和Keepalived搭配使用&#xff0c;防止nginx挂掉 二、nginx配置 ########### 每个指令必须有分号结束。################# #user administrator administrators; #配置用户或者组&#xf…

RPA的安全风险及应对策略

RPA已经深度革新了工作流程&#xff0c;大大提升效率并减少了人为错误&#xff0c;使企业运营更加高效。据预测&#xff0c;至2030年&#xff0c;全球RPA市场将以39.9%的复合年增长率持续发展&#xff0c;这显示了RPA对企业生产力的巨大推动力。 RPA能够承担人类的繁琐工作&am…

【新书推荐】当 Python 遇到 ChatGPT —— 自动化办公落地

文章目录 当 Python 遇到 ChatGPT&#xff1a;一种强大的组合1. 文本生成2. 自动翻译3. 对话生成4. 情感分析 新书推荐《Python自动化办公应用大全&#xff08;ChatGPT版&#xff09;&#xff1a;从零开始教编程小白一键搞定烦琐工作&#xff08;上下册&#xff09;》前言内容简…

Linux--FTP服务器功能--项目

一个子进程负责一条连接通道 一、项目要求&#xff1a;实现以下内容 客户端支持的指令&#xff1a; 远程控制&#xff1a; 1、获取服务器当前路径的文件&#xff1a;get xxx2、 展示服务器有哪些文件&#xff1a;ls3、进入服务器某文件夹 &#xff1a;cd xxx4、上传文件到服…

Neo4j深度学习

Neo4j的简介 Neo4j是用Java实现的开源NoSQL图数据库。从2003年开始开发&#xff0c;2007年正式发布第一版&#xff0c;其源码托管于GitHtb。Neo4j作为图数据库中的代表产品&#xff0c;已经在众多的行业项目中进行了应用&#xff0c;如&#xff1a;网络管理、软件分析、组织和…

国内手机安装 Google Play 服务 (GMS/Google Mobile Services)

目录 1. 国内手机安装 Google Play 服务 (GMS/Google Mobile Services)1.1. 什么是 GMS1.2. 国内手机只需要安装 3 个 APP1.2.1. Google Services Framework 服务框架1.2.2. Google Play Services1.2.3. Google Play Store 应用商店 1.3. 问题1.3.1. 谷歌地图闪退 2. 小米手机 …

gorm库的Find方法引发的问题

一、问题引入&#xff1f; 笔者在学习一个项目时&#xff0c;有一个登录需求&#xff0c;在登录时需要判断用户是否存在&#xff0c;特引入了Find方法做查询&#xff0c;然后根据返回值做判断&#xff0c;没想到因为Find的特性&#xff0c;导致判断存在问题&#xff0c;不管用…

C# App.xaml.cs的一些操作

一、保证只有一个进程 1.1 关闭旧的&#xff0c;打开新的 protected override void OnStartup(StartupEventArgs e) {base.OnStartup(e);var process Process.GetProcessesByName("Dog");if (process.Count() > 1) {var list process.ToList();list.Sort((p1,p2…

DirectX12_Windows_GameDevelop_3:Direct3D的初始化

引言 查看龙书时发现&#xff0c;第四章介绍预备知识的代码不太利于学习。因为它不像是LearnOpenGL那样从头开始一步一步教你敲代码&#xff0c;导致你没有一种整体感。如果你把它当作某一块的代码进行学习&#xff0c;你跟着敲会发现&#xff0c;总有几个变量是没有定义的。这…

乌班图22.04 kubeadm简单搭建k8s集群

1. 我遇到的问题 任何部署类问题实际上对于萌新来说都不算简单&#xff0c;因为没有经验&#xff0c;这里我简单将部署的步骤和想法给大家讲述一下 2. 简单安装步骤 准备 3台标准安装的乌班图server22.04&#xff08;采用vm虚拟机安装&#xff0c;ip为192.168.50.3&#xff0…

STC89C51基础及项目第10天:LCD显示字符(非标协议外设)

1. 初识LCD1602&#xff08;233.79&#xff09; 非标协议外设 LCD1602显示 LCD1602&#xff08;Liquid Crystal Display&#xff09;是一种工业字符型液晶&#xff0c;能够同时显示 1602 即 32 字符(16列两行) 引脚说明 第 1 脚&#xff1a; VSS 为电源地第 2 脚&#xff1…

Spark SQL 外部数据源

1.简介 1.1 多数据源支持 Spark 支持以下六个核心数据源,同时 Spark 社区还提供了多达上百种数据源的读取方式,能够满足绝大部分使用场景。 - CSV - JSON - Parquet - ORC - JDBC/ODBC connections - Plain-text files 1.2 读数据格式 所有读取 API 遵循以下调用格式: // …

SpringBoot项目默认使用HikariDataSource数据库连接池修改使用Druid连接池

1.启动项目&#xff0c;查看正在使用的链接池。 2.在pom.xml文件中引入驱动 <dependency><groupId>com.alibaba</groupId><artifactId>druid-spring-boot-starter</artifactId><version>1.2.8</version></dependency> 3.在ap…

【不定期更新】什么是Linux内核

2023年10月7日&#xff0c;周日晚上 目录 硬件资源管理 进程调度 文件系统管理 设备驱动 网络功能 Linux内核中的主要网络功能 安全 系统调用接口 Linux内核提供了哪些系统调用接口&#xff1f; 总结 Linux内核是Linux操作系统的核心&#xff0c;它起到以下主要的作用…

机器视觉工程师,公司设置奖金,真的为了奖励你吗?其实和你没关系

​据说某家大厂&#xff0c;超额罚款&#xff0c;有奖有罚很正常&#xff0c;但是我觉得你罚款代理商员工就不一样了&#xff0c;把代理商当成你的员工&#xff0c;我就觉得这些大厂的脑回路有问题。 有人从来没听说过项目奖金&#xff0c;更没有奖金。那么为什么设置奖金呢&a…

数字化转型频频失败?一体化模式提供新的思考

数字化连续6年出现在政府报告中&#xff0c;从《中小企业数字化赋能专项行动方案》到《关于推进“上云用数赋智”行动》、《“十四五” 规划和 2035 年远景目标建议》、《中小企业数字化转型指南》&#xff0c;再到2023年2月《数字中国建设整体布局规划》&#xff0c;加快数字化…

Java spring boot常用注解

Spring Boot是一个基于Spring框架的开发框架&#xff0c;它简化了Spring应用的配置和部署过程&#xff0c;提供了一系列注解来帮助开发人员快速构建可靠和高效的应用程序。本文将详细介绍Spring Boot中常用的注解及其使用方法。 Spring Boot注解可以分为以下几个方面&#xff…

MM-Camera架构-ProcessCaptureRequest 流程分析

文章目录 processCaptureRequest\_3\_41.1 mDevice1.2 mDevice->ops->process\_capture\_request1.3 hardware to vendor mct\_shimlayer\_process\_event2.1 mct\_shimlayer\_handle\_parm2.2 mct\_shimlayer\_reg\_buffer processCaptureRequest_3_4 sdm660的摄像头走…