IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 游戏开发 -> 【[中文直播] 第21期 | UE4数据驱动开发 | Epic 大钊】课程笔记 -> 正文阅读

[游戏开发]【[中文直播] 第21期 | UE4数据驱动开发 | Epic 大钊】课程笔记

视频链接

点击我进入B站观看视频

课程资源在简介上也写了,可以进入下载,资源很小不用借助其他的软件,直接百度网盘下载就可以了。
最近百度网盘青春版也出来了,不过不推荐保存这个进去下载,有次数限制,加这个进去就太亏咯!

关于无法编译成功的问题

将启用状态改成false。
在这里插入图片描述
DataDriven.Build.cs中添加PublicDependencyModuleNames.Add("DeveloperSettings");

然后就可以成功编译运行了!

在名为Watch代码的文件夹中有部分插件中的代码,可以借鉴参考一下。

为什么要使用数据驱动?

  • 使用数据驱动的话能将职责进行分离,不用经常在代码里面进行数据的修改,而是在外部的数据表中进行修改。
  • 减少了硬编码数据。
  • 减少了硬编码类, 可以储存类的链接进行软引用。
  • 自动化配置数据流程,比如说可以直接在游戏运行中实时修改数据,提高了调试效率。

大钊大佬将配置数据的方式分为了四个方面:初级、中级、高级、工具,我将从以上四个方面进行整理,让大家和自己以后要用的时候方便提取。

初级

BP CDO/Instance

这个就是直接在实例或者蓝图类中直接进行赋值的方式。

优点缺点
使用简单,能够满足简易的数据配置需求散乱到各个蓝图中,不方便统一处理

在这里插入图片描述

C++ ConstructHelpers

类似这种,直接将链接添加到其中:
ConstructorHelpers::FObjectFinder<USkeletalMesh> objFinder(TEXT("/Game/Mannequin/Character/Mesh/SK_Mannequin"));

(感觉很麻烦。。。)

这个存在的一个问题就是:如何去写他的路径。

  • 类名’/路径/包名.对象名:子对象名’
  • 类名其实会被忽略掉,所以不写也行
  • 类: /路径/包名.类名_C (包名往往和类名一致)
  • 对象:/路径/包名.对象名(包名往往和对象一致)
  • 路径:/Root/Path1/Path2/……
  • 路径Root:/Engine, /Game , /Module

右键对象目标文件,有个 Copy File Path ,然后就可以得到他的绝对路径,对路径进行修改即可。

Class Reference

在C++ 中可以这么用,<>里面表示的是一个类,所有这个类的子类都可以加载到这个变量中进行保存。
TSubclassOf<UBaseItemWdget> ItemWidgetClass;

在蓝图中就比较简单了,这个创建类型的时候点击这个紫色的就可以了。
在这里插入图片描述

中级

DataTable/CompositeDataTable

  • DataTable

先在蓝图/C++中创建一个结构体,然后新建:
DataTable
然后填表就可以了。
在这里插入图片描述
优点 :

  1. 容易编辑
  2. 可以导入导出CSV/JSON
  3. 可用FDataTableRowHandle引用一行
  4. 配置大量数据很方便

缺点:

  1. 不能包含UObjects
  2. 无父子层级关系
  3. 无序,用TMap<FName,T>保存
  4. 不可引用其他表格
  5. 依然强绑定资产的加载
  • CompositeDataTable

在这里插入图片描述

他可以堆叠覆盖多个DataTable。
在这里插入图片描述
从图中可以看出来,后来的数据表中的属性值会覆盖掉前面数据表中有相同属性的属性值,保留不同的。

DataAsset/PrimaryDataAsset

在这个位置可以新建一个DataAsset
在这里插入图片描述
优点:

  • 容易子类化

  • 可快速引用其他资产 UMyDataAsset*

  • 可用BulkEdit快速编辑多个DataAsset:在这里插入图片描述

  • 可容纳UObject实例

  • 灵活的数据配置方式

  • 方便配置树状组织数据,全局配置,数据对象集合, 每一个DataAsset都是一份数据对象

  • 继承自UPrimaryDataAsset可方便管理数据对象的加载释放

缺点:

  • 不好管理大量的配置对象

(关于具体如何使用,先不写,等到用到这个功能了再来补充这块的笔记。)

CurveTable/CompositeCurveTables

创建一个.csv文件,配置对应的数据,
格式:
在这里插入图片描述
然后导入到虚幻引擎中,
在这里插入图片描述
选择CurveTable即可。
这个也支持数据覆盖。

Custom WorldSettings

创建一个继承自AWorldSettings的类,
class AShopWorldSettings : public AWorldSettings

在其中添加上需要的属性值。

在这里插入图片描述
然后在项目设置中配置上这个属性,重启编译器,新建关卡,就可以在关卡中看到这个值啦!

在C++中可以通过这种方式来调用这个属性值:

AWorldSettings* settings= UGameplayStatics::GetGameMode(this)->GetWorld()->GetWorldSettings();
AShopWorldSettings* shopSettings=Cast<AShopWorldSettings>(settings);

优点:

  • 可对每个关卡进行单独的全局配置

缺点:

  • 数据依赖存在于关卡内部

高级

Settings(.ini)

创建一个类,像下面这个例子一样的写法写就行:

UCLASS(config = DataDrivenProjectSettings, defaultconfig)
class UDataDrivenProjectSettings : public UDeveloperSettings
{
	GENERATED_BODY()
public:
	/** 选择配置设置是在项目还是在编辑器中。 */
	virtual FName GetContainerName() const override { return TEXT("Project"); }
	/** 获取设置的类别,一些高级分组,例如,编辑器、引擎、游戏……等*/
	virtual FName GetCategoryName() const override { return TEXT("DataDriven"); }
	/** 设置部分的唯一名称使用类的FName。*/
	virtual FName GetSectionName() const override { return TEXT("DataDriven"); }
public:
	UDataDrivenProjectSettings();
	
	UFUNCTION(BlueprintPure, DisplayName = "DataDrivenProjectSettings")
		static UDataDrivenProjectSettings* Get() { return GetMutableDefault<UDataDrivenProjectSettings>(); }
public:
	UPROPERTY(Config, EditAnywhere, BlueprintReadWrite, Category = Shop)
		float PriceOff = 50.f;

	UPROPERTY(Config, EditAnywhere, BlueprintReadWrite, Category = Player)
		float PlayerStartLevel = 10.f;

};

对应的效果:
在这里插入图片描述
还可以给Actor类中加上类似这种属性:

  • .h
UCLASS(config = CustomConfig) // 定义一个配置名
class ACustomConfigActor :public AActor
{
	GENERATED_BODY()
public:
	ACustomConfigActor();
public:
	UPROPERTY(Config, EditAnywhere, BlueprintReadWrite, Category = MyCustom) // 参与配置,那么就使用 Config,让引擎对他进行序列化。
		float MaxLife = 100.f;

	UPROPERTY(transient, EditAnywhere, BlueprintReadWrite, Category = MyCustom) //不参与配置,使用 transient,引擎对他不进行序列化。
		float CurrentLife = 50.f;
public:
	virtual void BeginPlay()override;

	UFUNCTION(BlueprintCallable)
	void AddMaxLifeAndSave(float val);
};
  • .cpp
void ACustomConfigActor::BeginPlay()
{
	Super::BeginPlay();
	// 在初始化的时候加载配置
	LoadConfig();
}

void ACustomConfigActor::AddMaxLifeAndSave(float val)
{
	// 修改属性值,然后进行保存
	MaxLife+=val;
	SaveConfig();
}

然后再让蓝图继承他,并让他调用AddMaxLifeAndSave函数。
在这里插入图片描述
将其拖入到项目中,编译保存,你就可以在\Saved\Config\Windows目录下找到我们自定义的这个配置文件。
在这里插入图片描述

ConsoleVariables

这个我感觉对我目前这个阶段帮助最大!!
他可以在控制台读取设置的变量,大大提高了调试的效率!!

只要将我们要控制的变量改成这样的形式就可以对他进行修改:

static TAutoConsoleVariable<bool> CVarDDShopEnabled(
	TEXT("dd.ShopEnabled"),
	false,
	TEXT("The switch to enable shop or not."),
	ECVF_Default);


static TAutoConsoleVariable<类型名> 变量名(
	TEXT("控制台命令"),
	赋的初始值,
	TEXT("提示文本"),
	ECVF_Default);

获取变量值:CVarDDShopEnabled.GetValueOnGameThread()

Raw:txt/csv/json/xml

大佬给出了读取文件的使用方法:

FString ARawDataActor::ReadTxtFile(FString path)
{
	FString realPath = FPaths::ProjectDir() / path;
	FString content;
	FFileHelper::LoadFileToString(content, *realPath);
	
	return content;
}

bool ARawDataActor::LoadCsvAsTable(FString path, TArray<FShopItem>& Items)
{
	UDataTable* DataTable = NewObject<UDataTable>(GetTransientPackage(), FName(TEXT("TempDataTable")));
	DataTable->RowStruct = FShopItem::StaticStruct();

	FString realPath = FPaths::ProjectDir() / path;
	FString content;

	if (!FFileHelper::LoadFileToString(content, *realPath))
	{
		return false;
	}

	DataTable->CreateTableFromCSVString(content);	//CreateTableFromJSONString

	static const FString ContextString(TEXT("GENERAL"));
	DataTable->ForeachRow<FShopItem>(ContextString, [&Items](const FName& key, const FShopItem& value)
		{
			Items.Add(value);
		});

	return true;
}

bool ARawDataActor::LoadJsonAsTable(FString path, TArray<FShopItem>& Items)
{
	FString realPath = FPaths::ProjectDir() / path;
	FString content;
	FFileHelper::LoadFileToString(content, *realPath);

	TSharedPtr<FJsonObject> jsonRoot = MakeShareable(new FJsonObject());
	TSharedRef<FJsonStringReader> jsonReader= FJsonStringReader::Create(content);

	TArray<TSharedPtr<FJsonValue>> OutArray;

	if (FJsonSerializer::Deserialize<TCHAR>(jsonReader, OutArray))
	{
		for (TSharedPtr<FJsonValue> itemJson: OutArray)
		{
			const TSharedPtr<FJsonObject>& obj=itemJson->AsObject();
			FShopItem& newItem= Items.AddDefaulted_GetRef();
			newItem.Name=FText::FromString(obj->GetStringField(TEXT("Name")));
			newItem.Icon=LoadObject<UTexture2D>(nullptr,*obj->GetStringField(TEXT("Icon")));
			newItem.Price = obj->GetNumberField(TEXT("Price"));
		}
		return true;
	}
	return false;
}

优点:

  • 可快速编辑大量数据
  • 可比较容易源码管理比较

缺点:

  • 被排除在虚幻烘焙工具之外,需要手动添加
  • 不好做数据合法性验证
  • 没有UI做数值限制
  • 容易搞错格式
  • UE的Reference System无法识别,所以无法
  • 绑定是否可安全删除一个配置
  • 写对UE的资产引用路径很麻烦

Enhanced: sql/excel

  • excel

是DirectExcel插件的使用,我没有买这个插件,所以无法提供笔记,hhh~

  • sql

SQLiteStudio下载地址
觉得下载慢的话可以使用网易加速器,上GitHub特别快!

点击添加

在这里插入图片描述
在这里插入图片描述
创建一个表:
在这里插入图片描述
输入表的名字:
在这里插入图片描述
添加相应的字段:
在这里插入图片描述
然后点击:Add column
在这里插入图片描述
点击OK即可:
在这里插入图片描述
然后可以点击加号进行添加对应的信息:
在这里插入图片描述
这里可以输入对应的信息进行查询:
在这里插入图片描述
也可以在项目中直接进行查询:

bool ASqlDataActor::LoadSqlAsTable(FString path, TArray<FShopItem>& Items)
{
	FSQLiteDatabaseConnection db;
	if (!db.Open(*path, nullptr, nullptr))
	{
		return false;
	}

	FString query = FString::Printf(TEXT("SELECT Name,Icon,Price FROM ShopItems"));

	FDataBaseRecordSet* outRecords = nullptr;
	if (!db.Execute(*query, outRecords))
	{
		delete outRecords;
		return false;
	}

	int count = outRecords->GetRecordCount();
	if (count == 0)
	{
		delete outRecords;
		return false;
	}

	int result = -1;
	for (FDataBaseRecordSet::TIterator i(outRecords); i; ++i)
	{
		FShopItem& newItem = Items.AddDefaulted_GetRef();
		for (FDatabaseColumnInfo column : i->GetColumnNames())
		{
			if (column.ColumnName == TEXT("Name"))
			{
				newItem.Name = FText::FromString(i->GetString(*column.ColumnName));

			}
			else if (column.ColumnName == TEXT("Icon"))
			{
				newItem.Icon = LoadObject<UTexture2D>(nullptr, *i->GetString(*column.ColumnName));
			}
			else if (column.ColumnName == TEXT("Price"))
			{
				newItem.Price = i->GetFloat(*column.ColumnName);
			}

		}

	}
	delete outRecords;
	db.Close();
	
	return true;
}

注意,使用之前需要做以下配置
.uproject文件中添加

"Plugins": [
	{
		"Name": "SQLiteSupport",
		"Enabled": true
	}
]

.Build.cs文件中添加:

PublicDependencyModuleNames.AddRange(new string[] { "SQLiteSupport" });

工具

BulkEdit

这个讲了,点击移动到出现位置

EditorUtility

将场景中的信息提取到表格里面。

在这里插入图片描述
其中有一些函数是无法直接弄出来的,所以要在C++中添加一些东西,将函数暴露在蓝图界面中:

  • .h
UCLASS(Blueprintable, Category = "DataDriven")
class UMyDataTableFunctionLibrary : public UBlueprintFunctionLibrary
{
	GENERATED_BODY()
public:
#if WITH_EDITOR
	UFUNCTION(BlueprintCallable, Category = "Editor Scripting | DataTable", DisplayName = "AddRowToDataTable")
		static void AddVolumeToDataTable(UDataTable* DataTable, FName rowName,const FVolumeItem& row);

	UFUNCTION(BlueprintCallable, Category = "Editor Scripting | DataTable", DisplayName = "ClearDataTable")
		static void ClearDataTable(UDataTable* DataTable);

	UFUNCTION(BlueprintCallable, Category = "Editor Scripting | DataTable", DisplayName = "DataTable_RemoveRow")
		static void DataTable_RemoveRow(UDataTable* DataTable, FName rowName);
#endif
};
  • .cpp
void UMyDataTableFunctionLibrary::AddVolumeToDataTable(UDataTable* DataTable, FName rowName, const FVolumeItem& row)
{
	if (DataTable == nullptr)
	{
		return;
	}
	DataTable->AddRow(rowName, row);
	DataTable->GetOutermost()->MarkPackageDirty();
}

void UMyDataTableFunctionLibrary::ClearDataTable(UDataTable* DataTable)
{
	if (DataTable == nullptr)
	{
		return;
	}
	DataTable->EmptyTable();
	DataTable->GetOutermost()->MarkPackageDirty();
}

void UMyDataTableFunctionLibrary::DataTable_RemoveRow(UDataTable* DataTable, FName rowName)
{
	if (DataTable == nullptr)
	{
		return;
	}
	DataTable->RemoveRow(rowName);
	DataTable->GetOutermost()->MarkPackageDirty();
}

DataTable->GetOutermost()->MarkPackageDirty();
这个代码就让小星星出现,代表已经对他进行了修改。
在这里插入图片描述

  游戏开发 最新文章
6、英飞凌-AURIX-TC3XX: PWM实验之使用 GT
泛型自动装箱
CubeMax添加Rtthread操作系统 组件STM32F10
python多线程编程:如何优雅地关闭线程
数据类型隐式转换导致的阻塞
WebAPi实现多文件上传,并附带参数
from origin ‘null‘ has been blocked by
UE4 蓝图调用C++函数(附带项目工程)
Unity学习笔记(一)结构体的简单理解与应用
【Memory As a Programming Concept in C a
上一篇文章      下一篇文章      查看所有文章
加:2021-12-24 18:50:05  更:2021-12-24 18:51:44 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/16 10:07:09-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码