【微信公眾號開發】【13】批量導出公眾號所有用戶信息到Excel


前言:

1,一次拉取調用最多拉取10000個關注者的OpenID,當公眾號關注者數量超過10000時,可通過填寫next_openid的值,從而多次拉取列表的方式來滿足需求

2,獲取OpenID列表后,要批量獲取用戶微信信息,最多支持一次拉取100條

3,處理Excel的jar包用的是jxl

4,導出過程中不要輸出內容到控制台(System.out.println,我開始是輸出了分頁頁碼)。會導致運行時間延長很多;而且控制台會遺漏打印,讓人以為有部分數據沒有獲取到,但其實導出到Excel表的數據是沒有問題的

正文:

1,獲取所有的OpenID

//獲取用戶列表
public static String USER_LIST_URL = "https://api.weixin.qq.com/cgi-bin/user/get?access_token=%s&next_openid=%s";
//獲取所有的openId
public static List<String> getAllOpenId(String appId)
{
    List<String> openIds = getOpenIds("", appId);
    logger.info(String.format("公眾號人數:%s", openIds.size()));
    return openIds;
}
  
private static List<String> getOpenIds(String next_openid, String appId)
{
    List<String> openIdList=new ArrayList<String>();
      
    String accessToken = WeixinUtil.getAccessToken(appId);
    String requestUrl = String.format(WxConfig.USER_LIST_URL, accessToken, next_openid); //拼接請求地址          
    JSONObject jsonObject = HttpRequest.httpRequest(requestUrl, "GET", null); //獲取用戶列表
    
    if (null != jsonObject) {
        try {
            next_openid = jsonObject.getString("next_openid");
            int count = jsonObject.getInt("count");             
            JSONObject openIdObject = jsonObject.getJSONObject("data");
            
            if (count > 0) {
                JSONArray openids = openIdObject.getJSONArray("openid");
                List<String> temp_openIdList = JSONArray.toList(openids, new String(), new JsonConfig());    
                openIdList.addAll(temp_openIdList);
            }
            
            if (StringUtils.isNotEmpty(next_openid)) {
                List<String> list = getOpenIds(next_openid, appId);
                openIdList.addAll(list);
            }            
        } catch (Exception e) {
            logger.info(String.format("獲取用戶列表失敗 errcode:%s;errmsg:%s", jsonObject.getInt("errcode"), jsonObject.getString("errmsg")));
        }
    }
    return openIdList;
}

2,根據openId列表獲取用戶信息

//批量獲取用戶基本信息
public static String USER_LIST_INFO_URL = "https://api.weixin.qq.com/cgi-bin/user/info/batchget?access_token=%s";
public static List<WxUserAll> getUserInfo(List<String> userOpenids, String appId)
{
    List<WxUserAll> user_info_list = new ArrayList<WxUserAll>();
            
    String accessToken = WeixinUtil.getAccessToken(appId);
    String requestUrl = String.format(WxConfig.USER_LIST_INFO_URL, accessToken); //拼接請求地址
    
    int total = userOpenids.size();
    int temp = 100;  //一次獲取100
    int page = 0;  //當前頁面
    int count = 0;  //總共獲取多少次
    int index = 0;
    
    if(total != 0){
        if(total > temp){
            count = total/100 + ((total % 100 > 0) ? 1 : 0);
        }else{
            count = 1;
        }
        
        while (page < count) {
            System.out.println("page="+page);
            
            List<Map> userList = new ArrayList<Map>();
            index = (temp*(page+1)) > total ? total : (temp*(page+1));

            for (int i = page*temp; i <index; i++)
            {
                String openid = userOpenids.get(i);
                Map tUserMap = new HashMap<String, String>();
                tUserMap.put("openid", openid);
                tUserMap.put("lang", "zh_CN");
                userList.add(tUserMap);
            }
            Map requestMap = new HashMap<String, List>();
            requestMap.put("user_list", userList);
            String tUserJSON = JSONObject.fromObject(requestMap).toString();
            
            if(StringUtils.isNotEmpty(tUserJSON)){
                JSONObject jsonObject = HttpRequest.httpRequest(requestUrl, "POST", tUserJSON); //獲取用戶列表
                if(jsonObject != null){
                    JSONArray user_list = jsonObject.getJSONArray("user_info_list");
                    List<WxUserAll> temp_user_info_list = JSONArray.toList(user_list, new WxUserAll(), new JsonConfig());    
                    user_info_list.addAll(temp_user_info_list);
                }
                page++;    
            }else{
                break;
            }                    
        }
    }    

    return user_info_list;
}

3,jxl,jar包

<dependency>
    <groupId>net.sourceforge.jexcelapi</groupId>
    <artifactId>jxl</artifactId>
    <version>2.6.12</version>
</dependency>

4,導出數據到Excel,字段自己選擇要哪些

import java.io.File;
import java.util.Date;
import java.util.List;

import org.apache.commons.lang.StringUtils;

import com.bf.base.utils.CalendarUtil;
import com.bf.wxChange.params.WxUserAll;

import jxl.Workbook;
import jxl.write.Label;
import jxl.write.WritableCellFormat;
import jxl.write.WritableFont;
import jxl.write.WritableSheet;
import jxl.write.WritableWorkbook;

public class SaveToExcel {
    public static File saveToExcel(List<WxUserAll> userInfos){
        File file=null;
        try {
          WritableWorkbook wwb = null;

          String fileName = "wxList.xls"; //創建可寫入的Excel工作簿
          file = new File(fileName); //以fileName為文件名來創建一個Workbook          
          wwb = Workbook.createWorkbook(file);

          // 創建工作表
          WritableSheet ws = wwb.createSheet("wxList", 0);
          ws.setColumnView(0,35);
          ws.setColumnView(1,20);
          ws.setColumnView(2,20);
          ws.setColumnView(3,35);
          ws.setColumnView(4,15);
          ws.setColumnView(5,35);
          
          ws.mergeCells(0,0,12,0); //合並第一列第一行到第七列第一行的所有單元格 
          WritableFont font1= new WritableFont(WritableFont.TIMES,16,WritableFont.BOLD); 
          WritableCellFormat format1=new WritableCellFormat(font1); 
          format1.setAlignment(jxl.format.Alignment.CENTRE);
          Label top= new Label(0, 0, "all user info",format1);
          ws.addCell(top);
          
          //要插入到的Excel表格的行號,默認從0開始
          Label label0 = new Label(0, 1, "wxId");
          Label label1 = new Label(1, 1, "nickname");
          Label label2 = new Label(2, 1, "createTime");
          Label label3 = new Label(3, 1, "avatar");
          Label label4 = new Label(4, 1, "subscribe");
          Label label5 = new Label(5,1, "wxUnionId");
          ws.addCell(label0);
          ws.addCell(label1);
          ws.addCell(label2);
          ws.addCell(label3);
          ws.addCell(label4);
          ws.addCell(label5);

          for (int i = 0; i < userInfos.size(); i++) {
              Label user0 = new Label(0, i+2, userInfos.get(i).getOpenid());
              Label user1 = new Label(1, i+2, userInfos.get(i).getNickname());
              String subscribeTime = userInfos.get(i).getSubscribe_time();
              if (StringUtils.isNotEmpty(subscribeTime)) {
                  subscribeTime = CalendarUtil.DtoS(new Date(Long.parseLong(subscribeTime) * 1000L));
              }
              Label user2 = new Label(2, i+2, subscribeTime);
              Label user3 = new Label(3, i+2, userInfos.get(i).getHeadimgurl());
              Label user4 = new Label(4, i+2, userInfos.get(i).getSubscribe() == 0 ? "0" : "1");
              Label user5 = new Label(5, i+2, userInfos.get(i).getUnionid());

              ws.addCell(user0);
              ws.addCell(user1);
              ws.addCell(user2);
              ws.addCell(user3);
              ws.addCell(user4);
              ws.addCell(user5);             
          }

          wwb.write(); //寫進文檔          
          wwb.close(); //關閉Excel工作簿對象
        } catch (Exception e) {
          // TODO Auto-generated catch block
          e.printStackTrace();
        }
        return file;
    }
}

 

5,測試類

@RunWith(SpringRunner.class)
@SpringBootTest
public class SaveToExcelTest {
    
    private static String APP_ID = Config.JOINT_CARE_APP_NAME;

    @Test
    public void ExcelTest() {
        UserUtil.getAllOpenId(APP_ID);
    }
    
    @Test
    public void AllExcelTest() {
        List<WxUserAll> allUserInfo = UserUtil.getUserInfo(UserUtil.getAllOpenId(APP_ID), APP_ID);
        
        if (!allUserInfo.isEmpty()) {
            SaveToExcel.saveToExcel(allUserInfo);
        }else {
          System.out.println("目前暫無用戶...");
        }
    }    
}

6,其他代碼

WxUserAll

public class WxUserAll {
    private Integer subscribe;
    
    private String openid;
    
    private String nickname;
    
    private Integer sex;
    
    private String language;
    
    private String city;
    
    private String province;
    
    private String country;
    
    private String headimgurl;
    
    private String subscribe_time;
    
    private String unionid;
    
    private String remark;
    
    private Integer groupid;
     
    private List<String> tagid_list;
    
    private String subscribe_scene;
    
    private Integer qr_scene;
    
    private String qr_scene_str;

    @Override
    public String toString() {
        return "WxUserAll [subscribe=" + subscribe + ", openid=" + openid + ", nickname=" + nickname + ", sex=" + sex
                + ", language=" + language + ", city=" + city + ", province=" + province + ", country=" + country
                + ", headimgurl=" + headimgurl + ", subscribe_time=" + subscribe_time + ", unionid=" + unionid
                + ", remark=" + remark + ", groupid=" + groupid + ", tagid_list=" + tagid_list + ", subscribe_scene="
                + subscribe_scene + ", qr_scene=" + qr_scene + ", qr_scene_str=" + qr_scene_str + "]";
    }

    public Integer getSubscribe() {
        return subscribe;
    }

    public void setSubscribe(Integer subscribe) {
        this.subscribe = subscribe;
    }

    public String getOpenid() {
        return openid;
    }

    public void setOpenid(String openid) {
        this.openid = openid;
    }

    public String getNickname() {
        return nickname;
    }

    public void setNickname(String nickname) {
        this.nickname = nickname;
    }

    public Integer getSex() {
        return sex;
    }

    public void setSex(Integer sex) {
        this.sex = sex;
    }

    public String getLanguage() {
        return language;
    }

    public void setLanguage(String language) {
        this.language = language;
    }

    public String getCity() {
        return city;
    }

    public void setCity(String city) {
        this.city = city;
    }

    public String getProvince() {
        return province;
    }

    public void setProvince(String province) {
        this.province = province;
    }

    public String getCountry() {
        return country;
    }

    public void setCountry(String country) {
        this.country = country;
    }

    public String getHeadimgurl() {
        return headimgurl;
    }

    public void setHeadimgurl(String headimgurl) {
        this.headimgurl = headimgurl;
    }

    public String getSubscribe_time() {
        return subscribe_time;
    }

    public void setSubscribe_time(String subscribe_time) {
        this.subscribe_time = subscribe_time;
    }

    public String getUnionid() {
        return unionid;
    }

    public void setUnionid(String unionid) {
        this.unionid = unionid;
    }

    public String getRemark() {
        return remark;
    }

    public void setRemark(String remark) {
        this.remark = remark;
    }

    public Integer getGroupid() {
        return groupid;
    }

    public void setGroupid(Integer groupid) {
        this.groupid = groupid;
    }

    public List<String> getTagid_list() {
        return tagid_list;
    }

    public void setTagid_list(List<String> tagid_list) {
        this.tagid_list = tagid_list;
    }

    public String getSubscribe_scene() {
        return subscribe_scene;
    }

    public void setSubscribe_scene(String subscribe_scene) {
        this.subscribe_scene = subscribe_scene;
    }

    public Integer getQr_scene() {
        return qr_scene;
    }

    public void setQr_scene(Integer qr_scene) {
        this.qr_scene = qr_scene;
    }

    public String getQr_scene_str() {
        return qr_scene_str;
    }

    public void setQr_scene_str(String qr_scene_str) {
        this.qr_scene_str = qr_scene_str;
    }    
}

參考博客:

1,微信公眾號開發之如何一鍵導出微信所有用戶信息到Excel - 酷玩時刻 - 博客園 https://www.cnblogs.com/zyw-205520/p/5958560.html


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM