[Spring cloud 一步步實現廣告系統] 17. 根據流量類型查詢廣告

  • 2019 年 10 月 3 日
  • 筆記

廣告檢索服務

功能介紹

媒體方(手機APP打開的展示廣告,走在路上看到的大屏幕廣告等等)

請求數據對象實現

從上圖我們可以看出,在媒體方向我們的廣告檢索系統發起請求的時候,請求中會有很多的請求參數信息,他們分為了三個部分,我們來編碼封裝這幾個參數對象信息以及我們請求本身的信息。Let’s code.

  • 創建廣告檢索請求接口
  /**   * ISearch for 請求接口,   * 根據廣告請求對象,獲取廣告響應信息   *   * @author <a href="mailto:[email protected]">Isaac.Zhang | 若初</a>   */  @FunctionalInterface  public interface ISearch {        /**       * 根據請求返回廣告結果       */      SearchResponse fetchAds(SearchRequest request);  }
  • 創建SearchRequest,包含三部分:mediaId,RequestInfo,FeatureInfo
@Data  @NoArgsConstructor  @AllArgsConstructor  public class SearchRequest {        //媒體方請求標示      private String mediaId;      //請求基本信息      private RequestInfo requestInfo;      //匹配信息      private FeatureInfo featureInfo;          @Data      @NoArgsConstructor      @AllArgsConstructor      public static class RequestInfo {          private String requestId;            private List<AdSlot> adSlots;          private App app;          private Geo geo;          private Device device;      }        @Data      @NoArgsConstructor      @AllArgsConstructor      public static class FeatureInfo {            private KeywordFeature keywordFeature;          private DistrictFeature districtFeature;          private HobbyFeatrue hobbyFeatrue;            private FeatureRelation relation = FeatureRelation.AND;      }  }

其他的對象大家可以去github傳送門 & gitee傳送門 下載源碼。

UTOOLS1565403569539.png

檢索響應對象定義
/**   * SearchResponse for 檢索API響應對象   *   * @author <a href="mailto:[email protected]">Isaac.Zhang | 若初</a>   */  @Data  @Builder  @NoArgsConstructor  @AllArgsConstructor  public class SearchResponse {        //一個廣告位,可以展示多個廣告      //Map key為廣告位 AdSlot#adSlotCode      public Map<String, List<Creative>> adSlotRelationAds = new HashMap<>();        @Data      @Builder      @NoArgsConstructor      @AllArgsConstructor      public static class Creative {            private Long adId;          private String adUrl;          private Integer width;          private Integer height;          private Integer type;          private Integer materialType;            //展示監控url          private List<String> showMonitorUrl = Arrays.asList("www.life-runner.com", "www.babydy.cn");          //點擊監控url          private List<String> clickMonitorUrl = Arrays.asList("www.life-runner.com", "www.babydy.cn");      }        /**       * 我們的檢索服務針對的是內存中的索引檢索,那麼我們就需要一個轉換方法       */      public static Creative convert(CreativeIndexObject object) {            return Creative.builder()                         .adId(object.getAdId())                         .adUrl(object.getAdUrl())                         .width(object.getWidth())                         .height(object.getHeight())                         .type(object.getType())                         .materialType(object.getMaterialType())                         .build();      }  }
根據流量類型廣告過濾

流量類型本身屬於推廣單元下的類目,有很多種類貼片廣告,開屏廣告等等,這些類型需要同步到媒體方,媒體方會根據不同的流量類型發起不同的廣告請求,我們需要先定義一個流量類型的信息類。

public class AdUnitConstants {      public static class PositionType{          //App啟動時展示的、展示時間短暫的全屏化廣告形式。          private static final int KAIPING = 1;          //電影開始之前的廣告          private static final int TIEPIAN = 2;          //電影播放中途廣告          private static final int TIEPIAN_MIDDLE = 4;          //暫停視頻時候播放的廣告          private static final int TIEPIAN_PAUSE = 8;          //視頻播放完          private static final int TIEPIAN_POST = 16;      }  }

從上述類型的數字,我們可以看出是2的倍數,這是為了使用位運算提升性能。

com.sxzhongf.ad.index.adunit.AdUnitIndexObject中,我們添加類型校驗方法:

public static boolean isAdSlotType(int adSlotType, int positionType) {          switch (adSlotType) {              case AdUnitConstants.PositionType.KAIPING:                  return isKaiPing(positionType);              case AdUnitConstants.PositionType.TIEPIAN:                  return isTiePian(positionType);              case AdUnitConstants.PositionType.TIEPIAN_MIDDLE:                  return isTiePianMiddle(positionType);              case AdUnitConstants.PositionType.TIEPIAN_PAUSE:                  return isTiePianPause(positionType);              case AdUnitConstants.PositionType.TIEPIAN_POST:                  return isTiePianPost(positionType);              default:                  return false;          }      }        /**       * 與運算,低位取等,高位補零。       * 如果 > 0,則為開屏       */      private static boolean isKaiPing(int positionType) {          return (positionType & AdUnitConstants.PositionType.KAIPING) > 0;      }      private static boolean isTiePianMiddle(int positionType) {          return (positionType & AdUnitConstants.PositionType.TIEPIAN_MIDDLE) > 0;      }        private static boolean isTiePianPause(int positionType) {          return (positionType & AdUnitConstants.PositionType.TIEPIAN_PAUSE) > 0;      }        private static boolean isTiePianPost(int positionType) {          return (positionType & AdUnitConstants.PositionType.TIEPIAN_POST) > 0;      }        private static boolean isTiePian(int positionType) {          return (positionType & AdUnitConstants.PositionType.TIEPIAN) > 0;      }

無所如何,我們都是需要根據positionType進行數據查詢過濾,我們在之前的com.sxzhongf.ad.index.adunit.AdUnitIndexAwareImpl中添加2個方法來實現過濾:

/**       * 過濾當前是否存在滿足positionType的UnitIds       */      public Set<Long> match(Integer positionType) {          Set<Long> adUnitIds = new HashSet<>();          objectMap.forEach((k, v) -> {              if (AdUnitIndexObject.isAdSlotType(positionType, v.getPositionType())) {                  adUnitIds.add(k);              }          });          return adUnitIds;      }        /**       * 根據UnitIds查詢AdUnit list       */      public List<AdUnitIndexObject> fetch(Collection<Long> adUnitIds) {          if (CollectionUtils.isEmpty(adUnitIds)) {              return Collections.EMPTY_LIST;          }          List<AdUnitIndexObject> result = new ArrayList<>();          adUnitIds.forEach(id -> {              AdUnitIndexObject object = get(id);              if (null == object) {                  log.error("AdUnitIndexObject does not found:{}", id);                  return;              }              result.add(object);          });            return result;      }
  • 實現Search服務接口

上述我們準備了一系列的查詢方法,都是為了根據流量類型查詢廣告單元信息,我們現在開始實現我們的查詢接口,查詢接口中,我們可以獲取到媒體方的請求對象信息,它帶有一系列查詢所需要的過濾參數:

/**   * SearchImpl for 實現search 服務   *   * @author <a href="mailto:[email protected]">Isaac.Zhang | 若初</a>   */  @Service  @Slf4j  public class SearchImpl implements ISearch {      @Override      public SearchResponse fetchAds(SearchRequest request) {            //獲取請求廣告位信息          List<AdSlot> adSlotList = request.getRequestInfo().getAdSlots();            //獲取三個Feature信息          KeywordFeature keywordFeature = request.getFeatureInfo().getKeywordFeature();          HobbyFeatrue hobbyFeatrue = request.getFeatureInfo().getHobbyFeatrue();          DistrictFeature districtFeature = request.getFeatureInfo().getDistrictFeature();          //Feature關係          FeatureRelation featureRelation = request.getFeatureInfo().getRelation();              //構造響應對象          SearchResponse response = new SearchResponse();          Map<String, List<SearchResponse.Creative>> adSlotRelationAds = response.getAdSlotRelationAds();            for (AdSlot adSlot : adSlotList) {              Set<Long> targetUnitIdSet;              //根據流量類型從緩存中獲取 初始 廣告信息              Set<Long> adUnitIdSet = IndexDataTableUtils.of(                      AdUnitIndexAwareImpl.class              ).match(adSlot.getPositionType());          }          return null;      }  }