Apex基础
Apex 语言亮点
像其他面向对象的编程语言一样,这些是Apex支持的一些语言结构:
- 类,接口,属性和集合(包括数组)。
- 对象和数组表示法。
- 表达式,变量和常量。
- 条件语句(if-then-else)和控制流语句(for循环和while循环)。
与其他面向对象的编程语言不同,Apex支持:
- 作为Apex的云开发是在云中存储,编译和执行的。
- 触发器,类似于数据库系统中的触发器。
- 数据库语句,允许您直接进行数据库调用和查询语言来查询和搜索数据。
- 事务和回滚。
- 全局访问修饰符,它比public修饰符更宽松,并允许跨命名空间和应用程序访问。
- 自定义代码的版本。
另外,Apex是一个不区分大小写的语言。
Apex语法
- Switch
String waterLevel = 'empty';
//option 1 using a single value
switch on waterLevel{
when 'empty'{
System.debug('Fill the tea kettle');
}
when 'half'{
System.debug('Fill the tea kettle');
}
when 'full'{
System.debug('The tea kettle is full');
}
when else{
System.debug('Error!');
}
}
//option 2 using multiple values
switch on waterLevel{
when 'empty', 'half'{ //when waterLevel is either empty or half
System.debug('Fill the tea kettle');
}
when 'full'{
System.debug('The tea kettle is full');
}
when else{
System.debug('Error!');
}
输出结果为:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-4V3CrfjK-1572587509250)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\1571644156217.png)]
APEX动态运行代码行 (调试|测试)
- 打开开发人员控制台( Developer console )
- 点击Debug | Open Execute Anonymous Window. [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ovCiC68A-1572587509251)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\1571104642554.png)]
- 运行代码:
①点击Execute(运行全部)
②点击**Execute Highlighted(运行选中部分代码) **
DML
使用数据操作语言(缩写为DML)在Salesforce中创建和修改记录。使用数据操作语言(缩写为DML)在Salesforce中创建和修改记录。
DML语句
语句 | 作用 | 语法 |
insert | 新增 | insert user; |
update | 修改 | |
upsert | 更新插入 | |
delete | 删除 | |
undelete | 回复删除 | |
merge | 合并 |
- insert
新增单条记录
// 创建一个客户对象
Account acct = new Account(Name='Acme', Phone='(415)555-1212', NumberOfEmployees=100);
// 使用DML插入客户信息
insert acct;
//插入记录时,系统会为每个记录分配一个ID。
//ID值还会自动填充到在DML调用中用作参数的sObject变量上。(也就是说将分配的ID赋值给对象)
ID acctID = acct.Id;
System.debug('ID = ' + acctID);
批量新增
List<Contact> conList = new List<Contact> {
new Contact(FirstName='Joe',LastName='Smith',Department='Finance'),
new Contact(FirstName='Kathy',LastName='Smith',Department='Technology'),
new Contact(FirstName='Caroline',LastName='Roth',Department='Finance'),
new Contact(FirstName='Kim',LastName='Shain',Department='Education')};
// Bulk insert all contacts with one DML call
insert conList;
- update
批量修改
List<Contact> conList = new List<Contact> {
new Contact(FirstName='Joe',LastName='Smith',Department='Finance'),
new Contact(FirstName='Kathy',LastName='Smith',Department='Technology'),
new Contact(FirstName='Caroline',LastName='Roth',Department='Finance'),
new Contact(FirstName='Kim',LastName='Shain',Department='Education')};
insert conList;
List<Contact> listToUpdate = new List<Contact>();
for(Contact con : conList) {
if (con.Department == 'Finance') {
con.Title = 'Financial analyst';
// Add updated contact sObject to the list.
listToUpdate.add(con);
}
}
update listToUpdate
- Upsert
什么时候用Upsert?
如果你有一个包含新纪录和现有记录的集合,则可以使用Upsert来对所有记录进行插入和更新。
Upsert的优势?
如果集合的记录在数据库中存在,则会更新记录。如果集合的记录在数据库中不存在,则会新增记录。Upsert有助于避免重复记录的创建,并可以节省您的时间,因为您不必先确定哪些记录。
注意:Upsert语句通过比较一个字段的值将”操作的数据“和”数据库中的数据“作对比。
如果你没指定字段,Upsert使用当前对象的ID做对比。
①指定字段
upsert sObjectList Account.Fields.MyExternalId;//指定字段作为对比值
②不指定字段
Contact josh = new Contact(FirstName='Josh',LastName='Kaplan',Department='Finance');
insert josh;
josh.Description = 'Josh\'s record has been updated by the upsert operation.';
Contact kathy = new Contact(FirstName='Kathy',LastName='Brown',Department='Technology');
List<Contact> contacts = new List<Contact> { josh, kathy };
upsert contacts;
Upsert对比结果:
- 如果字段值不匹配,则会创建一个新的对象记录。
- 如果字段值匹配一次,则现有对象记录将更新。
- 如果字段值多次匹配,则会生成错误,并且对象记录不会插入或更新。
- delete
删除的记录不会从 Lightning Platform (数据库)上永久删除,但是会将它们放置在回收站中15天内,从那里可以恢复它们。
语法:
Contact[] contactsDel = [SELECT Id FROM Contact WHERE LastName='Smith'];
delete contactsDel;
DML语句异常
如果DML操作失败,则返回类型为的异常 DmlException。您可以在代码中捕获异常以处理错误情况。
这个例子产生了一个 DmlException因为它尝试插入没有必填名称字段的帐户。异常捕获在catch块中。
try {
Account acct = new Account();
insert acct;
} catch (DmlException e) {
System.debug('发生了DML异常: ' +
e.getMessage());
}
数据库方法
Apex包含内置的Database类,该类提供执行DML操作并镜像DML语句副本的方法。
这些数据库方法是静态的,并在类名上调用。
- Database.insert()
- Database.update()
- Database.upsert()
- Database.delete()
- Database.undelete()
- Database.merge()
与DML语句不同,数据库方法具有可选的allOrNone 参数,该参数使您可以指定操作是否允许部分成功。当此参数设置为false时,如果部分记录操作发生错误,则将提交成功的记录,并为失败的记录返回错误。此外,部分操作成功的记录不会引发任何异常。
Database.insert(recordList, false);
数据库方法返回结果对象,其中包含每个记录的成功或失败信息。
Database.SaveResult[] results = Database.insert(recordList, false);
默认情况下,allOrNone参数是true,这意味着Database方法的行为如果遇到失败,将抛出异常,回滚所有操作。
以下语句的效果一致
Database.insert(recordList);
Database.insert(recordList, true);
insert recordList;
关联操作(主从)
insert关联记录(需要多次DML)
Account acct = new Account(Name='SFDC Account');
insert acct;
ID acctID = acct.ID;
//将客户关联到对应的联系人(需要执行俩次DML)
Contact mario = new Contact(
FirstName='Mario',
LastName='Ruiz',
Phone='415.555.1212',
AccountId=acctID);
insert mario;
update关联记录(需要多次DML)
Contact queriedContact = [SELECT Account.Name
FROM Contact
WHERE FirstName = 'Mario' AND LastName='Ruiz'
LIMIT 1];
queriedContact.Phone = '(415)555-1213';
queriedContact.Account.Industry = 'Technology';
update queriedContact;//更新联系人信息
update queriedContact.Account; //更新联系人的客户信息
删除关联记录
delete操作支持级联删除, 如果删除父对象,就会自动删除其子对象。
例如,删除帐户也将删除其相关联系人。
Account[] queriedAccounts = [SELECT Id FROM Account WHERE Name='SFDC Account'];
delete queriedAccounts;
SOQL(类似SQL)
SOQL对象查询语言
SOQL每次查询单个表的数据上限是200,如果超过200条数据,则会再发另外一个请求去查询剩余的数据。
您无需在查询中指定Id字段,因为它始终在Apex查询中返回,除非你只查询ID字段,这个时候则需要指定ID字段。
在查询编辑器中运行查询时,您可能还需要指定ID字段,因为除非指定,否则不会显示ID字段。
当SOQL嵌入Apex中时,称为内联SOQL。
Account[] accts = [SELECT Name,Phone FROM Account];
使用查询编辑器
- 在开发人员控制台中,单击Query Editor[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Y6ehhmiZ-1572587509252)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\1571127124119.png)]
- 输入你要测试的sql,然后点击Execute运行
使用条件过滤查询结果
SELECT Name,Phone FROM Account WHERE (Name='SFDC Computing' OR (NumberOfEmployees>25 AND BillingCity='Los Angeles'))
排序
您可以对大多数字段进行排序,包括数字和文本字段。您无法对富文本格式和多选选择列表之类的字段进行排序。
SELECT Name,Phone FROM Account ORDER BY Name ASC
SELECT Name,Phone FROM Account ORDER BY Name DESC
限制返回的记录数
例如,此查询检索返回的第一个帐户。请注意,当limit 1的时候使用时返回的值是一个帐户而不是数组。
Account oneAccountOnly = [SELECT Name,Phone FROM Account LIMIT 1];
在SOQL查询中调用Apex中的变量
使用:引用变量
String targetDepartment = 'Wingo';
Contact[] techContacts = [SELECT FirstName,LastName
FROM Contact WHERE Department=:targetDepartment];
查询关联记录
当查询名称为"SFDC Computing"的客户时,同时查询出该客户对应的联系人
SELECT Name, (SELECT LastName FROM Contacts) FROM Account WHERE Name = 'SFDC Computing'
使用关联查询出来的结果还能通过.调用相关联的数据
Account[] acctsWithContacts = [SELECT Name, (SELECT FirstName,LastName FROM Contacts)
FROM Account
WHERE Name = 'SFDC Computing'];
// 获取从表数据
Contact[] cts = acctsWithContacts[0].Contacts;
System.debug('Name of first associated contact: '
+ cts[0].FirstName + ', ' + cts[0].LastName);
也可以从子表数据调用主表数据
Contact[] cts = [SELECT Account.Name FROM Contact];
Contact carol = cts[0];
String acctName = carol.Account.Name;
System.debug('account name is ' + acctName);
使用外部变量+Like语句
[SELECT Name FROM Contact WHERE Name like :firstName+'%' ]
SOSL
Salesforce对象搜索语言(SOSL)是一种Salesforce搜索语言,用于在记录中执行文本搜索。使用SOSL在Salesforce中跨多个标准和自定义对象记录搜索字段。文本搜索不区分大小写 。
语法:
//IN ALL FIELDS 默认查询全部字段所以可有可无
FIND {bo} RETURNING Contact
FIND {bo} IN ALL FIELDS RETURNING Contact(name)
//如果需要指定特定字段查找
FIND {bo} IN Email FIELDS RETURNING Contact(name)
//跟在对象后面的字段是返回的内容
FIND {bo} IN ALL FIELDS RETURNING Contact(id,name)
//Order by 排序某个字段
FIND {Cloud Kicks} RETURNING Account (Name ORDER BY Name)
//limit设置返回的最大记录数
FIND {Cloud Kicks} RETURNING Account (Name ORDER BY Name LIMIT 10)
//offset将起始行偏移量设置为结果
FIND {Cloud Kicks} RETURNING Account (Name ORDER BY Name LIMIT 10 OFFSET 25)
这是一个SOSL查询的示例,该查询搜索具有任何字段带有单词“ SFDC”的字段的客户和联系人。
List<List<SObject>> searchList = [FIND 'SFDC' IN ALL FIELDS
RETURNING Account(Name), Contact(FirstName,LastName)];
案例:在客户的字段Name,联系人字段FirstName,LastName,Department,中查找值为Wingo的记录
FIND {Wingo} IN ALL FIELDS RETURNING Account(Name), Contact(FirstName,LastName,Department)
SOSL关键字
关键字 | 作用 |
IN | 限制要搜索的字段类型,包括电子邮件,姓名或电话 |
LIMIT | 指定要返回的最大行数 |
OFFSET | 在多页上显示搜索结果 |
RETURNING | 限制对象和字段返回 |
WITH DATA CATEGORY | 指定要返回的数据类型 |
WITH DivisionFilter | 指定要返回的除法字段 |
WITH NETWORK | 指定要返回的社区ID |
WITH PricebookId | 指定要返回的price book ID |
SOQL与SOSL之间的异同
SOSL与SOQL一样,SOSL允许您在组织的记录中搜索特定信息。与一次只能查询一个标准或自定义对象的SOQL不同,单个SOSL查询可以搜索所有对象。
另一个区别是,SOSL根据单词匹配来匹配字段,而默认情况下(当不使用通配符时)SOQL执行完全匹配。例如,在SOSL中搜索“数字”将返回字段值为“数字”或“数字公司”的记录,但SOQL仅返回字段值为“数字”的记录。
SOQL和SOSL是两种具有不同语法的独立语言。每种语言都有不同的用例:
- 使用SOQL检索单个对象的记录。
- 使用SOSL在多个对象之间搜索字段。SOSL查询可以搜索对象上的大多数文本字段。
APEX触发器
Apex触发器使您可以在事件发生之前或之后对Salesforce中的记录(例如插入,更新或删除)执行自定义操作。就像数据库系统支持触发器一样,Apex为管理记录提供触发器支持。
您可以使用触发器来执行在Apex中可以做的任何事情,包括执行SOQL和DML或调用自定义Apex方法。
管理触发器
你可以在此页面上编辑、删除等操作
禁用触发器
- 在设置中,使用左侧搜索栏搜索Apex 触发器。
- 点击触发器旁边的编辑按钮
- 取消 Is Active的勾选[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-0V8XE9GS-1572587509253)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\1571218889699.png)]
触发器的语法
trigger ContextExampleTrigger on Account (before insert, after insert, after delete) {
if (Trigger.isInsert) {
if (Trigger.isBefore) {
// Process before insert
} else if (Trigger.isAfter) {
// Process after insert
}
}
else if (Trigger.isDelete) {
// Process after delete
}
}
触发上下文变量
变量 | 用法 |
isExecuting | 如果Apex代码的当前上下文是触发器,而不是Visualforce页面,Web服务或控件,则返回true。 executeanonymous() API调用。 |
isInsert | 如果此触发器是由来自Salesforce用户界面、Apex或API的插入操作触发的,则返回true。 |
isUpdate | 如果此触发器是由来自Salesforce用户界面、Apex或API的更新操作触发的,则返回true。 |
isDelete | 如果此触发器是由Salesforce用户界面、Apex或API中的删除操作触发的,则返回true。 |
isBefore | 如果在保存任何记录之前触发此触发器,则返回true。 |
isAfter | 如果在所有记录都是savec之后触发此触发器,则返回true |
isUndelete | 如果在从回收站恢复记录后触发此触发器,则返回true。此恢复可以在从Salesforce用户界面、Apex或API执行取消删除操作后发生。 |
new | 返回sObject记录的新版本列表。这个sObject列表只能在insert、update和undelete触发器中使用,记录只能在before触发器中修改。 |
newMap | 将IDs映射到sObject记录的新版本。此映射仅在更新前、插入后、更新后和取消删除触发器后可用。 |
old | 返回sObject记录的旧版本列表。此sObject列表仅在更新和删除触发器中可用。 |
old map | IDs到sObject记录的旧版本的映射。此映射仅在更新和删除触发器中可用。 |
operationType | 返回一个System类型的枚举。对应于当前操作的TriggerOperation。系统的可能值。TriggerOperation enum是:BEFORE_INSERT、BEFORE_UPDATE、BEFORE_DELETE、AFTER_INSERT、AFTER_UPDATE、AFTER_DELETE和AFTER_UNDELETE。如果您根据不同的触发器类型改变您的编程逻辑,那么可以考虑使用switch语句,使其具有惟一触发器执行枚举状态的不同排列。 |
size | 触发器调用中的记录总数,包括旧的和新的。 |
触发器的实际操作
在触发器内部调用类方法,触发器除了可以在内部改变当前记录的值,还能调用外部的方法
//内部修改值
trigger HelloWorldTrigger on Account (before insert) {
for(Account a : Trigger.New) {
a.Description = 'New description';
}
}
//内部调用外部类方法
//当联系人有新纪录新增或删除时,调用邮箱类的方法发信息给管理员
trigger ExampleTrigger on Contact (after insert, after delete) {
if (Trigger.isInsert) {
Integer recordCount = Trigger.New.size();
// Call a utility method from another class
EmailManager.sendMail('Your email address', 'Trailhead Trigger Tutorial',
recordCount + ' contact(s) were inserted.');
}
else if (Trigger.isDelete) {
// Process after delete
}
}
使用触发器添加关联记录
//当客户记录被新增或者修改时,遍历所有客户对象,如果当前客户没有关联的业务机会,则自动帮客户创建一个关联当前用户的业务机会
trigger AddRelatedRecord on Account(after insert, after update) {
List<Opportunity> oppList = new List<Opportunity>();
// Get the related opportunities for the accounts in this trigger
Map<Id,Account> acctsWithOpps = new Map<Id,Account>(
[SELECT Id,(SELECT Id FROM Opportunities) FROM Account WHERE Id IN :Trigger.New]);
// Add an opportunity for each account if it doesn't already have one.
// Iterate through each account.
for(Account a : Trigger.New) {
System.debug('acctsWithOpps.get(a.Id).Opportunities.size()=' + acctsWithOpps.get(a.Id).Opportunities.size());
// Check if the account already has a related opportunity.
if (acctsWithOpps.get(a.Id).Opportunities.size() == 0) {
// If it doesn't, add a default opportunity
oppList.add(new Opportunity(Name=a.Name + ' Opportunity',
StageName='Prospecting',
CloseDate=System.today().addMonths(1),
AccountId=a.Id));
}
}
if (oppList.size() > 0) {
insert oppList;
}
触发器异常
有时您需要对某些数据库操作添加限制,例如在满足某些条件时阻止保存记录。
以下触发条件可防止删除具有相关业务机会的客户。默认情况下,删除客户会导致其所有相关记录的级联删除。此触发器可防止业务机会的级联删除。
trigger AccountDeletion on Account (before delete) {
// Prevent the deletion of accounts if they have related opportunities.
for (Account a : [SELECT Id FROM Account
WHERE Id IN (SELECT AccountId FROM Opportunity) AND
Id IN :Trigger.old]) {
Trigger.oldMap.get(a.Id).addError(
'Cannot delete account with related opportunities.');
}
}
触发器和标注(Triggers and Callouts)
Apex允许您调用Apex代码并将其与外部Web服务集成。对外部Web服务的Apex调用称为标注。
//标有@future(callout=true)的方法都是异步方法
public class CalloutClass {
@future(callout=true)
public static void makeCallout() {
HttpRequest request = new HttpRequest();
// Set the endpoint URL.
String endpoint = 'http://yourHost/yourService';
request.setEndPoint(endpoint);
// Set the HTTP verb to GET.
request.setMethod('GET');
// Send the HTTP request and get the response.
HttpResponse response = new HTTP().send(request);
}
}
//异步调用外部类方法
trigger CalloutTrigger on Account (before insert, before update) {
CalloutClass.makeCallout();
}
触发器批量操作
执行批量SOQL
反例:查询每个客户的业务机会,需要发n次SOQL(n为客户数量)
trigger SoqlTriggerNotBulk on Account(after update) {
for(Account a : Trigger.New) {
// Get child records for each account
// Inefficient SOQL query as it runs once for each account!
Opportunity[] opps = [SELECT Id,Name,CloseDate
FROM Opportunity WHERE AccountId=:a.Id];
// Do some other processing
}
}
如何优化批量SOQL?
将Trigger.New里面所有的客户作为一个集,使用IN查询此集里满足条件的业务机会对象
trigger SoqlTriggerBulk on Account(after update) {
// Perform SOQL query once.
// Get the related opportunities for the accounts in this trigger,
// and iterate over those records.
for(Opportunity opp : [SELECT Id,Name,CloseDate FROM Opportunity
WHERE AccountId IN :Trigger.New]) {
// Do some other processing
}
}
执行批量DML
反例:将DML操作写在for循环中,会极大的影响其效率。
trigger DmlTriggerNotBulk on Account(after update) {
// Get the related opportunities for the accounts in this trigger.
List<Opportunity> relatedOpps = [SELECT Id,Name,Probability FROM Opportunity
WHERE AccountId IN :Trigger.New];
// Iterate over the related opportunities
for(Opportunity opp : relatedOpps) {
// Update the description when probability is greater
// than 50% but less than 100%
if ((opp.Probability >= 50) && (opp.Probability < 100)) {
opp.Description = 'New description for opportunity.';
// Update once for each opportunity -- not efficient!
update opp;
}
}
}
如何优化批量DML?
创建一个新的集合,装新的记录数据。遍历完之后再执行单个DML操作,操作集合。
trigger DmlTriggerBulk on Account(after update) {
// Get the related opportunities for the accounts in this trigger.
List<Opportunity> relatedOpps = [SELECT Id,Name,Probability FROM Opportunity
WHERE AccountId IN :Trigger.New];
List<Opportunity> oppsToUpdate = new List<Opportunity>();
// Iterate over the related opportunities
for(Opportunity opp : relatedOpps) {
// Update the description when probability is greater
// than 50% but less than 100%
if ((opp.Probability >= 50) && (opp.Probability < 100)) {
opp.Description = 'New description for opportunity.';
oppsToUpdate.add(opp);
}
}
// Perform DML on a collection
update oppsToUpdate;
}
APEX单元测试
编写Test类基本步骤可以分成4步:
1.创建测试数据;
2.调用Test.startTest()方法;
3.调用需要测试的方法();
4.调用Test.stopTest()方法。
测试方法语法
第一种
@isTest static void testName() {
// code_block
}
第二种
static testMethod void testName() {
// code_block
}
第三种
@isTest
private class MyTestClass {
@isTest static void myTest() {
// code_block
}
}
使用==System.assertEquals()==验证。它有两个参数:第一个是期望值,第二个是实际值。
@isTest static void testWarmTemp() {
Decimal number = 10/2;
System.assertEquals(5,number);
}
也可以有三个参数,第一个是期望值,第二个是实际值,第三个是测试失败后提示的语句。
@isTest static void testBoilingPoint() {
Decimal number = 10/2;
// Simulate failure
System.assertEquals(0,number,'number 不是预期的值,测试失败');
}
等于运算符(==)执行不区分大小写的字符串比较
在单元测试中,建立代码测试覆盖100%,最低要求75%。
蓝色(覆盖)线和红色(未覆盖)线
如何全部覆盖?
再写一个测试方法保证他能进到红色的代码中。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-iIdPdtnW-1572587509254)(https://res.cloudinary.com/hy4kyit2a/f_auto,fl_lossy,q_70/learn/modules/apex_testing/apex_testing_intro/images/4177c5d150c4add6e968944ddd37e9de_apex_testing_code_coverage_partial.png)]
查看测试代码的覆盖程度
- 在运行完测试类或者测试套件后点击控制台下方的Tests选项卡
- 在右侧Overall Code Coverage下能看到每个测试类的结果。[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-hqqvj3Fo-1572587509255)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\1571218273379.png)]
- 双击某一个测试类还能看到详细的覆盖情况。[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-CgiXdyGP-1572587509256)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\1571218310818.png)]
测试套件
创建测试套件
- 点击 Test | New Suite
- 给测试套件命名
- 选择你需要测试的测试类[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-KbJD9j5Y-1572587509257)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\1571216920139.png)]
- 点击save保存套件。
执行测试套件
- 选择 Test | New Suite Run 。
- 选择你要测试的套件名称,然后单击 **>**移动套件名称到“Selected Test Suites ”列。
- 单击Run Suites。
- 在“测试”选项卡上,监视测试运行的状态。展开测试运行,然后再次展开,直到看到已运行的各个测试的列表。就像在各种测试方法中一样,您可以双击方法名称以查看详细的测试结果。[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-zgBlccVF-1572587509258)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\1571217135915.png)]
创建Apex测试的测试数据
我们需要在测试类本身创建测试类的数据。默认情况下,测试类不能访问组织数据,但是如果您设置@isTest(seeAllData = true),那么它将有权访问组织的数据。
在测试类中使用的DML,不会影响真实数据库的数据。并且你还能创建测试数据以便测试。如下:
@isTest
public class TestDataFactory {
public static List<Account> createAccountsWithOpps(Integer numAccts, Integer numOppsPerAcct) {
List<Account> accts = new List<Account>();
for(Integer i=0;i<numAccts;i++) {
Account a = new Account(Name='TestAccount' + i);
accts.add(a);
}
insert accts;
List<Opportunity> opps = new List<Opportunity>();
for (Integer j=0;j<numAccts;j++) {
Account acct = accts[j];
// For each account just inserted, add opportunities
for (Integer k=0;k<numOppsPerAcct;k++) {
opps.add(new Opportunity(Name=acct.Name + ' Opportunity ' + k,
StageName='Prospecting',
CloseDate=System.today().addMonths(1),
AccountId=acct.Id));
}
}
// Insert all opportunities for all accounts.
insert opps;
return accts;
}
}
TestMethod关键字
单元测试方法是不带参数,不向数据库提交数据,不发送电子邮件,并在方法定义中使用testMethod关键字或isTest注释声明的方法。此外,测试方法必须在测试类中定义,即用isTest注释的类。
Test.startTest()和Test.stopTest()
这些是可用于测试类的标准测试方法。这些方法包含我们将模拟我们的测试的事件或动作。就像在这个例子中,我们将测试我们的触发器和帮助类来模拟火灾触发器,通过更新记录,我们已经做了开始和停止块。这也为在开始和停止块中的代码提供单独的调节器限制。
System.assert()
此方法用实际检查所需的输出。在这种情况下,我们期望插入一个发票记录,所以我们添加了assert来检查。
@isTest
private class TestAccountDeletion {
@isTest static void TestDeleteAccountWithOneOpportunity() {
// Test data setup
// Create one account with one opportunity by calling a utility method
Account[] accts = TestDataFactory.createAccountsWithOpps(1,1);
// Perform test
Test.startTest();
Database.DeleteResult result = Database.delete(accts[0], false);
Test.stopTest();
// Verify that the deletion should have been stopped by the trigger,
// so check that we got back an error.
System.assert(!result.isSuccess());
System.assert(result.getErrors().size() > 0);
System.assertEquals('Cannot delete account with related opportunities.',
result.getErrors()[0].getMessage());
}
}
``
private class TestAccountDeletion {
@isTest static void TestDeleteAccountWithOneOpportunity() {
// Test data setup
// Create one account with one opportunity by calling a utility method
Account[] accts = TestDataFactory.createAccountsWithOpps(1,1);
// Perform test
Test.startTest();
Database.DeleteResult result = Database.delete(accts[0], false);
Test.stopTest();
// Verify that the deletion should have been stopped by the trigger,
// so check that we got back an error.
System.assert(!result.isSuccess());
System.assert(result.getErrors().size() > 0);
System.assertEquals('Cannot delete account with related opportunities.',
result.getErrors()[0].getMessage());
}
}