4.1 JS中原始值和引用值的区别

本文详细阐述了JavaScript中原始值(如undefined、null、bool、num、str、sym)与引用值(对象)的区别,涉及动态属性、值复制、参数传递以及类型检测。实例演示了两者如何操作以及typeof与instanceof在判断中的应用。

JS中原始值和引用值的区别

  ECMAScript 变量可能包含两种不同类型的值:原始值引用值

  • 原始值:指的是简单的数据,包括 UndefinedNullBooleanNumberStringSymbol六种基本数据类型
  • 引用值:指的是保存在内存中的对象。

  保存原始值的变量是按值(by value)访问的,因为我们操作的就是存储在变量中的实际值。保存引用值的变量是按引用(by reference)访问的。操作的是对该对象的引用(reference)而非实际的对象本身。

1 动态的属性

  对于引用类型的值,可以为其随时添加属性和方法,也可以改变和删除其属性和方法。例如:

let person = new Object();
person.name = "andy";
alert(person.name);     // andy

  不能给基本类型的值添加属性,虽然不会导致任何错误,但是没有作用。例如:

let a = "andy";
a.age = 27;
alert(a.age);       //undefined

2 复制值

  从一个变量向另一个变量复制原始值和引用值时,方式也是不同的。

  • 复制基本数据类型值

  如果复制的是基本类型的值,会在变量对象上创建一个新值,然后把该值复制到新变量分配的位置上。例子:

var num1 = 5;
var num2 = num1;

  如上,两个变量的值都是5,但互不干扰,相互独立。过程如下图。

复制基本数据类型值的过程
  • 复制引用数据类型值

  当从一个变量向另一个变量复制引用数据类型时,同样也会将存储在变量对象中的值复制一份放到新变量分配的空间中。不同的是这个值是一个指针,指向存储在堆内存当中的一个对象。结果是两个变量都引用同一个对象。如下例子:

var obj1 = new Object();
var obj2 = obj1;
obj1.name = "andy";
alert(obj2.name);   // andy
复制引用数据类型值的过程

3 传递参数

  ECMAScript 中所有函数的参数都是按值传递的。基本类型值的传递如同基本类型变量的复制一样,而引用类型值的传递,则如同引用类型变量的复制一样。

  • 向参数传递基本类型值

  被传递的值会被复制给一个局部变量(即命名参数,就是arguments 中的一个元素)。如下例子:

function addTen(num){
    num += 10;
    return num;
}
var count = 20;
var result = addTen(count);
alert(count);       // 20,没有变化
alert(result);      // 30

  在上述函数中,参数 num 实际上是函数的局部变量,变量 count 作为参数被传递给函数,数值 20 被复制给参数 num 进行函数内部的计算,参数 num 和 count 是互不相识的。不会影响彼此

  • 向参数传递引用类型值

  在向参数传递引用类型的值时,会把这个值在内存中的地址复制给一个局部变量,因此这个局部变量的变化会反映在函数的外部。例子:

function setName(obj){
    obj.name = "andy";
}
var person = new Object();
setName(person);
alert(person.name);     // andy

  上述代码主要步骤执行如下图所示:

代码执行过程

4 确定类型

  检测一个变量是不是基本数据类型可以用 typeof 操作符,它可以确定一个变量是字符串、数值、布尔值、还是 undefined。如果变量的值是一个对象或 null,则 typeof 操作符会返回 Object。如下所示:

var s = "andy";
var b = true;
var i = 2;
var u;
var n = null;
var o = new Object();
function f(){
}
alert(typeof s);    // string
alert(typeof b);    // boolean
alert(typeof i);    // number
alert(typeof u);    // undefined
alert(typeof n);    // object
alert(typeof o);    // object
alert(typeof f);    // function

  typeof 在检测函数时会返回 function。typeof 在检测基本数据类型时非常得力,但在检测引用数据类型的值时,它只能告诉你这是一个 object 。不能告诉你它具体是什么类型的对象,为此,ECMAScript 提供了 instanceof 操作符,其语法如下所示:

result= variable instanceof constructor

  如果变量是给定引用类型的实例(根据它的原型链来识别)。那么 instanceof 会返回 true。例子如下:

var person = new Object();
// 变量 person 是 Object 吗?
alert(person instanceof Object);    // true

  所有引用类型的值都是 Object 类型的实例。如果使用 instanceof 操作符检测基本数据类型的值,始终会返回 false ,因为基本类型不是对象。

<think> 我们需要在ECharts中配置坐标轴刻度标签(axisLabel)的显示规则。对于y轴,我们可以通过设置formatter函数来过滤掉0最后一个值(最大值)的显示。对于x轴,我们希望始终显示第一个最后一个值,而中间的值根据需求显示(比如只显示整点)。 但是注意:y轴的起始值0最后一个值(最大值)不显示,我们可以通过axisLabel的formatter函数判断,如果是0或者最大值就返回空字符串。然而,这种方法需要知道最大值,而最大值可能随着数据变化而变化。另一种方法是使用intervalinside参数,但更灵活的方式是使用formatter函数。 然而,ECharts并没有直接提供隐藏特定刻度标签的方法,所以我们需要在formatter中根据条件返回字符串。 对于x轴,我们同样可以使用formatter函数,确保第一个最后一个值始终显示,中间的值我们可以按照之前的策略(比如只显示整点)。 下面是一个示例配置(基于原生ECharts,但原理适用于ucharts等封装库): </think> ### 在 ECharts 中设置坐标轴起始值结束值显示 下面是实现方案,通过配置 `xAxis` `yAxis` 的 `axisLabel` 属性控制显示: ```javascript option = { xAxis: { type: 'category', data: ['00:00', '01:00', '02:00', '03:00', '04:00', '05:00', '06:00'], axisLabel: { // 始终显示首尾标签 showMinLabel: true, showMaxLabel: true, // 自定义标签显示函数 formatter: function(value, index) { // 始终显示第一个最后一个值 if (index === 0 || index === this.axis.getExtent()[1]) { return value; } // 其他标签根据条件显示(这里示例只显示整点) const hour = parseInt(value.split(':')[0]); return hour % 2 === 0 ? value : ''; } } }, yAxis: { type: 'value', axisLabel: { // 自定义显示函数 formatter: function(value, index) { // 获取坐标轴最大值 const max = this.axis.scale.getExtent()[1]; // 不显示0值最大值 if (value === 0 || value === max) { return ''; } return value; } }, // 确保坐标轴范围包含0值但不显示 min: 0 }, series: [{ type: 'line', data: [120, 200, 150, 80, 70, 110, 130] }] }; ``` ### 关键配置解释: 1. **X 轴配置**: ```javascript axisLabel: { showMinLabel: true, // 强制显示第一个标签 showMaxLabel: true, // 强制显示最后一个标签 formatter: function(value, index) { // index === 0 表示第一个值 // this.axis.getExtent()[1] 获取最后一个索引 if (index === 0 || index === this.axis.getExtent()[1]) { return value; // 显示首尾标签 } // 其他自定义显示逻辑 return hour % 2 === 0 ? value : ''; } } ``` 2. **Y 轴配置**: ```javascript axisLabel: { formatter: function(value, index) { const max = this.axis.scale.getExtent()[1]; // 隐藏0值最大值 if (value === 0 || value === max) { return ''; } return value; } }, min: 0 // 确保坐标轴从0开始 ``` 3. **实现原理**: - 使用 `axisLabel.formatter` 自定义标签显示逻辑 - 通过 `axis.getExtent()` 获取坐标轴索引范围 - `this.axis.scale.getExtent()` 获取数值范围 - 返回空字符串 `''` 表示隐藏该标签 ### 完整示例(UniApp + uCharts): ```html <template> <view> <qiun-data-charts type="line" :opts="opts" :chartData="chartData"/> </view> </template> <script> export default { data() { return { opts: { xAxis: { disableGrid: true, axisLabel: { showMinLabel: true, showMaxLabel: true, formatter: (val, idx, axis) => { if (idx === 0 || idx === axis.data.length - 1) { return val; } // 只显示整点(假设时间格式为HH:mm) return val.endsWith(':00') ? val : ''; } } }, yAxis: { axisLabel: { formatter: (val, idx, axis) => { const max = Math.max(...axis.scale.max, ...this.chartData.series[0].data); return val === 0 || val === max ? '' : val; } } } }, chartData: { categories: ['00:00', '01:00', '02:00', '03:00', '04:00', '05:00', '06:00'], series: [{ name: '数据', data: [23, 45, 37, 28, 40, 32, 38] }] } }; } }; </script> ``` ### 注意事项: 1. **uCharts适配**: - uCharts 的 formatter 函数参数略有不同(val, idx, axis) - 使用 `axis.data.length - 1` 获取最后一个索引 2. **动态数据**: - 当数据更新时,需要重新计算最大值 - 可以使用计算属性或 watch 监听数据变化 3. **替代方案**: ```javascript // 使用interval控制显示间隔(不够灵活) axisLabel: { interval: (index, value) => { return index === 0 || index === data.length - 1; } } ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ItDaChuang

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值