CSOM - FileRef vs ServerUrl vs sp.listitem.ServerRelativeUrl

时间:2017-09-28 13:42:39

标签: sharepoint csom

我最近一直致力于SharePoint CSOM,并且一直在处理文档库中的文件处理(上载/下载/元数据更新)。我有几种方法来获取文件的服务器相对URL,即listitem [“FileRef”]或listitem.ServerRelativeUrl。此链接a link提供sharepoint字段的内部名称列表。请注意,它将“服务器相对URL”列为“ServerUrl”。但是,当我尝试访问类似listitem [“ServerUrl”]之类的东西时会抛出异常:

尚未初始化属性或字段。尚未请求或请求尚未执行。可能需要明确请求。

继承我的代码示例

    using (ClientContext clientContext = new ClientContext(siteUrl))
{
clientContext.Credentials = new NetworkCredential("abc", "abc", "cde");
List oList = clientContext.Web.Lists.GetByTitle("abcd");
clientContext.Load(oList);
clientContext.ExecuteQuery();
CamlQuery camlQuery = new CamlQuery();

ListItemCollectionPosition itemPosition = null;

conn.Open();
while (true)
{
    camlQuery = new CamlQuery();
    camlQuery.ListItemCollectionPosition = itemPosition;

        camlQuery.ViewXml =
            //@"<View Scope='RecursiveAll'>
            string.Format(@"<View Scope='RecursiveAll'>
                            <Query>
                                <Where>
                                    <And>
                                        <Eq>
                                            <FieldRef Name='FSObjType' />
                                            <Value Type='Integer'>0</Value>
                                        </Eq>
                                        <In>
                                            <FieldRef Name='FileLeafRef' />
                                            <Values>
                                                {0}
                                            </Values>
                                        </In>                                
                                    </And>
                                </Where>
                            </Query>
                            <FieldRef Name='FileLeafRef' /><FieldRef Name='FileRef'><FieldRef Name='ServerUrl'>
                            <RowLimit>100</RowLimit>
                          </View>", CAMLcoll);

    ListItemCollection listItems = oList.GetItems(camlQuery);

    clientContext.Load(listItems);
    clientContext.ExecuteQuery();
    itemPosition = listItems.ListItemCollectionPosition;

    foreach (ListItem listItem in listItems)
    {
        try
        {
            clientContext.Load(listItem, eachitem => eachitem.File.ServerRelativeUrl);
            clientContext.ExecuteQuery();
            string url = listItem["FileRef"].ToString(); //This works
            url = listItem.ServerRelativeUrl.ToString(); //This works
            url = listItem["ServerUrl"].ToString(); //This throws exception!!
        }
        catch (Exception ex)
        {
            MessageBox.Show(ex.ToString());
        }
    }
    if (itemPosition == null)
        break;
    //Debug.Print(itemPosition.PagingInfo);
}

我有几个问题:

  1. 这3种访问项目服务器相对URL的方式有什么区别?

  2. 为什么 url = listItem [“ServerUrl”]。ToString()失败?

  3. 我对CSOM编程很陌生,所以请原谅我的无知。

    由于

    阿玛尔

1 个答案:

答案 0 :(得分:1)

首先,ServerRelativeUrl仅适用于File类。

因此,您的代码应为listItem.File.ServerRelativeUrl.ToString();

关于你与差异有关的第一个问题,我认为他们都给出了相同的结果。这意味着,FileReflistItem.File.ServerRelativeUrlServerUrl为您提供文档库中文档的相同服务器相对路径。

关于第二个问题,在SharePoint CSOM中,您需要明确加载要使用的属性。提供了一些基本属性,但对于某些其他属性,例如ServerUrl,您需要显式加载它以便可以使用它。

第三,您的代码正在执行并请求每个列表项的数据。这根本不需要。您可以在列表项集合中加载这些属性。

修改,简化和优化您的代码。看看:

ListItemCollection listItems = oList.GetItems(camlQuery);
clientContext.Load(listItems,items => items.Include(
                        item => item["FileRef"],
                        item => item.File.ServerRelativeUrl,
                        item => item["ServerUrl"]));
clientContext.ExecuteQuery();

itemPosition = listItems.ListItemCollectionPosition;

foreach (ListItem listItem in listItems)
{
    try
    {

        string url = listItem["FileRef"].ToString(); //This works
        url = listItem.File.ServerRelativeUrl.ToString(); //This should work
        url = listItem["ServerUrl"].ToString(); //This will work now
    }
    catch (Exception ex)
    {
        MessageBox.Show(ex.ToString());
    }
}