# SQL 基礎

<http://www.postgresqltutorial.com/>

## 不選空值 NOT NULL AND EMPTY

```sql
SELECT device_token FROM users WHERE device_token IS NOT NULL AND device_token != '';
```

## 選不重複

```sql
SELECT DISTINCT ON device_token FROM users;
```

## Data type

<https://www.w3schools.com/sql/sql_datatypes.asp>

## 設定column預設值

```
ALTER TABLE <Table名稱> ALTER COLUMN <column名稱> SET DEFAULT <預設值>;
```

或是建表時設定

{% embed url="<https://www.postgresql.org/docs/9.3/static/ddl-default.html>" %}

## 使用變數

可以使用 with as，後面一定要接著 select 語句

{% embed url="<https://stackoverflow.com/a/18010878/4622645>" %}

## 地理位置距離計算

{% embed url="<https://www.postgresql.org/docs/current/earthdistance.html>" %}

> 也可參考 postGIS <https://postgis.net/>

```sql
// 先在 SQL 輸入以下，安裝插件
create extension cube;
create extension earthdistance;

// 搜尋範例
select (point(-0.1277,51.5073) <@> point(-74.006,40.7144)) as distance;
```

<https://stackoverflow.com/a/25140461>

![記得使用 point 類型](https://2356031413-files.gitbook.io/~/files/v0/b/gitbook-legacy-files/o/assets%2F-M0u4DAqfidGmTt67qrG%2F-MYJNSPJ0rvJTwoS9HNM%2F-MYJQ1njNMGAF6F70yOc%2F%E6%88%AA%E5%9C%96%202021-04-15%20%E4%B8%8B%E5%8D%883.25.59.png?alt=media\&token=776f3df8-5042-4f26-962a-b9b79e44f2c6)

搜尋兩人距離範例：

```sql
select (
  point(
    (select latlng from user_account where first_name = 'Manel')
  ) 
<@> 
  point(
    (select latlng from user_account where first_name = 'Ben')
  )
) as distance
```

## 計算與每個人的經緯度的距離

完整範例：

{% embed url="<https://stackoverflow.com/questions/67118067/calculating-distance-then-finding-people-within-specific-range-of-given-latitude/67122368#67122368>" %}

## DISTINCT 與 ORDER BY 一起使用

> DISTINCT ON 只有 postgres 有， MySQL 要使用 group by，另外 postgres 的 group by 不能單獨選擇特定對象，必須要包含在 select 裡面

```
SELECT *
FROM  (
    SELECT DISTINCT ON (address_id) *
    FROM   purchases
    WHERE  product_id = 1
    ) p
ORDER  BY purchased_at DESC;
```

<https://stackoverflow.com/a/9796104/4622645>

## DISTINCT ON 並選擇要從重複的選擇哪些

如果 A 有多筆資料，想要 DISTINCT A 中最新的，之後把選出來的 A, B, C 再依照時間排序

```sql
SELECT *
    FROM  (
        SELECT DISTINCT ON (customer_name) *
        FROM orders
        ORDER BY customer_name, create_time DESC
        ) p
    ORDER BY create_time DESC;
```

## &#x20;使用 TRANSACTION

<https://stackoverflow.com/questions/43351168/node-postgres-transactions-with-callbacks-or-async-await>
