【JMX】JAVA监控的基石

目录

1.概述

2.MBean

2.1.Standard MBean

2.2.Dynamic MBean

2.3.Model Bean

2.4.Dynamic MBean和Model Bean的区别

2.5.MXBean

2.6.Open Bean

3.控制台


1.概述

什么是JMX,首先来看一段对话:

Java Management Extensions(JMX)是一种Java标准,用于管理和监控Java应用程序,特别是分布式系统。它提供了一种标准化的方式来管理应用程序的各种方面,包括性能监控、配置更改、事件通知等。目前JMX最常用的就是用来做JAVA程序的监控,市面上常见的Java 监控框架基本都是基于JMX来实现的。

JMX分为三部分:

  1. MBean:MBean是用来提供功能的,MBean是管理对象的标准接口,通过MBean可以获取和操作应用程序的各种属性和操作。Java提供了几种类型的MBean,包括标准MBean、动态MBean、模型MBean等,每种类型适用于不同的场景。

  2. MBean服务器:MBean服务器是用来管理MBean的,它是一个注册表,用于注册和管理MBean。应用程序通过MBean服务器暴露MBean,并且其他应用程序或工具可以连接到MBean服务器以检索和操作这些MBean。

  3. JMX代理:JMX代理是用来访问MBean服务器的,可以理解为CS架构中的client端,支持多种访问方式。JMX代理是一个位于应用程序和MBean服务器之间的中间层,用于管理远程应用程序。它允许远程管理工具连接到远程应用程序的MBean服务器,并执行各种管理操作,如监控、配置和诊断。

img

JMX标准的实现通常由Java虚拟机(JVM)提供支持。Java虚拟机会实现JMX标准,以便应用程序可以利用JMX的功能。JVM会提供一个MBean服务器,应用程序可以将MBean注册到该服务器中,使其可供其他应用程序或工具访问和操作。

Java虚拟机通过提供标准的MBean服务器和相关的API,为应用程序提供了一种标准化的方式来管理和监控应用程序的各个方面。这种标准化使得开发人员可以使用相同的接口和工具来管理不同的Java应用程序,从而提高了管理的效率和一致性。

2.MBean

MBean服务器由于是JVM实现的,所以对我们应用开发者来说,最核心的其实就是MBean了,因为MBean是用来实现功能的。JMX标准中将MBean分为以下几种:

  1. Standard MBean(标准 MBean):它是最简单的一种 MBean 类型。一个标准 MBean 由一个 Java 接口和一个对应的实现类组成。接口中定义了 MBean 的属性和操作,实现类则提供了对应的实现。标准 MBean 的命名约定为 YourInterfaceNameMBean

  2. Dynamic MBean(动态 MBean):与标准 MBean 不同,动态 MBean 可以在运行时动态地确定其属性和操作。动态 MBean 的属性和操作是在运行时通过方法调用来确定的,因此可以根据需要进行动态修改。

  3. Model MBean(模型 MBean):模型 MBean 是一种特殊的 MBean,它通过描述符来描述其管理接口,可以在运行时动态地配置其属性和操作。模型 MBean 可以通过 XML 或者通过程序动态地加载描述符。

  4. MXBean:MXBean 是一种专门用于 JMX 的 MBean 类型。它定义了一种简单的 Java 接口和实现约定,用于描述 MBean 的属性和操作,同时这些接口和实现类遵循 Java Bean 的命名约定。MXBean 在 JMX 中用于管理特定类型的对象,例如内存使用情况、线程信息等。

  5. Open MBean:开放 MBean 是一种通用的 MBean 类型,它可以包含各种类型的属性和操作。开放 MBean 不限制其属性和操作的类型,因此可以包含任意类型的数据。

2.1.Standard MBean

所谓的Standard MBean,就是说按照JMX的命名规范来定义接口然后注册到MBean Server中,MBean就能被JMX的client端访问到。具体该MBean提供了什么功能,在这个自定义的MBean中去定义属性和方法仔细实现即可。

代码示例:

// 定义一个接口,命名约定为 YourInterfaceNameMBean
public interface HelloMBean {String getName();void setName(String name);void sayHello();
}
​
// 实现上述接口的类
public class Hello implements HelloMBean {private String name;
​@Overridepublic String getName() {return name;}
​@Overridepublic void setName(String name) {this.name = name;}
​@Overridepublic void sayHello() {System.out.println("Hello, " + name + "!");}
}
​
// 主程序
public class Main {public static void main(String[] args) throws Exception {// 创建标准 MBean 实例HelloMBean helloMBean = new Hello();// 创建 MBeanServerMBeanServer mbs = ManagementFactory.getPlatformMBeanServer();// 创建 ObjectName,用于标识 MBeanObjectName name = new ObjectName("com.example:type=Hello");// 注册 MBean 到 MBeanServermbs.registerMBean(helloMBean, name);// 等待一段时间,使得可以通过 JConsole 或者其他 JMX 客户端来操作 MBeanThread.sleep(Long.MAX_VALUE);}
}

2.2.Dynamic MBean

动态 MBean 是 Java Management Extensions (JMX) 中的一种特殊类型的 MBean,它与标准 MBean 不同之处在于,动态 MBean 允许在运行时动态地确定其属性和操作,这使得动态 MBean 能够在运行时根据需要动态地调,而不是在编译时就确定了。

说人话就是很方便动态的去修改Dynamic MBean的属性,这非常适合拿来做一些动态的功能,比如:

  • 动态配置管理,从外部去加载配置文件,实现配置的动态变化。

  • 动态监控,从外部去获取监控信息,比如实时的CPU、内存的信息,实现对系统的动态监控。

动态 MBean 的核心是实现 DynamicMBean 接口,该接口定义了一组方法,用于动态地获取和设置 MBean 的属性,执行 MBean 的操作,以及获取 MBean 的元数据信息。以下是 DynamicMBean 接口的几个核心方法:

  • getAttribute(String attribute):根据属性名获取属性值。

  • setAttribute(Attribute attribute):设置指定属性的值。

  • getAttributes(String[] attributes):批量获取属性值。

  • setAttributes(AttributeList attributes):批量设置属性值。

  • invoke(String actionName, Object[] params, String[] signature):执行指定的操作。

  • getMBeanInfo():获取 MBean 的元数据信息,包括属性、操作和描述信息。

代码示例:

import javax.management.*;
import java.lang.management.ManagementFactory;
​
public class DynamicHello implements DynamicMBean {private String name;
​@Overridepublic Object getAttribute(String attribute) throws AttributeNotFoundException, MBeanException, ReflectionException {if (attribute.equals("Name")) {return name;} else {throw new AttributeNotFoundException("Attribute '" + attribute + "' not found");}}
​@Overridepublic void setAttribute(Attribute attribute) throws AttributeNotFoundException, InvalidAttributeValueException, MBeanException, ReflectionException {if (attribute.getName().equals("Name")) {name = (String) attribute.getValue();} else {throw new AttributeNotFoundException("Attribute '" + attribute.getName() + "' not found");}}
​@Overridepublic AttributeList getAttributes(String[] attributes) {AttributeList attrs = new AttributeList();for (String attr : attributes) {try {Object value = getAttribute(attr);attrs.add(new Attribute(attr, value));} catch (Exception e) {// Ignore if attribute not found}}return attrs;}
​@Overridepublic AttributeList setAttributes(AttributeList attributes) {AttributeList responseAttrs = new AttributeList();for (Attribute attr : attributes.asList()) {try {setAttribute(attr);responseAttrs.add(new Attribute(attr.getName(), getAttribute(attr.getName())));} catch (Exception e) {// Ignore if attribute not found or set failed}}return responseAttrs;}
​@Overridepublic Object invoke(String actionName, Object[] params, String[] signature) throws MBeanException, ReflectionException {if (actionName.equals("sayHello")) {return "Hello, " + name + "!";} else {throw new UnsupportedOperationException("Operation '" + actionName + "' not supported");}}
​@Overridepublic MBeanInfo getMBeanInfo() {MBeanAttributeInfo[] attributes = new MBeanAttributeInfo[]{new MBeanAttributeInfo("Name", "java.lang.String", "Name of the person", true, true, false)};
​MBeanOperationInfo[] operations = new MBeanOperationInfo[]{new MBeanOperationInfo("sayHello", "Say hello to the person", null, "java.lang.String", MBeanOperationInfo.ACTION)};
​return new MBeanInfo(this.getClass().getName(), "Dynamic Hello MBean", attributes, null, operations, null);}
​public static void main(String[] args) throws Exception {DynamicHello dynamicHello = new DynamicHello();MBeanServer mbs = ManagementFactory.getPlatformMBeanServer();ObjectName name = new ObjectName("com.example:type=DynamicHello");mbs.registerMBean(dynamicHello, name);Thread.sleep(Long.MAX_VALUE);}
}

在这个示例中,我们实现了 DynamicMBean 接口,并重写了它的方法来提供 MBean 的属性和操作。动态 MBean 允许在运行时确定其属性和操作,因此我们需要在实现中动态地处理属性的获取、设置和操作的调用。

getMBeanInfo() 方法中,我们定义了 MBean 的元数据信息,包括属性、操作和描述信息。在 main() 方法中,我们创建了 DynamicHello 的实例,并将其注册到 MBeanServer 中。然后通过 Thread.sleep(Long.MAX_VALUE) 使程序保持运行,以便通过 JConsole 或其他 JMX 客户端来操作该动态 MBean。

2.3.Model Bean

Model MBean 的核心是实现 ModelMBean 接口,该接口继承自 DynamicMBean 接口,因此具有动态 MBean 的所有特性,同时还添加了一些额外的方法用于管理描述符。以下是 ModelMBean 接口的一些重要方法:

  • setModelMBeanInfo(ModelMBeanInfo mbi):设置 Model MBean 的元数据信息,包括属性、操作和描述符。

  • setManagedResource(Object mr, String mr_type):指定 Model MBean 管理的资源对象及其类型。

  • sendNotification(Notification ntfyObj):发送通知给注册的监听器。

  • setAttribute(Attribute attribute)getAttribute(String attribute):获取和设置属性,这些属性可以通过描述符来配置。

代码示例:

import javax.management.*;
import javax.management.modelmbean.*;
import java.lang.management.ManagementFactory;
​
public class SimpleModelMBeanExample {public static void main(String[] args) throws Exception {// 创建 ModelMBeanInfo 对象ModelMBeanInfo modelMBeanInfo = createModelMBeanInfo();
​// 创建 ModelMBean 实例ModelMBean modelMBean = createModelMBean(modelMBeanInfo);
​// 创建 MBeanServerMBeanServer mbs = ManagementFactory.getPlatformMBeanServer();
​// 创建 ObjectName,用于标识 MBeanObjectName name = new ObjectName("com.example:type=SimpleModelMBean");
​// 注册 ModelMBean 到 MBeanServermbs.registerMBean(modelMBean, name);
​// 等待一段时间,使得可以通过 JConsole 或者其他 JMX 客户端来操作 MBeanThread.sleep(Long.MAX_VALUE);}
​private static ModelMBeanInfo createModelMBeanInfo() throws Exception {// 创建属性描述符Descriptor nameDesc = new DescriptorSupport();nameDesc.setField("name", "Name");nameDesc.setField("descriptorType", "attribute");
​// 创建操作描述符Descriptor sayHelloDesc = new DescriptorSupport();sayHelloDesc.setField("name", "sayHello");sayHelloDesc.setField("descriptorType", "operation");
​// 创建属性列表MBeanAttributeInfo[] attributes = new MBeanAttributeInfo[]{new MBeanAttributeInfo("Name", "java.lang.String", "Name of the person", true, true, false, nameDesc)};
​// 创建操作列表MBeanOperationInfo[] operations = new MBeanOperationInfo[]{new MBeanOperationInfo("sayHello", "Say hello to the person", null, "java.lang.String", MBeanOperationInfo.ACTION, sayHelloDesc)};
​// 创建 ModelMBeanInfoModelMBeanInfo modelMBeanInfo = new ModelMBeanInfoSupport(SimpleModelMBeanExample.class.getName(),"Simple Model MBean Example",attributes,null,operations,null);
​return modelMBeanInfo;}
​private static ModelMBean createModelMBean(ModelMBeanInfo modelMBeanInfo) throws Exception {// 创建 ModelMBean 实例RequiredModelMBean modelMBean = new RequiredModelMBean();modelMBean.setModelMBeanInfo(modelMBeanInfo);
​// 创建 MBean 实例ObjectName name = new ObjectName("com.example:type=SimpleModelMBean");SimpleModelMBean example = new SimpleModelMBean();
​// 设置 MBean 实例modelMBean.setManagedResource(example, "ObjectReference");
​return modelMBean;}
}

在这个示例中,我们首先创建了一个 ModelMBeanInfo 对象,该对象描述了 Model MBean 的元数据信息,包括属性和操作。然后,我们创建了一个 ModelMBean 实例,并将 ModelMBeanInfo 设置到该实例中。接着,我们创建了一个 ObjectName 对象,用于标识 MBean,然后将 ModelMBean 实例注册到了 MBeanServer 中。

同时,我们还定义了一个 SimpleModelMBean 类,该类实现了 Model MBean 管理的资源对象。在这个类中,我们可以定义属性和操作的具体实现逻辑。

2.4.Dynamic MBean和Model Bean的区别

其实Model Bean本来就是继承自Dynamic MBean的,所以后者有的特点,前者也有,其二者本质的区别在于使用场景上:

  1. 动态性质

    • Dynamic MBean:Dynamic MBean 的设计理念是允许在运行时动态地确定其属性和操作。它适用于那些在编译时无法确定其管理接口的场景,例如,那些需要根据外部配置文件或运行时状态来确定其管理接口的应用程序。

    • Model MBean:Model MBean 的设计理念是通过描述符来描述其管理接口,使得可以在运行时动态地加载和配置 MBean 的属性和操作。它适用于需要动态加载和配置管理接口的场景,例如,根据不同的环境或条件来动态地调整管理接口。

  2. 自描述性

    • Dynamic MBean:Dynamic MBean 不提供自描述性,开发人员需要手动实现属性和操作的获取和设置逻辑。它主要关注于动态性和灵活性,适用于那些需要在运行时动态确定管理接口的场景。

    • Model MBean:Model MBean 提供了详细的元数据信息,包括属性和操作的描述信息,使得管理客户端可以动态地了解 MBean 的结构和功能。它适用于那些需要自描述性和灵活性的场景,例如,需要通过管理客户端来动态地了解和操作 MBean 的情况。

  3. 适应性

    • Dynamic MBean:Dynamic MBean 具有较高的适应性,可以根据外部条件或运行时状态动态地调整其管理接口。它适用于那些需要根据外部条件或运行时状态来动态调整管理接口的场景。

    • Model MBean:Model MBean 具有较高的适应性,可以根据不同的环境或条件动态地调整管理接口。它适用于那些需要根据不同的部署环境或运行时条件来动态地配置 MBean 的属性和操作的场景。

2.5.MXBean

在 Java 中,一切都被视为对象,而 JMX 则提供了一种标准化的方式来监控和管理 Java 程序的运行时状态、性能指标以及运行环境。而MXBean其实就是专门用于管理和监控这其中一些标准化类型的对象的。什么叫标准化的类型对象?其实就是一些定量的指标,例如内存使用情况、线程信息、操作系统属性等。这些MBean是拿来即用的。

Java 平台中有一些预定义的 MXBean,例如 MemoryMXBeanThreadMXBeanOperatingSystemMXBean 等,用于管理和监控 JVM 的内存、线程、操作系统属性等。

代码示例:

import java.lang.management.ManagementFactory;
import java.lang.management.MemoryMXBean;
import java.lang.management.MemoryUsage;
​
public class MXBeanExample {public static void main(String[] args) {// 获取 MemoryMXBean 实例MemoryMXBean memoryMXBean = ManagementFactory.getMemoryMXBean();
​// 获取堆内存使用情况MemoryUsage heapMemoryUsage = memoryMXBean.getHeapMemoryUsage();System.out.println("Heap Memory Usage:");System.out.println("    Init: " + heapMemoryUsage.getInit() / (1024 * 1024) + " MB");System.out.println("    Used: " + heapMemoryUsage.getUsed() / (1024 * 1024) + " MB");System.out.println("    Committed: " + heapMemoryUsage.getCommitted() / (1024 * 1024) + " MB");System.out.println("    Max: " + heapMemoryUsage.getMax() / (1024 * 1024) + " MB");
​// 获取非堆内存使用情况MemoryUsage nonHeapMemoryUsage = memoryMXBean.getNonHeapMemoryUsage();System.out.println("Non-Heap Memory Usage:");System.out.println("    Init: " + nonHeapMemoryUsage.getInit() / (1024 * 1024) + " MB");System.out.println("    Used: " + nonHeapMemoryUsage.getUsed() / (1024 * 1024) + " MB");System.out.println("    Committed: " + nonHeapMemoryUsage.getCommitted() / (1024 * 1024) + " MB");System.out.println("    Max: " + nonHeapMemoryUsage.getMax() / (1024 * 1024) + " MB");}
}

可以自定义MXBean吗?当然可以。

自定义 MXBean 主要包括两个步骤:

  1. 定义接口:首先,需要定义一个 Java 接口来描述 MXBean 的管理接口。这个接口中定义了一组属性和操作,用于描述被管理的对象的状态和行为。

  2. 实现类:然后,需要编写一个实现类来实现定义的接口。在实现类中,需要提供属性和操作的具体实现逻辑,以及可能需要的管理逻辑。

代码示例:

// 定义 MXBean 接口
public interface MyMXBean {// 属性:名称String getName();void setName(String name);// 操作:打印消息void printMessage(String message);
}
​
// 实现类
public class MyMXBeanImpl implements MyMXBean {private String name;// 实现属性的 getter 和 setter 方法@Overridepublic String getName() {return name;}@Overridepublic void setName(String name) {this.name = name;}// 实现操作的方法@Overridepublic void printMessage(String message) {System.out.println(message);}
}
​
// 在管理应用程序中注册和使用自定义的 MXBean
public class MyMXBeanRegistration {public static void main(String[] args) throws Exception {// 创建 MXBean 实例MyMXBean mxBean = new MyMXBeanImpl();// 注册 MXBean 到 MBeanServerMBeanServer mbs = ManagementFactory.getPlatformMBeanServer();ObjectName name = new ObjectName("com.example:type=MyMXBean");mbs.registerMBean(mxBean, name);// 等待一段时间,使得可以通过 JConsole 或者其他 JMX 客户端来操作 MXBeanThread.sleep(Long.MAX_VALUE);}
}

在这个示例中,我们定义了一个名为 MyMXBean 的接口,包含了一个属性 name 和一个操作 printMessage。然后,我们编写了一个实现类 MyMXBeanImpl,实现了这个接口,并提供了属性和操作的具体实现逻辑。最后,我们在管理应用程序中注册和使用了这个自定义的 MXBean。

2.6.Open Bean

与传统的 Standard MBean 和 Dynamic MBean 不同,Open MBean 不需要预先定义接口,而是使用开放式的数据类型和操作来描述 MBean 的管理接口,使得可以更灵活地适应各种场景和需求。

代码示例:

import javax.management.*;
import javax.management.openmbean.*;
​
public class OpenMBeanExample {public static void main(String[] args) throws Exception {// 创建 OpenTypeOpenType<String> stringType = SimpleType.STRING;OpenType<Integer> integerType = SimpleType.INTEGER;
​// 创建 OpenMBeanAttributeInfoOpenMBeanAttributeInfoSupport nameAttrInfo = new OpenMBeanAttributeInfoSupport("Name","Name of the person",stringType,true,false,false);OpenMBeanAttributeInfoSupport ageAttrInfo = new OpenMBeanAttributeInfoSupport("Age","Age of the person",integerType,true,false,false);
​// 创建 OpenMBeanOperationInfoOpenMBeanParameterInfo[] params = new OpenMBeanParameterInfo[]{};OpenMBeanOperationInfoSupport sayHelloOpInfo = new OpenMBeanOperationInfoSupport("sayHello","Say hello to the person",params,stringType,OpenMBeanOperationInfo.ACTION);
​// 创建 OpenMBeanInfoOpenMBeanInfoSupport mbeanInfo = new OpenMBeanInfoSupport(SimpleOpenMBean.class.getName(),"Simple Open MBean Example",new OpenMBeanAttributeInfo[]{nameAttrInfo, ageAttrInfo},null,new OpenMBeanOperationInfo[]{sayHelloOpInfo},null);
​// 创建 OpenMBean 实例SimpleOpenMBean mbean = new SimpleOpenMBean("John", 30);
​// 创建 MBeanServerMBeanServer mbs = ManagementFactory.getPlatformMBeanServer();
​// 创建 ObjectName,用于标识 MBeanObjectName name = new ObjectName("com.example:type=SimpleOpenMBean");
​// 注册 OpenMBean 到 MBeanServermbs.registerMBean(mbean, name);
​// 等待一段时间,使得可以通过 JConsole 或者其他 JMX 客户端来操作 MBeanThread.sleep(Long.MAX_VALUE);}
}
​
// 实现 OpenMBean
public class SimpleOpenMBean implements OpenMBean {private String name;private int age;
​public SimpleOpenMBean(String name, int age) {this.name = name;this.age = age;}
​@Overridepublic Object getAttribute(String attribute) throws AttributeNotFoundException, MBeanException, ReflectionException {if (attribute.equals("Name")) {return name;} else if (attribute.equals("Age")) {return age;} else {throw new AttributeNotFoundException("Attribute " + attribute + " not found");}}
​@Overridepublic void setAttribute(Attribute attribute) throws AttributeNotFoundException, InvalidAttributeValueException, MBeanException, ReflectionException {throw new UnsupportedOperationException("setAttribute not supported");}
​@Overridepublic Object invoke(String actionName, Object[] params, String[] signature) throws MBeanException, ReflectionException {if (actionName.equals("sayHello")) {return "Hello, " + name + "!";} else {throw new UnsupportedOperationException("Operation " + actionName + " not supported");}}
​@Overridepublic MBeanInfo getMBeanInfo() {// 实现 OpenMBean 接口的 getMBeanInfo 方法,返回 MBean 的元数据信息return null;}
}

3.控制台

JDK自带了一个JMX的控制台JConsole,启动后选择进程,就能看到相应程序中的MBean。

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

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

【Wio Terminal教程】使用LCD屏幕(4)

使用LCD屏幕&#xff08;4&#xff09; 一、TFT LCD的API例子1、实用图形2、数据显示3、字体4、作为背景显示 二、如何在Wio Terminal上使用LVGL图形库1、安装Seeed_Arduino_LvGL2、示例1. Bench Mark2. Stress Test3.资源 一、TFT LCD的API例子 本节为TFT LCD库的例子提供了一…

ERROR: Could not build wheels for roslz4

Python bugs 最近在安装python的rosbag包时出现了诸多问题&#xff0c;特别记录下。 python版本&#xff1a;3.11 系统版本&#xff1a;Windows10 x86_64 使用conda虚拟环境进行包管理。 运行命令 pip3 install roslz4 --extra-index-url https://rospypi.github.io/simple…

【数据结构与算法】【小白也能学的数据结构与算法】递归 分治 迭代 动态规划 无从下手?一文通!!!

&#x1f389;&#x1f389;欢迎光临&#x1f389;&#x1f389; &#x1f3c5;我是苏泽&#xff0c;一位对技术充满热情的探索者和分享者。&#x1f680;&#x1f680; &#x1f31f;特别推荐给大家我的最新专栏《数据结构与算法&#xff1a;初学者入门指南》&#x1f4d8;&am…

|Python新手小白低级教程之项目篇——turtle库|第一章:turtle库基础(1)

项目篇—文章目录 一、预告二、turtle基础1.导入2.画图代码&#xff08;1&#xff09;turtle.forward(长度)练习1.1 画线段 &#xff08;2&#xff09;turtle.left()和turtle.right()操作符练习2.1 画出边长为100正方形练习2.2 画出边长为100的三角形 &#xff08;3&#xff09…

腾讯云4核8G12M轻量应用服务器性能够用吗?支持多少人?

腾讯云轻量4核8G12M轻量应用服务器支持多少人同时在线&#xff1f;通用型-4核8G-180G-2000G&#xff0c;2000GB月流量&#xff0c;系统盘为180GB SSD盘&#xff0c;12M公网带宽&#xff0c;下载速度峰值为1536KB/s&#xff0c;即1.5M/秒&#xff0c;假设网站内页平均大小为60KB…

Web前端-移动web开发_rem布局

文章目录 移动web开发之rem布局1.0 rem基础1.1 rem单位(重点)1.2 em单位(了解)1.3 媒体查询什么是媒体查询媒体查询语法规范 1.4 less 基础维护css弊端Less 介绍Less安装Less 使用之变量使用node编译less的指令Less 编译 vocode Less 插件Less 嵌套Less 运算Less中的Mixin混入L…

车载测试Vector工具——常见问题汇总

车载测试Vector工具——常见问题汇总 我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师(Wechat:gongkenan2013)。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 屏蔽力是信息过载时代一个人的特殊竞争力,任何 消耗你的人和事,多看一眼都是你的…

C++重新入门-循环

目录 1.循环类型 while循环&#xff1a; for循环 基于范围的for循环(C11) do...while 循环 2.循环控制语句 3.无限循环 有的时候&#xff0c;可能需要多次执行同一块代码。一般情况下&#xff0c;语句是顺序执行的&#xff1a;函数中的第一个语句先执行&#xff0c;接着…

《零基础实践深度学习》基于线性回归实现波士顿房价预测任务1.3.3

1.3.3 基于线性回归实现波士顿房价预测任务 深度学习不仅实现了模型的端到端学习&#xff0c;还推动了人工智能进入工业大生产阶段&#xff0c;产生了标准化、自动化和模块化的通用框架。不同场景的深度学习模型具备一定的通用性&#xff0c;五个步骤即可完成模型的构建和训练&…

vs2019 - signtool签名和验签的手工操作

文章目录 vs2019 - signtool签名和验签的手工操作概述笔记导入根证书时间戳服务器的选择code sign - 签名文件在代码签名(code sign)前后的区别签名后, 查看属性, 是正常的.用signtool命令行进行验签移除签名END vs2019 - signtool签名和验签的手工操作 概述 signtool是进行c…

linux优化空间完全卸载mysql——centos7.9

文章目录 ⭐前言⭐linux命令使用&#x1f496; 基础命令&#x1f496; 内存优化&#x1f496; 完全删除mysql ⭐结束 ⭐前言 大家好&#xff0c;我是yma16&#xff0c;linux优化空间&完全卸载mysql——centos7.9。 linux内存分配 在Linux中&#xff0c;内存分配是通过几种…

Linux---网络基础

计算机中的常见概念 协议&#xff08;Protocol&#xff09;&#xff1a; 协议是计算机网络中用于通信的规则和约定的集合。它规定了数据传输的格式、序列、错误检测和纠正方法等。常见的网络协议包括TCP/IP、HTTP、FTP等。 IP地址&#xff08;IP Address&#xff09;&#xf…

Python3集合 set 心得

Python3集合 set 心得 本文是个人在学习中的一些心得&#xff0c;不能保证文中的内容准确&#xff0c;欢迎大家批评指正。 集合 set 是Python中的一种数据容器&#xff0c;集合有2种定义方式。 1、花括号 {} &#xff0c;花括号内加入集合的元素&#xff0c;元素之间用逗号 …

通过Dynamo删除Revit中族参数探究

起因是这样的&#xff0c;有位同事想在项目中直接删除revit族的参数&#xff0c;而不打开族&#xff0c;避免因为重载族&#xff0c;造成一些管件连接断开&#xff0c;于是就有了下面这些研究&#xff0c;当然是因为我没办法实现这个想法&#xff0c;所以这次可以分享下研究的过…

linux系统定时任务管理

crontab使用 一、crontab简介 crontab 这个指令所设置的工作将会循环的一直进行下去&#xff01;可循环的时间为分钟、小时、每周、每月或每年等。crontab 除了可以使用指令执行外&#xff0c;亦可编辑 /etc/crontab 来支持。 至于让 crontab 可以生效的服务则是 crond 这个服…

39 MDK 的编译过程及文件类型全解

文章目录 39.1 编译过程39.1.1 编译过程简介 39.2 程序的组成、存储与运行39.3 编译工具链39.3.1 MDK编译文件39.3.2 设置环境变量39.3.3 运行39.3.4 armcc、armasm 及 armlink39.3.4.1 armcc39.3.4.2 查询具体的MDK编译选项的具体信息39.3.4.3 armasm39.3.4.4 armlink39.3.4.5…

“OLED屏幕,色彩绚丽,画面清晰,让每一帧都生动无比。“#IIC协议【上】

"OLED屏幕&#xff0c;色彩绚丽&#xff0c;画面清晰&#xff0c;让每一帧都生动无比。"#IIC协议【上】 前言预备知识1.IIC协议初识1.1 IIC协议概述1.2 IIC协议的特点1.3 IIC协议的构成 2. IIC起始信号和终止信号2.1起始型号和终止信号时序图分析2.2起始信号函数和终…

JAVA学习笔记9

1.Java API 文档 1.java类的组织形式 2.字符类型(char) 1.基本介绍 ​ *字符类型可以表示单个字符&#xff0c;字符类型是char&#xff0c;char是两个字节&#xff08;可以存放汉字&#xff09;&#xff0c;多个字符我们用字符串String ​ eg:char c1 ‘a’; ​ char c2…

【Android】使用Android Studio运行Hello World项目

文章目录 1. JDK的安装与配置2. Android Studio的安装3. 运行Hello World项目3.1 新建项目3.2 修改项目配置3.2.1 修改UI界面3.2.2 配置 Android SDK 3.3 添加并运行虚拟设备3.4 运行项目 1. JDK的安装与配置 想要使用Android Studio&#xff0c;必须先配置Java环境&#xff0…

vue3初识

目录 一、前言二、主观感受三、vue3初探 原文以及该系列教程文章后续可点击这里查看&#xff1a;vue初识 一、前言 Vue.js是一款流行的前端框架&#xff0c;最初由尤雨溪&#xff08;Evan You&#xff09;于2014年创建&#xff0c;非常的年轻。官网为vue3&#xff0c; 但要注…