HarmonyOS 鸿蒙Next json解析嵌套对象不走set方法 plainToClass仅对最外层类new有效
HarmonyOS 鸿蒙Next json解析嵌套对象不走set方法 plainToClass仅对最外层类new有效
import { plainToClass, ClassConstructor } from 'class-transformer'
import "reflect-metadata"
import { BusinessError } from '@kit.BasicServicesKit';
@Observed
class ProductModel{
name:string = ''
private _count: number = 0;
public set count(value: number) {
this._count = value;
this.flagCount = value;//最外层这里是走set方法的
}
public get count(): number {
return this._count;
}
flag: number = 0;
details:ProductDetailModel[] = []
flagCount: number = 0;
}
@Observed
class ProductDetailModel{
detailName:string = ''
private _detailCount: number = 0;
public set detailCount(value: number) {
this._detailCount = value;
this.flagCount = value;//嵌套的这里就不走set方法了
}
public get detailCount(): number {
return this._detailCount;
}
flagCount:number = 0
}
function jsonToArray<T>(cls: ClassConstructor<T>, jsonStr: string): Array<T> {
try {
return plainToClass(cls, JSON.parse(jsonStr), { enableImplicitConversion: false, exposeDefaultValues: true }) as Array<T>
} catch (err) {
let error = err as BusinessError;
return []
}
}
@Entry
@Component
struct ProductListView {
jsonString:string = '[{"name":"商品1", "count":"1", "details":[{"detailName":"子商品1", "detailCount":"1"}]},{"name":"商品2", "count":"1", "details":[{"detailName":"子商品2", "detailCount":"1"}]},{"name":"商品3", "count":"1", "details":[{"detailName":"子商品3", "detailCount":"1"}] }]'
@State productList: ProductModel[] = [];
aboutToAppear(): void {
// this.productList = JSON.parse(this.jsonString);
this.productList = jsonToArray(ProductModel, this.jsonString);
}
build() {
Column({space:10}){
ForEach(this.productList,(item:ProductModel)=>{
ProductView({item:item})
})
}.padding(20)
}
}
@Component
struct ProductView {
@ObjectLink item: ProductModel;
build() {
Column(){
Row(){
Text(this.item.name)
Text(`${this.item.count}`)
Text(`${this.item.flagCount}`)//最外层这里走了set方法,所以这里的值变了
Button('数量-1').onClick(()=>{
this.item.count--
})
Button('数量+1').onClick(()=>{
this.item.count++
})
}.width('100%').justifyContent(FlexAlign.SpaceBetween)
Column({space:10}){
ForEach(this.item.details,(detailItem:ProductDetailModel)=>{
ProductDetailsView({parItem:this.item,item:detailItem})
})
}.padding(20)
}
}
}
@Component
struct ProductDetailsView {
@ObjectLink parItem:ProductModel;
@ObjectLink item: ProductDetailModel;
build() {
Column(){
Row(){
Text(this.item.detailName)
Text(`${this.item.detailCount}`)
Text(`${this.item.flagCount}`)//嵌套这里没走set方法,所以这里的值没有发生改变,但是detailCount的值是变了的
Button('数量-1').onClick(()=>{
this.parItem.flag --;
this.item.detailCount--
})
Button('数量+1').onClick(()=>{
this.parItem.flag ++;
this.item.detailCount++
})
}.width('100%').justifyContent(FlexAlign.SpaceBetween)
}
}
}
2 回复
aboutToAppear(): void {
// this.productList = JSON.parse(this.jsonString);
this.productList = jsonToArray(ProductModel, this.jsonString);
for (let i = 0; i < this.productList.length; i++) {
let temp: ProductDetailModel[] = this.productList[i].details
this.productDetailsList =
this.productDetailsList.concat(plainToClass(ProductDetailModel, temp) as ProductDetailModel[]);
}
console.log("productDetailsList:" + JSON.stringify(this.productDetailsList))
}
在HarmonyOS鸿蒙系统中,使用plainToClass
方法进行JSON解析时,如果遇到嵌套对象不走set方法的问题,这通常是由于类转换器(class transformer)没有正确应用到嵌套对象上。plainToClass
函数默认只会对最外层的类进行实例化,而不会自动递归地处理嵌套对象。
要解决这个问题,可以确保以下几点:
-
嵌套类也需要使用装饰器或注解:确保所有的嵌套类都使用了相应的装饰器(如
@Type
)来指明其类型,这样类转换器才能识别。 -
自定义转换器:如果默认转换器不支持你的需求,可以编写自定义转换器来处理嵌套对象的解析,并在调用
plainToClass
时传入。 -
使用
class-transformer
库的最新版本:确保你使用的class-transformer
库是最新版本,因为新版本可能修复了旧版本中的bug。 -
检查JSON数据结构:确保JSON数据结构正确无误,嵌套对象的字段名与类中定义的字段名完全一致。
如果以上方法都无法解决问题,可能是因为特定的库版本或框架限制。此时,建议直接查阅HarmonyOS鸿蒙系统的开发者文档或联系相关技术支持。如果问题依旧没法解决请联系官网客服,官网地址是:https://www.itying.com/category-93-b0.html