大隐于市

用艺术的眼光看待编程!

用户操作
[即时聊天] [发私信] [加为好友]
孙东风ID:dongfengsun
156961次访问,排名540,好友61人,关注者68人。
03年从西安交通大学毕业后一直从事移动平台的开发工作。崇尚一切自由的东西,热爱交流。目前专注于手机二维码技术研究并负责技术管理工作,希望能和同行认识、交流、共享。
dongfengsun的文章
原创 77 篇
翻译 2 篇
转载 33 篇
评论 285 篇
最近评论
xmsnjci:有道理!事业比金钱更重要,个人的技术、能力也是不可缺少的!
phased_array:纯粹C++的一个低级错误,看不出和Symbian有任何关系,难道其他平台就可以这么写了吗?
陈虹全:的确是一个常见的问题,但是个人认为将函数返回值定为引用可不一定是暗示调用者不用释放内存,这是一个非常非常容易出现的问题。作者兄弟遇到的问题相信大家都遇到过。
单就这个例子而言个人认为是设计上的问题。从函数名字和意图来看是要创建一个新的内存单位(一个对象),既然是一个内存单位(说法可能欠妥,姑且用之),那调用方就一定是要以一个指针来指向,除非这个函数的调用要直接作为另一个函数的参数,……
老牛:问题不难 出现也情有可原, 谁没犯低级错误的时候呢

题目却太恶心
darkread:我没有看前文,但是这个文章里面,楼主的举例肯定对的。
不能因为你测试一下,没有非法操作,没有warning就可以了,因为是不可知错误
返回函数内临时对象的引用,这是被严格禁止的!但是可以返回一个对象,前提是有拷贝构造,而且拷贝构造要正确,才会起效果。
大家争论的这段代码,如果有人上过C++的课,一般老师都会讲到的,几乎已经是定论了,忘了C++ primer还是c……
文章分类
收藏
    相册
    存档
    订阅我的博客
    XML聚合  FeedSky

    原创 遍历Symbian某目录下的所有文件收藏

    新一篇: 版本控制软件SubVersion 入门 | 旧一篇: Symbian中左右软键的处理

    遍历Symbian某目录下的所有文件应该是Symbian中常用到的功能模块,比如你想写一个类似“程序管理器”的程序,那么首先的任务就是要先知道某目录下到底有那些文件,然后再筛选出你所需要的文件。

    遍历Symbian某目录下的所有文件有两种方法

      我们首先学习点预备知识

    查看SDK HELP中的GetDir()方法,你会看到如下的内容:

    GetDir()

    TInt GetDir(const TDesC& aName,TUint anEntryAttMask,TUint anEntrySortKey, CDir*& anEntryList) const;

    Description

    Gets a filtered list of a directory's contents. The bitmask determines which file and directory entry types should be listed. The sort key determines the order in which they are listed.

    注释

    得到一个目录下内容的过滤列表。

    anEntryAttMask决定哪个文件和目录应该被列出。

    anEntrySortKey决定这些内容按照什么顺序列出。

    Notes:

    • If sorting by UID (ESortByUid is OR'ed with the entry sort key), then UID information will be included in the listing whether or not KEntryAttAllowUid is specified in anEntryAttMask.(如果按照UID排序,即anEntrySortKey参数的值定义为ESortByUid。那么UID信息将被包含在列表中不管anEntryAttMask参数是否定义了KEntryAttAllowUid)
    • The function sets anEntryList to NULL, then allocates memory for it before appending entries to the list. Therefore, anEntryList should have no memory allocated to it before this function is called, otherwise this memory will become orphaned.(这个函数把anEntryList参数设置为NULL,然后在添加文件项到列表之前为anEntryList分配内存空间。因此,该函数调用之前anEntryList应该没有分配内存,否则这部分内存会变成垃圾而被遗弃)
    • The caller of this function is responsible for deleting anEntryList after the function has returned.(此函数的调用者有责任删除anEntryList在函数返回)

    Parameters

    const TDesC& aName

    Name of the directory for which a listing is required. Wildcards may be used to specify particular files.

    TUint anEntryAttMask

    Bitmask indicating the attributes of interest. Only files and directories whose attributes match those specified here can be included in the listing. For more information see KEntryAttMatchMask and the other directory entry details. Also see KEntryAttNormal and the other file or directory attributes.

    TUint anEntrySortKey

    Flag indicating the order in which the entries are to be sorted. This flag is defined in TEntryKey.

    CDir*& anEntryList

    On return contains a list of directory and file entries.

    Return value

    TInt

    KErrNone if successful, otherwise another of the system-wide error codes.

     

    这是RFs类中的一个方法,从上面的SDK HELP内容我们可以看出:如果我们想获取某个目录下的所有内容,那么只需获取相应目录的CDir指针即可。

    那么我们再看看CDir这个类的SDK HELP

    Class CDir

    CDir

    Support

    Supported from 5.0

    Description

    Array of directory entries that has been read into memory from the file system — abstract base class. It can be read and sorted by user programs, but cannot be created by them.

    This class is not intended for user derivation.

    注释:一个数组,这个数组的内容是从文件系统扫描到的目录,目录被缓存到内存中。它可以被我们的程序读取和排序,但是不能创建。

    Derivation

    o    CBase - Base class for all classes to be instantiated on the heap

    §         CDir - Array of directory entries that has been read into memory from the file system — abstract base class

    Members

    Defined in CDir:
    Count(), NewL(), Sort(), operator[](), ~CDir()

    Inherited from CBase:
    operator new()

        可见,这个类派生自CBase,并且继承了CBasenew()运算符。它里面只有5个方法,去处析构函数,实际上只有4个有用的方法。并且它里面重新定义了[]操作符,因为CDir本身是一个目录的数组。它还可以排序、可以计算数量,这些都是数组的基本特性。NewL()只不过也是一个重载的运算符而已。

        再看看operator[]()的定义:

    operator[]()

    const TEntry& operator[](TInt anIndex) const;

    Description

    Returns an entry from the array.

    Parameters

    TInt anIndex

    Index of the desired entry within the array.

    Return value

    TEntry&

    A directory entry.

        可以看出,通过[]操作符,CDir指针返回给我们的是一个目录的封装类TEntry

     

    我们可以再跟踪一下TEntry这个类的SDK HELP

    Class TEntry

    TEntry

    Support

    Supported from 5.0

    Description

    Encapsulates an entry in a directory, which can be another (nested) directory, a file or a volume label. Each directory entry has a name which is relative to its owning directory and a type, which is indicated by its unique identifier (UID).

    注释:封装目录中的一项内容,内容可以是另一个(嵌套的)目录、一个文件或者是一个驱动器卷标。每个目录项都有一个名字和类型与它的所属目录相关联,也就是UID所显示出来的东西。

    An entry can be interrogated for the following properties:

    一项内容可以被提取以下的属性:

    ·         the kind of entry: stored in the entry UIDs, stored in iType(项的类型:储存在UIDsiType)

    ·         the entry attributes, stored in iAtt(项的属性:储存在iAtt)

    ·         the size of entry(项的尺寸)

    ·         the time the entry was last modified(项上次被修改的时间)

    Members

    Defined in TEntry:
    IsArchive(), IsDir(), IsHidden(), IsReadOnly(), IsSystem(), IsTypeValid(), IsUidPresent(), MostDerivedUid(), iAtt, iModified, iName, iSize, iType, operator[]()

    See also:

        通过以上SDK HELP的内容,我们就知道了大概的方向,从而尝试写出了以下的程序:

    void CCountEntryAppUi::ConstructL()

        {

        BaseConstructL();

     

        RFs iFs;

        User::LeaveIfError(iFs.Connect());

        _LIT(KDIR,"C:\\");

        CDir* dir; 

        User::LeaveIfError(iFs.GetDir(KDIR,KEntryAttNormal|KEntryAttMatchMask,ESortByDate,dir));

        TInt tempInt = dir->Count();

        for(int i = 0;i < tempInt;i++)

        {

            TEntry& iEntry = (TEntry&)dir[i];

            TBufC<256> iFileName = iEntry.iName;

        }

     

        delete dir;

        dir = NULL;

     

        iAppContainer = new (ELeave) CCountEntryContainer;

        iAppContainer->SetMopParent( this );

        iAppContainer->ConstructL( ClientRect() );

        AddToStackL( iAppContainer );

    }

    通过设置断点进行观察,可以看到如下结果:

    Symbian系统C盘下的文件内容如下:

    可见,此时程序所读取到的文件数目是正确的。

     

      下面我们来看第二中获取Symbian某目录下所有文件的方法,其基本原理是类似的,只不过是调用了不同的接口而已。

        RFs iFs;

        User::LeaveIfError(iFs.Connect());

        _LIT(KDIR,"C:\\");

     

        RDir oDir;

        oDir.Open(iFs, KDIR, KEntryAttNormal);

        TEntryArray* oArray = new (ELeave) TEntryArray;

        oDir.Read(*oArray);

        TInt iCount = oArray->Count();

     

        for(TInt i = 0;i < iCount;i++)

     

        {

            TEntry temp = (*oArray)[i];

            TName iTemp = temp.iName;

     

            if(i == 2)

                CEikonEnv::Static()->InfoMsg(iTemp);

        }  

    iFs.Close();

    和第一种方法不同的是,这里使用了RDir类作为目录项的存放数组。我们看看SDK HELP中这个类的帮助文档:

    RDir

    Support

    Supported from 5.0

    Description

    Reads the entries contained in a directory.

    读取一个目录中所包含的项,包括目录、文件以及驱动器卷标,这一点和上面的CDir类是一样的。

    You must first open the directory, specifying an attribute mask which is used by Read() calls to filter the entry types required. Then, use one of the Read() functions to read the filtered entries. When the operation is complete, the directory should be closed using Close(), defined in the base class RFsBase.

    你必须先打开目录,定义一个属性掩码用来过滤项类型。然后用Read()函数去读取过滤的项。当操作完成时,目录应该调用基类RFsBase中定义的Close()方法关闭。

    There are two types of Read(): one works with a single entry at a time, requiring programs to iterate through the entries explicitly. The other works with an entire TEntryArray, allowing multiple entries to be read in one call. As well as making application program logic somewhat simpler, this type uses fewer calls to the server, and is more efficient.

    有两种版本的Read()方法:一种是每次读取一个目录项,要求遍历所有目录项。另一种利用一个TEntryArray工作,它允许多个目录项一次读取完。为了让应用程序逻辑简单,这种类型和服务器的交互少,并且非常高效。

    Each type of Read() can be performed either synchronously or asynchronously.

    每种版本的Read()方法都可以表现为同步的或者异步的。

    It may be more convenient to use RFs::GetDir() than the Read() calls supported by this class. RFs::GetDir() has the advantage that it allows a directory's entries to be sorted in various ways. However, it does not provide asynchronous as well as synchronous variants and does not allow entries to be read individually.

    RFs::GetDir()比这个类所提供的方法更方便,而且RFs::GetDir()允许一个目录的项以不同的方式排序。尽管如此,RFs::GetDir()却不区分同步、异步的方式,并且不允许目录项逐个的读取。

    Derivation

      • RSubSessionBase - Client-side handle to a sub-session
        • RFsBase - Base class that provides a Close() function for file related clean-up
          • RDir - Reads the entries contained in a directory

    Members

    Defined in RDir:
    Open(), Open(), Read(), Read(), Read(), Read()

    Inherited from RFsBase:
    Close()

    Inherited from RSubSessionBase:
    CloseSubSession(), CreateSubSession(), Send(), SendReceive(), SubSessionHandle(), operator=()

     

    看到这里,再和第一种方法进行比较,就应该很容易明白开头所给出的代码了,这段代码在VS2003+Carbide.vs+ S60_2nd_FP2_SC SDK中也编译通过.

    发表于 @ 2007年05月18日 10:10:00|评论(loading...)|编辑

    新一篇: 版本控制软件SubVersion 入门 | 旧一篇: Symbian中左右软键的处理

    评论

    #laji1985 发表于2007-10-16 10:15:50  IP: 222.190.119.*
    写得不错,不过要指出一个错误:
    [code]
    void CCountEntryAppUi::ConstructL()

    {

    BaseConstructL();



    RFs iFs;

    User::LeaveIfError(iFs.Connect());

    _LIT(KDIR,"C:\\");

    CDir* dir;

    User::LeaveIfError(iFs.GetDir(KDIR,KEntryAttNormal|KEntryAttMatchMask,ESortByDate,dir));

    TInt tempInt = dir->Count();

    for(int i = 0;i < tempInt;i++)

    {

    TEntry& iEntry = (TEntry&)dir[i];

    TBufC<256> iFileName = iEntry.iName;

    }



    delete dir;

    dir = NULL;
    [/code]
    其中的
    TEntry& iEntry = (TEntry&)dir[i];

    TBufC<256> iFileName = iEntry.iName;
    两行写得不对,应该写作
    TEntry iEntry = (dir*)[i];

    TBufC<256> iFileName = iEntry.iName;
    照楼主原先写法,是读不到iEntry.iName的
    发表评论  


    登录
    Csdn Blog version 3.1a
    Copyright © 孙东风