我的程序中有一个列表,希望像资源管理器那样,能够一按列表头,就能对这一列进行排序,我的做法是:在消息响应函数中调用SortItems((PFNLVCOMPARE)CompareFunc,(DWORD)m_nSortedCol);它在调用回调函数CompareFunc,能对数据进行排序,但是在
列表框里没有显示排序后的数据,为什么?是不是要自己编代码实现,怎么实现,哪位大虾给点提示吧?多谢了!

解决方案 »

  1.   

    CListView类中包含了一个CListCtrl类的引用,并由这个CListCtrl类实现了List View窗口,窗口中包含了由程序添加的List Item。List Item则以LVITEM结构形式存在。
    在一个ListView窗口中,当所有的Item以Report的形式出现时,应该允许用户单击列标题,然后以该列的顺序进行升序排序或反序排序List Item。
    当用户单击列标题时,我们可在该动作的响应函数中调用CListCtrl类的SortItems函数来实现排序。该函数的原型是:BOOL SortItems( PFNLVCOMPARE pfnCompare, DWORD dwData ),其中pfnCompare是一个用户自定义的用来进行比较的函数,dwData是程序准备向pfnCompare传递的附加的信息。
    PFNLVCOMPARE的原型定义是这样的:
    int CALLBACK CompareFunc(LPARAM lParam1, LPARAM lParam2, 
       LPARAM lParamSort);
    lParam1与lParam2参数分别是两个Item所包含的LVITEM结构中的lParam成员变量,lParamSort参数则是上面SorItems函数传递过来的dwData数据,一般用它表示是按增序还是降序来进行排序。CompareFunc函数一般返回-1,0,1,表示此次比较的结果。
    SortItems函数通过循环调用CompareFunc函数来实现对List Item的排序,并根据CompareFunc函数的返回值决定每两个Item的相对位置。
     一般来说,每条List Item只能指定一个lParam数据,当我们要实现对不同的列进行排序时,则需要在用户单击列标题后,根据不同的列对每个List Item的lParam数据进行更新,使其反应每条List item相应Sub Item的内容,并提供给比较函数,使CListCtrl在排序时根据该列的内容进行排序。
    为此,我们可以在添加List Item时为List Item的lParam变量指定一个字符串指针保存当前排序列的文本信息,并在需要对lParam的数据进行更新时,指定不同的字符器指针来返回排序列的信息即可。
    下面举例说明上述操作。// 在类说明中必须将下面这个ListViewCompareFunc定义成static型
    int CALLBACK CMyListView::ListViewCompareFunc(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort)
    {
     // 得到排序方式
     int * pisortorder = (int *)lParamSort;
     
     // 得到两个列的排序信息
     TCHAR * sz1 = (TCHAR *)lParam1;
     TCHAR * sz2 = (TCHAR *)lParam2; // 比较列的信息并返回比较结果。
     // 若为减序,则将比较结果乘上-1。
     if (* pisortorder == LVS_SORTASCENDING)
      return lstrcmp(sz1, sz2);
     else
      return lstrcmp(sz1, sz2) * (-1);
    }
    void CMyListView::OnColumnclick(NMHDR* pNMHDR, LRESULT* pResult) 
    {
     static int ncurSortCol = -1; // 保存当前的排序列。
      // 一开始表示为-1,表示尚未按任何列排序。 NM_LISTVIEW* pNMListView = (NM_LISTVIEW*)pNMHDR;
     
     CListCtrl* lc = &GetListCtrl();
     LONG ws = GetWindowLong(lc->m_hWnd, GWL_STYLE);
     int nSortOrder; // 排序的方式 // 若点击列与当前排序列不同的列,则改变排序序,并将排序方式改为增序。
     // 若当前排序列与点击列相同,则更改增、减序的排序方式
     if (ncurSortCol == pNMListView->iSubItem)
     {
      if (ws & LVS_SORTASCENDING)
      {
       ws ^= LVS_SORTASCENDING;
       nSortOrder = LVS_SORTDESCENDING;
      }
      else
      {
       ws ^= LVS_ SORTDESCENDING;
       nSortOrder = LVS_SORTASCENDING;
      }
     }
     else
     {
      if (ws & LVS_SORTASCENDING)
      ncurSortCol = pNMListView->iSubItem;
      if (ws & LVS_SORTASCENDING)
       nSortOrder = LVS_SORTASCENDING;
      else
       nSortOrder = LVS_ SORTDESCENDING;
     } // 将当前的排序信息保存在窗口Style中,供以后使用
     ws |= nSortOrder;
     SetWindowLong(lc->m_hWnd, GWL_STYLE, ws); // 将各ITEM的LPARAM用新排序列的内容替换
     LVITEM li;
     li.mask = LVIF_PARAM|LVIF_TEXT;
     TCHAR szItemText[1024];
     for (int i = 0; i < lc.GetItemCount(); i++)
     {
      li.iItem = i;
      li.iSubItem = ncurSortCol;
      li.cchTextMax = 1024;
      li.pszText = szItemText;
      lc->GetItem(&li);
      TCHAR * szlparam = (TCHAR *)li.lParam;
      if (szlparam != NULL)
       // 删除以前的信息,释放空间
       // 添加List Item时应注意将lParam初始化NULL
       delete szlparam;
      
      // 复制当前列的szItemText到Item的lParam中
      szlparam = new TCHAR[lstrlen(szItemText) + 1];
      lstrcpy(szlparam, szItemText);
      lc.SetItemData(i, szlparam);
     } // 开始排序
    GetListCtrl().SortItems(ListViewCompareFunc,(LPARAM) (&nSortOrder)); *pResult = 0;
    }阿文
    http://tulipstudio.yeah.net