眾所周知,在LoadRunner中,關聯是一個很重要的動作,大多數的腳本在錄制完成后并不能直接回放,需要通過一定的關聯才能成功回放。關聯的技巧有很多,這里介紹的就是其中之一,以下用一個實際的例子來說明。
    腳本的背景如下:
    web_submit_data("classiLoanMaterial.jsf_2")(web_submit_data函數的其它部分省略,下同。)返回的頁面上可能存在多條記錄,可能一條,可能兩條,也可能三條,等等。我們需要將這些記錄逐個選中進行操作。注意:不是全部選中,而是要逐條記錄進行操作。同時,每一條記錄各有一個編號,這是需要進行關聯的值。在下面的操作中web_url("directAdjust.jsf",
        "URL=http://128.64.96.105:1158/clpmapp/bizprocess/loanservice/creditassetsriskclassi/
classiadjuststepbystep/directAdjust.jsf?approveFormNum=123456")需要使用到該編號,即黑體字部分的值。面對這樣的目的,很自然地,我們會想到用一個循環語句來實現。首先,在classiLoanMaterial.jsf頁面之前加一個關聯如下:
    web_reg_save_param("sor","LB=sor\" value=\"","RB=\"","Ord=ALL",LAST);將Ord參數值設定為ALL,則關聯函數將自動把符合條件的關聯值保存到參數數組里。在本例中,假設關聯值返回三條記錄,則LR分別將值保存到sor_1,sor_2,sor_3中,同時,LR還將自動創建一個sor_count變量來保存總的記錄數,在這里sor_count值等于3。利用這些信息,我們就可以很方便地在循環語句中實現我們的目的了。步驟如下:
1、聲明各變量:
    int count;
    int i;
    char sor[50];
    char sorvalue[50];
2、將返回的記錄數保存到count變量里:
count=atoi(lr_eval_string("{sor_count}"));
3、使用for循環:
for(i=1;i<=count;i++)
{
    sprintf(sor,"{sor_%d}",i); //分別將各個sor值保存到sor字符串中

    sprintf(sorvalue,"%s",lr_eval_string(sor));//通過lr_eval_string函數將字符串賦給sorvalue變量

}
4、在循環體中使用關聯值替換相關值:
web_url("directAdjust.jsf",
        "URL=http://128.64.96.105:1158/clpmapp/bizprocess/loanservice/creditassetsriskclassi/
classiadjuststepbystep/directAdjust.jsf?approveFormNum={sorvalue}")
 
    一切看起來似乎順理成章,然而如果按照以上的步驟做下來,將會很遺憾地發現:我們定義的{sorvalue}值根本就不被LR認可并接受,于是它將無情地給我們拋出一個錯誤,說該值是非法的。怎么辦?難道我們前面做的一切都白費了嗎?
    有句老話說得好:天無絕人之路。聰明而又善良的LR開發團隊已經為我們考慮到了這個問題,給我們預備了一個很有用的函數:lr_save_string,它可以幫助我們解決這個問題。于是我們祭出lr_save_string這道最后的殺手锏:
5、在使用關聯值之前進行字符串格式轉換:
    lr_save_string(sorvalue,"sorvalue1");
   
web_url("directAdjust.jsf",
        "URL=http://128.64.96.105:1158/clpmapp/bizprocess/loanservice/creditassetsriskclassi/
classiadjuststepbystep/directAdjust.jsf?approveFormNum={sorvalue1}")

需要特別注意lr_save_string的用法,它是參數值在前(sorvalue),參數名在后("sorvalue1"),這和一般的習慣用法正好反過來(真是好奇怪!)。而且"sorvalue1"這個參數名稱不需要事先聲明,它只是一個字符串而已(這也比較奇怪!^_^)。

到此,我們總算大功告成!腳本回放成功,并且正確達到了預期的效果!打完收工!

總結:C的變量不能直接在LR的API里調用,所以必須用lr_save_string進行轉換。

最后順便說一下,lr_save_string這個函數真的很好用,這個例子中提到的方法也適用于另外一些情況,比如說有時候,通過關聯函數出來的值我們不能直接使用,還需要做一些特殊的處理時,那么我們可以把關聯得到的值取出來,賦給一個字符串,對其進行一番修剪加工后,再用lr_save_string,就可以使用它來替代需要關聯的值了。

后記:我的這篇文章發布在網上以后,在廣大的測試同行中間引起了強烈的反響,他們紛紛發來賀電和表揚信,對我這種勇于探索、樂于分享的精神給予了充分的肯定。^_^當然,這中間也難免存在極個別的不和諧聲音,例如Zee同學就對我的這篇文章提出了不同看法,他覺得我的做法是把簡單的問題復雜化了,理由是可以只做一次關聯,每次只取第一筆記錄即可,當循環進行操作時,第一筆做完以后,第二筆記錄自然會上升到第一筆記錄的位置,因此沒有必要使用關聯數組。我認為他的疑問并非沒有道理,而且是比較有代表性的,因此我在這里做一個補充說明。在我接觸過的大多數應用系統中,確實都是按照Zee所說的方式進行處理,在這種情況下,腳本的處理的確沒有必要像我以上所述的那樣復雜。不過我在本例中談到的例子比較特殊,在操作完成后,它只是把每筆記錄的狀態位由“未完成”修改為“已完成”,而原有的記錄并沒有消失,而是仍然停留在原有的位置,此時如果按照Zee所說的方法,那么在執行第二次循環時,LR將取到操作狀態為“已完成”的第一筆記錄,而不會取到下一筆未完成的記錄,顯然這是不符合我們的要求的,因此在這里我需要做以上這樣復雜的一個處理。